dae/component/control/udp_endpoint.go

139 lines
3.0 KiB
Go
Raw Normal View History

2023-01-23 18:54:21 +07:00
/*
* SPDX-License-Identifier: AGPL-3.0-only
2023-01-28 12:56:06 +07:00
* Copyright (c) since 2022, v2rayA Organization <team@v2raya.org>
2023-01-23 18:54:21 +07:00
*/
package control
import (
2023-02-04 19:53:29 +07:00
"errors"
2023-01-23 18:54:21 +07:00
"fmt"
2023-01-29 12:38:15 +07:00
"github.com/mzz2017/softwind/pool"
"github.com/v2rayA/dae/component/outbound/dialer"
2023-01-23 18:54:21 +07:00
"net"
"net/netip"
"sync"
"time"
)
type UdpHandler func(data []byte, from netip.AddrPort) error
type UdpEndpoint struct {
conn net.PacketConn
// mu protects deadlineTimer
mu sync.Mutex
deadlineTimer *time.Timer
handler UdpHandler
NatTimeout time.Duration
Dialer *dialer.Dialer
2023-01-23 18:54:21 +07:00
}
func (ue *UdpEndpoint) start() {
buf := pool.Get(0xffff)
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.(*net.UDPAddr).AddrPort()); err != nil {
2023-02-04 19:53:29 +07:00
if errors.Is(err, SuspectedRushAnswerError) {
continue
}
2023-01-23 18:54:21 +07:00
break
}
}
ue.mu.Lock()
ue.deadlineTimer.Stop()
ue.mu.Unlock()
}
func (ue *UdpEndpoint) WriteToUDPAddrPort(b []byte, addr netip.AddrPort) (int, error) {
return ue.conn.WriteTo(b, net.UDPAddrFromAddrPort(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
DialerFunc func() (*dialer.Dialer, error)
2023-01-23 18:54:21 +07:00
// Target is useful only if the underlay does not support Full-cone.
Target netip.AddrPort
}
var DefaultUdpEndpointPool = NewUdpEndpointPool()
func NewUdpEndpointPool() *UdpEndpointPool {
return &UdpEndpointPool{
pool: make(map[netip.AddrPort]*UdpEndpoint),
}
}
func (p *UdpEndpointPool) GetOrCreate(lAddr netip.AddrPort, createOption *UdpEndpointOptions) (udpEndpoint *UdpEndpoint, err error) {
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, fmt.Errorf("createOption.Handler cannot be nil")
}
d, err := createOption.DialerFunc()
if err != nil {
return nil, err
}
udpConn, err := d.Dial("udp", createOption.Target.String())
2023-01-23 18:54:21 +07:00
if err != nil {
return nil, err
}
2023-01-28 00:50:21 +07:00
ue = &UdpEndpoint{
2023-01-23 18:54:21 +07:00
conn: udpConn.(net.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: d,
2023-01-23 18:54:21 +07:00
}
2023-01-28 00:50:21 +07:00
p.pool[lAddr] = ue
2023-01-23 18:54:21 +07:00
// Receive UDP messages.
go ue.start()
} else {
// Postpone the deadline.
ue.mu.Lock()
ue.deadlineTimer.Reset(ue.NatTimeout)
ue.mu.Unlock()
}
return ue, nil
}