sing/common/bufio/fallback.go
2023-12-06 19:01:05 +08:00

104 lines
2.3 KiB
Go

package bufio
import (
"net"
"github.com/sagernet/sing/common"
"github.com/sagernet/sing/common/buf"
M "github.com/sagernet/sing/common/metadata"
N "github.com/sagernet/sing/common/network"
)
var _ N.NetPacketConn = (*FallbackPacketConn)(nil)
type FallbackPacketConn struct {
N.PacketConn
writer N.NetPacketWriter
}
func NewNetPacketConn(conn N.PacketConn) N.NetPacketConn {
if packetConn, loaded := conn.(N.NetPacketConn); loaded {
return packetConn
}
return &FallbackPacketConn{
PacketConn: conn,
writer: NewNetPacketWriter(conn),
}
}
func (c *FallbackPacketConn) ReadFrom(p []byte) (n int, addr net.Addr, err error) {
buffer := buf.With(p)
destination, err := c.ReadPacket(buffer)
if err != nil {
return
}
n = buffer.Len()
if buffer.Start() > 0 {
copy(p, buffer.Bytes())
}
addr = destination.UDPAddr()
return
}
func (c *FallbackPacketConn) WriteTo(p []byte, addr net.Addr) (n int, err error) {
return c.writer.WriteTo(p, addr)
}
func (c *FallbackPacketConn) ReaderReplaceable() bool {
return true
}
func (c *FallbackPacketConn) WriterReplaceable() bool {
return true
}
func (c *FallbackPacketConn) Upstream() any {
return c.PacketConn
}
func (c *FallbackPacketConn) UpstreamWriter() any {
return c.writer
}
var _ N.NetPacketWriter = (*FallbackPacketWriter)(nil)
type FallbackPacketWriter struct {
N.PacketWriter
frontHeadroom int
rearHeadroom int
}
func NewNetPacketWriter(writer N.PacketWriter) N.NetPacketWriter {
if packetWriter, loaded := writer.(N.NetPacketWriter); loaded {
return packetWriter
}
return &FallbackPacketWriter{
PacketWriter: writer,
frontHeadroom: N.CalculateFrontHeadroom(writer),
rearHeadroom: N.CalculateRearHeadroom(writer),
}
}
func (c *FallbackPacketWriter) WriteTo(p []byte, addr net.Addr) (n int, err error) {
if c.frontHeadroom > 0 || c.rearHeadroom > 0 {
buffer := buf.NewSize(len(p) + c.frontHeadroom + c.rearHeadroom)
buffer.Resize(c.frontHeadroom, 0)
common.Must1(buffer.Write(p))
err = c.PacketWriter.WritePacket(buffer, M.SocksaddrFromNet(addr))
} else {
err = c.PacketWriter.WritePacket(buf.As(p), M.SocksaddrFromNet(addr))
}
if err != nil {
return
}
n = len(p)
return
}
func (c *FallbackPacketWriter) WriterReplaceable() bool {
return true
}
func (c *FallbackPacketWriter) Upstream() any {
return c.PacketWriter
}