dae/control/packet_sniffer_pool.go

105 lines
2.2 KiB
Go

/*
* SPDX-License-Identifier: AGPL-3.0-only
* Copyright (c) 2022-2023, daeuniverse Organization <dae@v2raya.org>
*/
package control
import (
"fmt"
"net/netip"
"sync"
"time"
"github.com/daeuniverse/dae/component/sniffing"
)
const (
PacketSnifferTtl = 3 * time.Second
)
type PacketSniffer struct {
*sniffing.Sniffer
deadlineTimer *time.Timer
Mu sync.Mutex
}
// PacketSnifferPool is a full-cone udp conn pool
type PacketSnifferPool struct {
pool sync.Map
createMuMap sync.Map
}
type PacketSnifferOptions struct {
Ttl time.Duration
}
type PacketSnifferKey struct {
LAddr netip.AddrPort
RAddr netip.AddrPort
}
var DefaultPacketSnifferPool = NewPacketSnifferPool()
func NewPacketSnifferPool() *PacketSnifferPool {
return &PacketSnifferPool{}
}
func (p *PacketSnifferPool) Remove(key PacketSnifferKey, sniffer *PacketSniffer) (err error) {
if ue, ok := p.pool.LoadAndDelete(key); ok {
sniffer.Close()
if ue != sniffer {
return fmt.Errorf("target udp endpoint is not in the pool")
}
}
return nil
}
func (p *PacketSnifferPool) Get(key PacketSnifferKey) *PacketSniffer {
_qs, ok := p.pool.Load(key)
if !ok {
return nil
}
return _qs.(*PacketSniffer)
}
func (p *PacketSnifferPool) GetOrCreate(key PacketSnifferKey, createOption *PacketSnifferOptions) (qs *PacketSniffer, isNew bool) {
_qs, ok := p.pool.Load(key)
begin:
if !ok {
createMu, _ := p.createMuMap.LoadOrStore(key, &sync.Mutex{})
createMu.(*sync.Mutex).Lock()
defer createMu.(*sync.Mutex).Unlock()
defer p.createMuMap.Delete(key)
_qs, ok = p.pool.Load(key)
if ok {
goto begin
}
// Create an PacketSniffer.
if createOption == nil {
createOption = &PacketSnifferOptions{}
}
if createOption.Ttl == 0 {
createOption.Ttl = PacketSnifferTtl
}
qs = &PacketSniffer{
Sniffer: sniffing.NewPacketSniffer(nil, createOption.Ttl),
Mu: sync.Mutex{},
deadlineTimer: nil,
}
qs.deadlineTimer = time.AfterFunc(createOption.Ttl, func() {
if _qs, ok := p.pool.LoadAndDelete(key); ok {
if _qs.(*PacketSniffer) == qs {
qs.Close()
} else {
// FIXME: ?
}
}
})
_qs = qs
p.pool.Store(key, qs)
// Receive UDP messages.
isNew = true
}
return _qs.(*PacketSniffer), isNew
}