mirror of
https://github.com/refraction-networking/uquic.git
synced 2025-04-03 20:27:35 +03:00
uTLS is not yet bumped to the new version, so this commit breaks the dependencies relationship by getting rid of the local replace.
115 lines
3.5 KiB
Go
115 lines
3.5 KiB
Go
package quic
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"net"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
"syscall"
|
|
"time"
|
|
|
|
"github.com/refraction-networking/uquic/internal/protocol"
|
|
"github.com/refraction-networking/uquic/internal/utils"
|
|
)
|
|
|
|
// OOBCapablePacketConn is a connection that allows the reading of ECN bits from the IP header.
|
|
// If the PacketConn passed to Dial or Listen satisfies this interface, quic-go will use it.
|
|
// In this case, ReadMsgUDP() will be used instead of ReadFrom() to read packets.
|
|
type OOBCapablePacketConn interface {
|
|
net.PacketConn
|
|
SyscallConn() (syscall.RawConn, error)
|
|
SetReadBuffer(int) error
|
|
ReadMsgUDP(b, oob []byte) (n, oobn, flags int, addr *net.UDPAddr, err error)
|
|
WriteMsgUDP(b, oob []byte, addr *net.UDPAddr) (n, oobn int, err error)
|
|
}
|
|
|
|
var _ OOBCapablePacketConn = &net.UDPConn{}
|
|
|
|
func wrapConn(pc net.PacketConn) (rawConn, error) {
|
|
if err := setReceiveBuffer(pc); err != nil {
|
|
if !strings.Contains(err.Error(), "use of closed network connection") {
|
|
setBufferWarningOnce.Do(func() {
|
|
if disable, _ := strconv.ParseBool(os.Getenv("QUIC_GO_DISABLE_RECEIVE_BUFFER_WARNING")); disable {
|
|
return
|
|
}
|
|
log.Printf("%s. See https://github.com/refraction-networking/uquic/wiki/UDP-Buffer-Sizes for details.", err)
|
|
})
|
|
}
|
|
}
|
|
if err := setSendBuffer(pc); err != nil {
|
|
if !strings.Contains(err.Error(), "use of closed network connection") {
|
|
setBufferWarningOnce.Do(func() {
|
|
if disable, _ := strconv.ParseBool(os.Getenv("QUIC_GO_DISABLE_RECEIVE_BUFFER_WARNING")); disable {
|
|
return
|
|
}
|
|
log.Printf("%s. See https://github.com/refraction-networking/uquic/wiki/UDP-Buffer-Sizes for details.", err)
|
|
})
|
|
}
|
|
}
|
|
|
|
conn, ok := pc.(interface {
|
|
SyscallConn() (syscall.RawConn, error)
|
|
})
|
|
var supportsDF bool
|
|
if ok {
|
|
rawConn, err := conn.SyscallConn()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if _, ok := pc.LocalAddr().(*net.UDPAddr); ok {
|
|
// Only set DF on sockets that we expect to be able to handle that configuration.
|
|
var err error
|
|
supportsDF, err = setDF(rawConn)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
c, ok := pc.(OOBCapablePacketConn)
|
|
if !ok {
|
|
utils.DefaultLogger.Infof("PacketConn is not a net.UDPConn. Disabling optimizations possible on UDP connections.")
|
|
return &basicConn{PacketConn: pc, supportsDF: supportsDF}, nil
|
|
}
|
|
return newConn(c, supportsDF)
|
|
}
|
|
|
|
// The basicConn is the most trivial implementation of a rawConn.
|
|
// It reads a single packet from the underlying net.PacketConn.
|
|
// It is used when
|
|
// * the net.PacketConn is not a OOBCapablePacketConn, and
|
|
// * when the OS doesn't support OOB.
|
|
type basicConn struct {
|
|
net.PacketConn
|
|
supportsDF bool
|
|
}
|
|
|
|
var _ rawConn = &basicConn{}
|
|
|
|
func (c *basicConn) ReadPacket() (receivedPacket, error) {
|
|
buffer := getPacketBuffer()
|
|
// The packet size should not exceed protocol.MaxPacketBufferSize bytes
|
|
// If it does, we only read a truncated packet, which will then end up undecryptable
|
|
buffer.Data = buffer.Data[:protocol.MaxPacketBufferSize]
|
|
n, addr, err := c.PacketConn.ReadFrom(buffer.Data)
|
|
if err != nil {
|
|
return receivedPacket{}, err
|
|
}
|
|
return receivedPacket{
|
|
remoteAddr: addr,
|
|
rcvTime: time.Now(),
|
|
data: buffer.Data[:n],
|
|
buffer: buffer,
|
|
}, nil
|
|
}
|
|
|
|
func (c *basicConn) WritePacket(b []byte, packetSize uint16, addr net.Addr, _ []byte) (n int, err error) {
|
|
if uint16(len(b)) != packetSize {
|
|
panic(fmt.Sprintf("inconsistent length. got: %d. expected %d", packetSize, len(b)))
|
|
}
|
|
return c.PacketConn.WriteTo(b, addr)
|
|
}
|
|
|
|
func (c *basicConn) capabilities() connCapabilities { return connCapabilities{DF: c.supportsDF} }
|