sing-box/transport/wireguard/client_bind.go

232 lines
5.1 KiB
Go
Raw Normal View History

2022-09-05 16:15:09 +00:00
package wireguard
import (
"context"
"net"
"net/netip"
2022-09-05 16:15:09 +00:00
"sync"
2023-08-07 09:46:51 +00:00
"time"
2022-09-05 16:15:09 +00:00
"github.com/sagernet/sing/common"
"github.com/sagernet/sing/common/bufio"
2023-04-13 08:02:28 +00:00
E "github.com/sagernet/sing/common/exceptions"
2024-11-01 16:39:02 +00:00
"github.com/sagernet/sing/common/logger"
2022-09-05 16:15:09 +00:00
M "github.com/sagernet/sing/common/metadata"
N "github.com/sagernet/sing/common/network"
2024-06-03 08:59:13 +00:00
"github.com/sagernet/sing/service"
"github.com/sagernet/sing/service/pause"
"github.com/sagernet/wireguard-go/conn"
2022-09-05 16:15:09 +00:00
)
var _ conn.Bind = (*ClientBind)(nil)
type ClientBind struct {
ctx context.Context
2024-11-01 16:39:02 +00:00
logger logger.Logger
2024-06-03 08:59:13 +00:00
pauseManager pause.Manager
bindCtx context.Context
bindDone context.CancelFunc
dialer N.Dialer
reservedForEndpoint map[netip.AddrPort][3]uint8
connAccess sync.Mutex
conn *wireConn
done chan struct{}
isConnect bool
connectAddr netip.AddrPort
reserved [3]uint8
2022-09-05 16:15:09 +00:00
}
2024-11-01 16:39:02 +00:00
func NewClientBind(ctx context.Context, logger logger.Logger, dialer N.Dialer, isConnect bool, connectAddr netip.AddrPort, reserved [3]uint8) *ClientBind {
2022-09-05 16:15:09 +00:00
return &ClientBind{
ctx: ctx,
2024-11-01 16:39:02 +00:00
logger: logger,
2024-06-03 08:59:13 +00:00
pauseManager: service.FromContext[pause.Manager](ctx),
dialer: dialer,
reservedForEndpoint: make(map[netip.AddrPort][3]uint8),
2024-03-20 02:46:54 +00:00
done: make(chan struct{}),
isConnect: isConnect,
connectAddr: connectAddr,
reserved: reserved,
2022-09-05 16:15:09 +00:00
}
}
func (c *ClientBind) connect() (*wireConn, error) {
serverConn := c.conn
if serverConn != nil {
select {
case <-serverConn.done:
serverConn = nil
default:
return serverConn, nil
}
}
c.connAccess.Lock()
defer c.connAccess.Unlock()
2024-06-03 08:59:13 +00:00
select {
case <-c.done:
return nil, net.ErrClosed
default:
}
2022-09-05 16:15:09 +00:00
serverConn = c.conn
if serverConn != nil {
select {
case <-serverConn.done:
serverConn = nil
default:
return serverConn, nil
}
}
if c.isConnect {
2024-06-03 08:59:13 +00:00
udpConn, err := c.dialer.DialContext(c.bindCtx, N.NetworkUDP, M.SocksaddrFromNetIP(c.connectAddr))
if err != nil {
2023-04-13 08:02:28 +00:00
return nil, err
}
c.conn = &wireConn{
PacketConn: bufio.NewUnbindPacketConn(udpConn),
done: make(chan struct{}),
}
} else {
2024-06-03 08:59:13 +00:00
udpConn, err := c.dialer.ListenPacket(c.bindCtx, M.Socksaddr{Addr: netip.IPv4Unspecified()})
if err != nil {
2023-04-13 08:02:28 +00:00
return nil, err
}
c.conn = &wireConn{
2023-04-13 08:02:28 +00:00
PacketConn: bufio.NewPacketConn(udpConn),
done: make(chan struct{}),
}
2022-09-05 16:15:09 +00:00
}
return c.conn, nil
}
func (c *ClientBind) Open(port uint16) (fns []conn.ReceiveFunc, actualPort uint16, err error) {
2022-09-23 05:14:31 +00:00
select {
case <-c.done:
2024-03-20 02:46:54 +00:00
c.done = make(chan struct{})
2022-09-23 05:14:31 +00:00
default:
}
2024-06-03 08:59:13 +00:00
c.bindCtx, c.bindDone = context.WithCancel(c.ctx)
2022-09-05 16:15:09 +00:00
return []conn.ReceiveFunc{c.receive}, 0, nil
}
2023-04-20 05:16:31 +00:00
func (c *ClientBind) receive(packets [][]byte, sizes []int, eps []conn.Endpoint) (count int, err error) {
2022-09-05 16:15:09 +00:00
udpConn, err := c.connect()
if err != nil {
2023-04-13 08:02:28 +00:00
select {
case <-c.done:
return
default:
}
2024-11-01 16:39:02 +00:00
c.logger.Error(E.Cause(err, "connect to server"))
2023-04-13 08:02:28 +00:00
err = nil
2024-06-03 08:59:13 +00:00
c.pauseManager.WaitActive()
2023-08-07 09:46:51 +00:00
time.Sleep(time.Second)
2022-09-05 16:15:09 +00:00
return
}
2023-04-20 05:16:31 +00:00
n, addr, err := udpConn.ReadFrom(packets[0])
2022-09-05 16:15:09 +00:00
if err != nil {
udpConn.Close()
2022-09-23 05:14:31 +00:00
select {
case <-c.done:
default:
2024-11-01 16:39:02 +00:00
c.logger.Error(context.Background(), E.Cause(err, "read packet"))
2023-04-20 05:16:31 +00:00
err = nil
2022-09-23 05:14:31 +00:00
}
return
2022-09-05 16:15:09 +00:00
}
2023-04-20 05:16:31 +00:00
sizes[0] = n
if n > 3 {
2023-04-20 05:16:31 +00:00
b := packets[0]
common.ClearArray(b[1:4])
}
eps[0] = Endpoint(M.AddrPortFromNet(addr))
2023-04-20 05:16:31 +00:00
count = 1
2022-09-05 16:15:09 +00:00
return
}
func (c *ClientBind) Close() error {
2022-09-23 05:14:31 +00:00
select {
case <-c.done:
default:
close(c.done)
}
2024-06-03 08:59:13 +00:00
if c.bindDone != nil {
c.bindDone()
}
c.connAccess.Lock()
defer c.connAccess.Unlock()
common.Close(common.PtrOrNil(c.conn))
2022-09-05 16:15:09 +00:00
return nil
}
func (c *ClientBind) SetMark(mark uint32) error {
return nil
}
2023-04-20 05:16:31 +00:00
func (c *ClientBind) Send(bufs [][]byte, ep conn.Endpoint) error {
2022-09-05 16:15:09 +00:00
udpConn, err := c.connect()
if err != nil {
2024-06-03 08:59:13 +00:00
c.pauseManager.WaitActive()
time.Sleep(time.Second)
2022-09-05 16:15:09 +00:00
return err
}
destination := netip.AddrPort(ep.(Endpoint))
2023-04-20 05:16:31 +00:00
for _, b := range bufs {
if len(b) > 3 {
reserved, loaded := c.reservedForEndpoint[destination]
if !loaded {
reserved = c.reserved
}
copy(b[1:4], reserved[:])
2023-04-20 05:16:31 +00:00
}
2024-03-20 02:46:54 +00:00
_, err = udpConn.WriteToUDPAddrPort(b, destination)
2023-04-20 05:16:31 +00:00
if err != nil {
udpConn.Close()
return err
}
2022-09-05 16:15:09 +00:00
}
2023-04-20 05:16:31 +00:00
return nil
2022-09-05 16:15:09 +00:00
}
func (c *ClientBind) ParseEndpoint(s string) (conn.Endpoint, error) {
ap, err := netip.ParseAddrPort(s)
if err != nil {
return nil, err
}
return Endpoint(ap), nil
2022-09-05 16:15:09 +00:00
}
2023-04-20 05:16:31 +00:00
func (c *ClientBind) BatchSize() int {
return 1
}
func (c *ClientBind) SetReservedForEndpoint(destination netip.AddrPort, reserved [3]byte) {
c.reservedForEndpoint[destination] = reserved
}
2022-09-05 16:15:09 +00:00
type wireConn struct {
2023-04-13 08:02:28 +00:00
net.PacketConn
2024-03-20 02:46:54 +00:00
conn net.Conn
2022-09-05 16:15:09 +00:00
access sync.Mutex
done chan struct{}
}
2024-03-20 02:46:54 +00:00
func (w *wireConn) WriteToUDPAddrPort(b []byte, addr netip.AddrPort) (int, error) {
if w.conn != nil {
return w.conn.Write(b)
}
return w.PacketConn.WriteTo(b, M.SocksaddrFromNetIP(addr).UDPAddr())
}
2022-09-05 16:15:09 +00:00
func (w *wireConn) Close() error {
w.access.Lock()
defer w.access.Unlock()
select {
case <-w.done:
return net.ErrClosed
default:
}
2023-04-13 08:02:28 +00:00
w.PacketConn.Close()
2022-09-05 16:15:09 +00:00
close(w.done)
return nil
}