mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-08 09:07:44 +00:00
2d604b3791
For the operations we perform on these types (mostly net6.Contains), this encoding is much faster. Part of #19. name old time/op new time/op delta Filter/icmp4-8 27.5ns ± 1% 28.0ns ± 2% +1.89% (p=0.016 n=5+5) Filter/tcp4_syn_in-8 38.8ns ± 2% 38.3ns ± 1% -1.24% (p=0.024 n=5+5) Filter/tcp4_syn_out-8 27.6ns ±12% 24.6ns ± 1% ~ (p=0.063 n=5+5) Filter/udp4_in-8 71.5ns ± 5% 65.9ns ± 1% -7.94% (p=0.008 n=5+5) Filter/udp4_out-8 132ns ±13% 119ns ± 1% -10.29% (p=0.008 n=5+5) Filter/icmp6-8 169ns ±10% 54ns ± 1% -68.35% (p=0.008 n=5+5) Filter/tcp6_syn_in-8 149ns ± 6% 43ns ± 1% -71.11% (p=0.008 n=5+5) Filter/tcp6_syn_out-8 37.7ns ± 4% 24.3ns ± 3% -35.51% (p=0.008 n=5+5) Filter/udp6_in-8 179ns ± 5% 103ns ± 1% -42.75% (p=0.008 n=5+5) Filter/udp6_out-8 156ns ± 3% 191ns ± 1% +22.54% (p=0.008 n=5+5) Signed-off-by: David Anderson <danderson@tailscale.com>
114 lines
2.9 KiB
Go
114 lines
2.9 KiB
Go
// Copyright (c) 2020 Tailscale Inc & AUTHORS All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
package packet
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"fmt"
|
|
|
|
"inet.af/netaddr"
|
|
)
|
|
|
|
// IP6 is an IPv6 address.
|
|
type IP6 struct {
|
|
Hi, Lo uint64
|
|
}
|
|
|
|
// IP6FromNetaddr converts a netaddr.IP to an IP6. Panics if !ip.Is6.
|
|
func IP6FromNetaddr(ip netaddr.IP) IP6 {
|
|
if !ip.Is6() {
|
|
panic(fmt.Sprintf("IP6FromNetaddr called with non-v6 addr %q", ip))
|
|
}
|
|
b := ip.As16()
|
|
return IP6{binary.BigEndian.Uint64(b[:8]), binary.BigEndian.Uint64(b[8:])}
|
|
}
|
|
|
|
// Netaddr converts ip to a netaddr.IP.
|
|
func (ip IP6) Netaddr() netaddr.IP {
|
|
var b [16]byte
|
|
binary.BigEndian.PutUint64(b[:8], ip.Hi)
|
|
binary.BigEndian.PutUint64(b[8:], ip.Lo)
|
|
return netaddr.IPFrom16(b)
|
|
}
|
|
|
|
func (ip IP6) String() string {
|
|
return ip.Netaddr().String()
|
|
}
|
|
|
|
func (ip IP6) IsMulticast() bool {
|
|
return (ip.Hi >> 56) == 0xFF
|
|
}
|
|
|
|
func (ip IP6) IsLinkLocalUnicast() bool {
|
|
return (ip.Hi >> 48) == 0xFE80
|
|
}
|
|
|
|
// ip6HeaderLength is the length of an IPv6 header with no IP options.
|
|
const ip6HeaderLength = 40
|
|
|
|
// IP6Header represents an IPv6 packet header.
|
|
type IP6Header struct {
|
|
IPProto IPProto
|
|
IPID uint32 // only lower 20 bits used
|
|
SrcIP IP6
|
|
DstIP IP6
|
|
}
|
|
|
|
// Len implements Header.
|
|
func (h IP6Header) Len() int {
|
|
return ip6HeaderLength
|
|
}
|
|
|
|
// Marshal implements Header.
|
|
func (h IP6Header) Marshal(buf []byte) error {
|
|
if len(buf) < h.Len() {
|
|
return errSmallBuffer
|
|
}
|
|
if len(buf) > maxPacketLength {
|
|
return errLargePacket
|
|
}
|
|
|
|
binary.BigEndian.PutUint32(buf[:4], h.IPID&0x000FFFFF)
|
|
buf[0] = 0x60
|
|
binary.BigEndian.PutUint16(buf[4:6], uint16(len(buf)-ip6HeaderLength)) // Total length
|
|
buf[6] = uint8(h.IPProto) // Inner protocol
|
|
buf[7] = 64 // TTL
|
|
binary.BigEndian.PutUint64(buf[8:16], h.SrcIP.Hi)
|
|
binary.BigEndian.PutUint64(buf[16:24], h.SrcIP.Lo)
|
|
binary.BigEndian.PutUint64(buf[24:32], h.DstIP.Hi)
|
|
binary.BigEndian.PutUint64(buf[32:40], h.DstIP.Lo)
|
|
|
|
return nil
|
|
}
|
|
|
|
// ToResponse implements Header.
|
|
func (h *IP6Header) ToResponse() {
|
|
h.SrcIP, h.DstIP = h.DstIP, h.SrcIP
|
|
// Flip the bits in the IPID. If incoming IPIDs are distinct, so are these.
|
|
h.IPID = (^h.IPID) & 0x000FFFFF
|
|
}
|
|
|
|
// marshalPseudo serializes h into buf in the "pseudo-header" form
|
|
// required when calculating UDP checksums.
|
|
func (h IP6Header) marshalPseudo(buf []byte) error {
|
|
if len(buf) < h.Len() {
|
|
return errSmallBuffer
|
|
}
|
|
if len(buf) > maxPacketLength {
|
|
return errLargePacket
|
|
}
|
|
|
|
binary.BigEndian.PutUint64(buf[:8], h.SrcIP.Hi)
|
|
binary.BigEndian.PutUint64(buf[8:16], h.SrcIP.Lo)
|
|
binary.BigEndian.PutUint64(buf[16:24], h.DstIP.Hi)
|
|
binary.BigEndian.PutUint64(buf[24:32], h.DstIP.Lo)
|
|
binary.BigEndian.PutUint32(buf[32:36], uint32(len(buf)-h.Len()))
|
|
buf[36] = 0
|
|
buf[37] = 0
|
|
buf[38] = 0
|
|
buf[39] = 17 // NextProto
|
|
return nil
|
|
}
|