/* * SPDX-License-Identifier: AGPL-3.0-only * Copyright (c) 2022-2023, daeuniverse Organization */ package control import ( "fmt" "net/netip" "sync" "time" "github.com/daeuniverse/dae/component/outbound/dialer" "github.com/mzz2017/softwind/netproxy" "github.com/mzz2017/softwind/pool" ) const ( EthernetMtu = 1500 ) type UdpHandler func(data []byte, from netip.AddrPort) error type UdpEndpoint struct { conn netproxy.PacketConn // mu protects deadlineTimer mu sync.Mutex deadlineTimer *time.Timer handler UdpHandler NatTimeout time.Duration Dialer *dialer.Dialer } func (ue *UdpEndpoint) start() { buf := pool.Get(EthernetMtu) buf = buf[:cap(buf)] defer pool.Put(buf) for { n, from, err := ue.conn.ReadFrom(buf[:]) if err != nil { break } ue.mu.Lock() ue.deadlineTimer.Reset(ue.NatTimeout) ue.mu.Unlock() if err = ue.handler(buf[:n], from); err != nil { break } } ue.mu.Lock() ue.deadlineTimer.Stop() ue.mu.Unlock() } func (ue *UdpEndpoint) WriteTo(b []byte, addr string) (int, error) { return ue.conn.WriteTo(b, addr) } func (ue *UdpEndpoint) Close() error { ue.mu.Lock() if ue.deadlineTimer != nil { ue.deadlineTimer.Stop() } ue.mu.Unlock() return ue.conn.Close() } // UdpEndpointPool is a full-cone udp conn pool type UdpEndpointPool struct { pool map[netip.AddrPort]*UdpEndpoint mu sync.Mutex } type UdpEndpointOptions struct { Handler UdpHandler NatTimeout time.Duration Dialer *dialer.Dialer // Network is useful for MagicNetwork Network string // Target is useful only if the underlay does not support Full-cone. Target string } var DefaultUdpEndpointPool = NewUdpEndpointPool() func NewUdpEndpointPool() *UdpEndpointPool { return &UdpEndpointPool{ pool: make(map[netip.AddrPort]*UdpEndpoint), } } func (p *UdpEndpointPool) Remove(lAddr netip.AddrPort, udpEndpoint *UdpEndpoint) (err error) { p.mu.Lock() defer p.mu.Unlock() if ue, ok := p.pool[lAddr]; ok { if ue != udpEndpoint { return fmt.Errorf("target udp endpoint is not in the pool") } ue.Close() delete(p.pool, lAddr) } return nil } func (p *UdpEndpointPool) GetOrCreate(lAddr netip.AddrPort, createOption *UdpEndpointOptions) (udpEndpoint *UdpEndpoint, isNew bool, err error) { // TODO: fine-grained lock. p.mu.Lock() defer p.mu.Unlock() ue, ok := p.pool[lAddr] if !ok { // Create an UdpEndpoint. if createOption == nil { createOption = &UdpEndpointOptions{} } if createOption.NatTimeout == 0 { createOption.NatTimeout = DefaultNatTimeout } if createOption.Handler == nil { return nil, true, fmt.Errorf("createOption.Handler cannot be nil") } udpConn, err := createOption.Dialer.Dial(createOption.Network, createOption.Target) if err != nil { return nil, true, err } if _, ok = udpConn.(netproxy.PacketConn); !ok { return nil, true, fmt.Errorf("protocol does not support udp") } ue = &UdpEndpoint{ conn: udpConn.(netproxy.PacketConn), deadlineTimer: time.AfterFunc(createOption.NatTimeout, func() { p.mu.Lock() defer p.mu.Unlock() if ue, ok := p.pool[lAddr]; ok { ue.Close() delete(p.pool, lAddr) } }), handler: createOption.Handler, NatTimeout: createOption.NatTimeout, Dialer: createOption.Dialer, } p.pool[lAddr] = ue // Receive UDP messages. go ue.start() isNew = true } else { // Postpone the deadline. ue.mu.Lock() ue.deadlineTimer.Reset(ue.NatTimeout) ue.mu.Unlock() } return ue, isNew, nil }