// Copyright (c) Tailscale Inc & AUTHORS // SPDX-License-Identifier: BSD-3-Clause // Package checksum provides functions for updating checksums in parsed packets. package checksum import ( "encoding/binary" "net/netip" "gvisor.dev/gvisor/pkg/tcpip" "gvisor.dev/gvisor/pkg/tcpip/header" "tailscale.com/net/packet" "tailscale.com/types/ipproto" ) // UpdateSrcAddr updates the source address in the packet buffer (e.g. during // SNAT). It also updates the checksum. Currently (2023-09-22) only TCP/UDP/ICMP // is supported. It panics if provided with an address in a different // family to the parsed packet. func UpdateSrcAddr(q *packet.Parsed, src netip.Addr) { if src.Is6() && q.IPVersion != 6 { panic("UpdateSrcAddr: cannot write IPv6 address to v4 packet") } else if src.Is4() && q.IPVersion != 4 { panic("UpdateSrcAddr: cannot write IPv4 address to v6 packet") } q.CaptureMeta.DidSNAT = true q.CaptureMeta.OriginalSrc = q.Src old := q.Src.Addr() q.Src = netip.AddrPortFrom(src, q.Src.Port()) b := q.Buffer() if src.Is6() { v6 := src.As16() copy(b[8:24], v6[:]) updateV6PacketChecksums(q, old, src) } else { v4 := src.As4() copy(b[12:16], v4[:]) updateV4PacketChecksums(q, old, src) } } // UpdateDstAddr updates the destination address in the packet buffer (e.g. during // DNAT). It also updates the checksum. Currently (2022-12-10) only TCP/UDP/ICMP // is supported. It panics if provided with an address in a different // family to the parsed packet. func UpdateDstAddr(q *packet.Parsed, dst netip.Addr) { if dst.Is6() && q.IPVersion != 6 { panic("UpdateDstAddr: cannot write IPv6 address to v4 packet") } else if dst.Is4() && q.IPVersion != 4 { panic("UpdateDstAddr: cannot write IPv4 address to v6 packet") } q.CaptureMeta.DidDNAT = true q.CaptureMeta.OriginalDst = q.Dst old := q.Dst.Addr() q.Dst = netip.AddrPortFrom(dst, q.Dst.Port()) b := q.Buffer() if dst.Is6() { v6 := dst.As16() copy(b[24:36], v6[:]) updateV6PacketChecksums(q, old, dst) } else { v4 := dst.As4() copy(b[16:20], v4[:]) updateV4PacketChecksums(q, old, dst) } } // updateV4PacketChecksums updates the checksums in the packet buffer. // Currently (2023-03-01) only TCP/UDP/ICMP over IPv4 is supported. // p is modified in place. // If p.IPProto is unknown, only the IP header checksum is updated. func updateV4PacketChecksums(p *packet.Parsed, old, new netip.Addr) { if len(p.Buffer()) < 12 { // Not enough space for an IPv4 header. return } o4, n4 := old.As4(), new.As4() // First update the checksum in the IP header. updateV4Checksum(p.Buffer()[10:12], o4[:], n4[:]) // Now update the transport layer checksums, where applicable. tr := p.Transport() switch p.IPProto { case ipproto.UDP, ipproto.DCCP: if len(tr) < header.UDPMinimumSize { // Not enough space for a UDP header. return } updateV4Checksum(tr[6:8], o4[:], n4[:]) case ipproto.TCP: if len(tr) < header.TCPMinimumSize { // Not enough space for a TCP header. return } updateV4Checksum(tr[16:18], o4[:], n4[:]) case ipproto.GRE: if len(tr) < 6 { // Not enough space for a GRE header. return } if tr[0] == 1 { // checksum present updateV4Checksum(tr[4:6], o4[:], n4[:]) } case ipproto.SCTP, ipproto.ICMPv4: // No transport layer update required. } } // updateV6PacketChecksums updates the checksums in the packet buffer. // p is modified in place. // If p.IPProto is unknown, no checksums are updated. func updateV6PacketChecksums(p *packet.Parsed, old, new netip.Addr) { if len(p.Buffer()) < 40 { // Not enough space for an IPv6 header. return } o6, n6 := tcpip.AddrFrom16Slice(old.AsSlice()), tcpip.AddrFrom16Slice(new.AsSlice()) // Now update the transport layer checksums, where applicable. tr := p.Transport() switch p.IPProto { case ipproto.ICMPv6: if len(tr) < header.ICMPv6MinimumSize { return } header.ICMPv6(tr).UpdateChecksumPseudoHeaderAddress(o6, n6) case ipproto.UDP, ipproto.DCCP: if len(tr) < header.UDPMinimumSize { return } header.UDP(tr).UpdateChecksumPseudoHeaderAddress(o6, n6, true) case ipproto.TCP: if len(tr) < header.TCPMinimumSize { return } header.TCP(tr).UpdateChecksumPseudoHeaderAddress(o6, n6, true) case ipproto.SCTP: // No transport layer update required. } } // updateV4Checksum calculates and updates the checksum in the packet buffer for // a change between old and new. The oldSum must point to the 16-bit checksum // field in the packet buffer that holds the old checksum value, it will be // updated in place. // // The old and new must be the same length, and must be an even number of bytes. func updateV4Checksum(oldSum, old, new []byte) { if len(old) != len(new) { panic("old and new must be the same length") } if len(old)%2 != 0 { panic("old and new must be of even length") } /* RFC 1624 Given the following notation: HC - old checksum in header C - one's complement sum of old header HC' - new checksum in header C' - one's complement sum of new header m - old value of a 16-bit field m' - new value of a 16-bit field HC' = ~(C + (-m) + m') -- [Eqn. 3] HC' = ~(~HC + ~m + m') This can be simplified to: HC' = ~(C + ~m + m') -- [Eqn. 3] HC' = ~C' C' = C + ~m + m' */ c := uint32(^binary.BigEndian.Uint16(oldSum)) cPrime := c for len(new) > 0 { mNot := uint32(^binary.BigEndian.Uint16(old[:2])) mPrime := uint32(binary.BigEndian.Uint16(new[:2])) cPrime += mPrime + mNot new, old = new[2:], old[2:] } // Account for overflows by adding the carry bits back into the sum. for (cPrime >> 16) > 0 { cPrime = cPrime&0xFFFF + cPrime>>16 } hcPrime := ^uint16(cPrime) binary.BigEndian.PutUint16(oldSum, hcPrime) }