mirror of
https://github.com/SagerNet/sing-tun.git
synced 2025-04-02 19:37:40 +03:00
195 lines
6.1 KiB
Go
195 lines
6.1 KiB
Go
// Copyright 2018 The gVisor Authors.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package header
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"math"
|
|
|
|
"github.com/sagernet/sing-tun/internal/gtcpip"
|
|
"github.com/sagernet/sing-tun/internal/gtcpip/checksum"
|
|
)
|
|
|
|
const (
|
|
udpSrcPort = 0
|
|
udpDstPort = 2
|
|
udpLength = 4
|
|
udpChecksum = 6
|
|
)
|
|
|
|
const (
|
|
// UDPMaximumPacketSize is the largest possible UDP packet.
|
|
UDPMaximumPacketSize = 0xffff
|
|
)
|
|
|
|
// UDPFields contains the fields of a UDP packet. It is used to describe the
|
|
// fields of a packet that needs to be encoded.
|
|
type UDPFields struct {
|
|
// SrcPort is the "source port" field of a UDP packet.
|
|
SrcPort uint16
|
|
|
|
// DstPort is the "destination port" field of a UDP packet.
|
|
DstPort uint16
|
|
|
|
// Length is the "length" field of a UDP packet.
|
|
Length uint16
|
|
|
|
// Checksum is the "checksum" field of a UDP packet.
|
|
Checksum uint16
|
|
}
|
|
|
|
// UDP represents a UDP header stored in a byte array.
|
|
type UDP []byte
|
|
|
|
const (
|
|
// UDPMinimumSize is the minimum size of a valid UDP packet.
|
|
UDPMinimumSize = 8
|
|
|
|
// UDPMaximumSize is the maximum size of a valid UDP packet. The length field
|
|
// in the UDP header is 16 bits as per RFC 768.
|
|
UDPMaximumSize = math.MaxUint16
|
|
|
|
// UDPProtocolNumber is UDP's transport protocol number.
|
|
UDPProtocolNumber tcpip.TransportProtocolNumber = 17
|
|
)
|
|
|
|
// SourcePort returns the "source port" field of the UDP header.
|
|
func (b UDP) SourcePort() uint16 {
|
|
return binary.BigEndian.Uint16(b[udpSrcPort:])
|
|
}
|
|
|
|
// DestinationPort returns the "destination port" field of the UDP header.
|
|
func (b UDP) DestinationPort() uint16 {
|
|
return binary.BigEndian.Uint16(b[udpDstPort:])
|
|
}
|
|
|
|
// Length returns the "length" field of the UDP header.
|
|
func (b UDP) Length() uint16 {
|
|
return binary.BigEndian.Uint16(b[udpLength:])
|
|
}
|
|
|
|
// Payload returns the data contained in the UDP datagram.
|
|
func (b UDP) Payload() []byte {
|
|
return b[UDPMinimumSize:]
|
|
}
|
|
|
|
// Checksum returns the "checksum" field of the UDP header.
|
|
func (b UDP) Checksum() uint16 {
|
|
return binary.BigEndian.Uint16(b[udpChecksum:])
|
|
}
|
|
|
|
// SetSourcePort sets the "source port" field of the UDP header.
|
|
func (b UDP) SetSourcePort(port uint16) {
|
|
binary.BigEndian.PutUint16(b[udpSrcPort:], port)
|
|
}
|
|
|
|
// SetDestinationPort sets the "destination port" field of the UDP header.
|
|
func (b UDP) SetDestinationPort(port uint16) {
|
|
binary.BigEndian.PutUint16(b[udpDstPort:], port)
|
|
}
|
|
|
|
// SetChecksum sets the "checksum" field of the UDP header.
|
|
func (b UDP) SetChecksum(xsum uint16) {
|
|
checksum.Put(b[udpChecksum:], xsum)
|
|
}
|
|
|
|
// SetLength sets the "length" field of the UDP header.
|
|
func (b UDP) SetLength(length uint16) {
|
|
binary.BigEndian.PutUint16(b[udpLength:], length)
|
|
}
|
|
|
|
// CalculateChecksum calculates the checksum of the UDP packet, given the
|
|
// checksum of the network-layer pseudo-header and the checksum of the payload.
|
|
func (b UDP) CalculateChecksum(partialChecksum uint16) uint16 {
|
|
// Calculate the rest of the checksum.
|
|
return checksum.Checksum(b[:UDPMinimumSize], partialChecksum)
|
|
}
|
|
|
|
// IsChecksumValid returns true iff the UDP header's checksum is valid.
|
|
func (b UDP) IsChecksumValid(src, dst tcpip.Address, payloadChecksum uint16) bool {
|
|
xsum := PseudoHeaderChecksum(UDPProtocolNumber, dst.AsSlice(), src.AsSlice(), b.Length())
|
|
xsum = checksum.Combine(xsum, payloadChecksum)
|
|
return b.CalculateChecksum(xsum) == 0xffff
|
|
}
|
|
|
|
// Encode encodes all the fields of the UDP header.
|
|
func (b UDP) Encode(u *UDPFields) {
|
|
b.SetSourcePort(u.SrcPort)
|
|
b.SetDestinationPort(u.DstPort)
|
|
b.SetLength(u.Length)
|
|
b.SetChecksum(u.Checksum)
|
|
}
|
|
|
|
// SetSourcePortWithChecksumUpdate implements ChecksummableTransport.
|
|
func (b UDP) SetSourcePortWithChecksumUpdate(new uint16) {
|
|
old := b.SourcePort()
|
|
b.SetSourcePort(new)
|
|
b.SetChecksum(^checksumUpdate2ByteAlignedUint16(^b.Checksum(), old, new))
|
|
}
|
|
|
|
// SetDestinationPortWithChecksumUpdate implements ChecksummableTransport.
|
|
func (b UDP) SetDestinationPortWithChecksumUpdate(new uint16) {
|
|
old := b.DestinationPort()
|
|
b.SetDestinationPort(new)
|
|
b.SetChecksum(^checksumUpdate2ByteAlignedUint16(^b.Checksum(), old, new))
|
|
}
|
|
|
|
// UpdateChecksumPseudoHeaderAddress implements ChecksummableTransport.
|
|
func (b UDP) UpdateChecksumPseudoHeaderAddress(old, new tcpip.Address, fullChecksum bool) {
|
|
xsum := b.Checksum()
|
|
if fullChecksum {
|
|
xsum = ^xsum
|
|
}
|
|
|
|
xsum = checksumUpdate2ByteAlignedAddress(xsum, old, new)
|
|
if fullChecksum {
|
|
xsum = ^xsum
|
|
}
|
|
|
|
b.SetChecksum(xsum)
|
|
}
|
|
|
|
// UDPValid returns true if the pkt has a valid UDP header. It checks whether:
|
|
// - The length field is too small.
|
|
// - The length field is too large.
|
|
// - The checksum is invalid.
|
|
//
|
|
// UDPValid corresponds to net/netfilter/nf_conntrack_proto_udp.c:udp_error.
|
|
func UDPValid(hdr UDP, payloadChecksum func() uint16, payloadSize uint16, netProto tcpip.NetworkProtocolNumber, srcAddr, dstAddr tcpip.Address, skipChecksumValidation bool) (lengthValid, csumValid bool) {
|
|
if length := hdr.Length(); length > payloadSize+UDPMinimumSize || length < UDPMinimumSize {
|
|
return false, false
|
|
}
|
|
|
|
if skipChecksumValidation {
|
|
return true, true
|
|
}
|
|
|
|
// On IPv4, UDP checksum is optional, and a zero value means the transmitter
|
|
// omitted the checksum generation, as per RFC 768:
|
|
//
|
|
// An all zero transmitted checksum value means that the transmitter
|
|
// generated no checksum (for debugging or for higher level protocols that
|
|
// don't care).
|
|
//
|
|
// On IPv6, UDP checksum is not optional, as per RFC 2460 Section 8.1:
|
|
//
|
|
// Unlike IPv4, when UDP packets are originated by an IPv6 node, the UDP
|
|
// checksum is not optional.
|
|
if netProto == IPv4ProtocolNumber && hdr.Checksum() == 0 {
|
|
return true, true
|
|
}
|
|
|
|
return true, hdr.IsChecksumValid(srcAddr, dstAddr, payloadChecksum())
|
|
}
|