2023-01-23 18:54:21 +07:00
/ *
* SPDX - License - Identifier : AGPL - 3.0 - only
2023-02-18 17:27:28 +07:00
* Copyright ( c ) 2022 - 2023 , v2rayA Organization < team @ v2raya . org >
2023-01-23 18:54:21 +07:00
* /
package control
import (
"encoding/binary"
2023-02-04 19:53:29 +07:00
"errors"
2023-01-23 18:54:21 +07:00
"fmt"
2023-02-21 15:10:44 +07:00
"github.com/mzz2017/softwind/pkg/zeroalloc/buffer"
2023-01-31 18:54:11 +07:00
"github.com/sirupsen/logrus"
2023-02-13 17:26:31 +07:00
"github.com/v2rayA/dae/common"
2023-01-23 19:01:24 +07:00
"github.com/v2rayA/dae/common/consts"
2023-02-04 14:28:48 +07:00
"github.com/v2rayA/dae/component/outbound/dialer"
2023-02-15 00:53:53 +07:00
"github.com/v2rayA/dae/component/sniffing"
2023-02-25 01:38:21 +07:00
internal "github.com/v2rayA/dae/pkg/ebpf_internal"
2023-01-23 18:54:21 +07:00
"golang.org/x/net/dns/dnsmessage"
"net"
"net/netip"
2023-02-06 12:56:43 +07:00
"syscall"
2023-01-23 18:54:21 +07:00
"time"
2023-02-21 15:10:44 +07:00
"unsafe"
2023-01-23 18:54:21 +07:00
)
const (
DefaultNatTimeout = 3 * time . Minute
DnsNatTimeout = 17 * time . Second // RFC 5452
2023-02-13 09:52:40 +07:00
MaxRetry = 2
2023-01-23 18:54:21 +07:00
)
2023-02-19 21:16:59 +07:00
func ChooseNatTimeout ( data [ ] byte , sniffDns bool ) ( dmsg * dnsmessage . Message , timeout time . Duration ) {
2023-02-25 01:38:21 +07:00
if sniffDns {
var dnsmsg dnsmessage . Message
if err := dnsmsg . Unpack ( data ) ; err == nil {
//log.Printf("DEBUG: lookup %v", dnsmsg.Questions[0].Name)
return & dnsmsg , DnsNatTimeout
}
2023-01-23 18:54:21 +07:00
}
return nil , DefaultNatTimeout
}
2023-02-21 15:10:44 +07:00
func ParseAddrHdr ( data [ ] byte ) ( hdr * bpfDstRoutingResult , dataOffset int , err error ) {
dataOffset = int ( unsafe . Sizeof ( bpfDstRoutingResult { } ) )
2023-01-23 18:54:21 +07:00
if len ( data ) < dataOffset {
return nil , 0 , fmt . Errorf ( "data is too short to parse AddrHdr" )
}
2023-02-21 15:10:44 +07:00
_hdr := * ( * bpfDstRoutingResult ) ( unsafe . Pointer ( & data [ 0 ] ) )
_hdr . Port = common . Ntohs ( _hdr . Port )
return & _hdr , dataOffset , nil
2023-01-23 18:54:21 +07:00
}
2023-02-25 01:38:21 +07:00
func sendPktWithHdrWithFlag ( data [ ] byte , realFrom netip . AddrPort , lConn * net . UDPConn , to netip . AddrPort , lanWanFlag consts . LanWanFlag ) error {
realFrom16 := realFrom . Addr ( ) . As16 ( )
2023-02-21 15:10:44 +07:00
hdr := bpfDstRoutingResult {
2023-02-25 01:38:21 +07:00
Ip : common . Ipv6ByteSliceToUint32Array ( realFrom16 [ : ] ) ,
Port : common . Htons ( realFrom . Port ( ) ) ,
2023-02-21 15:10:44 +07:00
RoutingResult : bpfRoutingResult {
Outbound : uint8 ( lanWanFlag ) , // Pass some message to the kernel program.
} ,
2023-01-23 18:54:21 +07:00
}
2023-02-25 01:38:21 +07:00
// Do not put this 'buf' because it has been taken by buffer.
b := buffer . NewBuffer ( int ( unsafe . Sizeof ( hdr ) ) + len ( data ) )
2023-02-21 15:10:44 +07:00
defer b . Put ( )
2023-02-25 01:38:21 +07:00
// Use internal.NativeEndian due to already big endian.
if err := binary . Write ( b , internal . NativeEndian , hdr ) ; err != nil {
2023-02-21 15:10:44 +07:00
return err
}
2023-02-25 01:38:21 +07:00
b . Write ( data )
//logrus.Debugln("sendPktWithHdrWithFlag: from", realFrom, "to", to)
2023-02-26 04:07:42 +07:00
if ipversion := consts . IpVersionFromAddr ( to . Addr ( ) ) ; consts . IpVersionFromAddr ( lConn . LocalAddr ( ) . ( * net . UDPAddr ) . AddrPort ( ) . Addr ( ) ) != ipversion {
// ip versions unmatched.
if ipversion == consts . IpVersionStr_4 {
// 4 to 6
to = netip . AddrPortFrom ( netip . AddrFrom16 ( to . Addr ( ) . As16 ( ) ) , to . Port ( ) )
} else {
// Shouldn't happen.
return fmt . Errorf ( "unmatched ipversions" )
}
}
2023-02-25 01:38:21 +07:00
_ , err := lConn . WriteToUDPAddrPort ( b . Bytes ( ) , to )
2023-01-23 18:54:21 +07:00
return err
}
2023-02-12 20:50:15 +07:00
// sendPkt uses bind first, and fallback to send hdr if addr is in use.
2023-02-13 11:54:04 +07:00
func sendPkt ( data [ ] byte , from netip . AddrPort , realTo , to netip . AddrPort , lConn * net . UDPConn , lanWanFlag consts . LanWanFlag ) ( err error ) {
2023-02-26 04:07:42 +07:00
2023-02-25 01:38:21 +07:00
if lanWanFlag == consts . LanWanFlag_IsWan {
return sendPktWithHdrWithFlag ( data , from , lConn , to , lanWanFlag )
}
2023-02-06 12:56:43 +07:00
d := net . Dialer { Control : func ( network , address string , c syscall . RawConn ) error {
return dialer . BindControl ( c , from )
} }
2023-02-12 20:50:15 +07:00
var conn net . Conn
conn , err = d . Dial ( "udp" , realTo . String ( ) )
2023-02-06 12:56:43 +07:00
if err != nil {
2023-02-12 20:50:15 +07:00
if errors . Is ( err , syscall . EADDRINUSE ) {
// Port collision, use traditional method.
2023-02-25 01:38:21 +07:00
return sendPktWithHdrWithFlag ( data , from , lConn , to , lanWanFlag )
2023-02-12 20:50:15 +07:00
}
2023-02-06 12:56:43 +07:00
return err
}
2023-02-12 20:50:15 +07:00
defer conn . Close ( )
2023-02-06 12:56:43 +07:00
uConn := conn . ( * net . UDPConn )
_ , err = uConn . Write ( data )
return err
}
2023-02-20 17:06:54 +07:00
func ( c * ControlPlane ) handlePkt ( lConn * net . UDPConn , data [ ] byte , src , pktDst , realDst netip . AddrPort , routingResult * bpfRoutingResult ) ( err error ) {
2023-02-13 11:54:04 +07:00
var lanWanFlag consts . LanWanFlag
2023-02-12 20:50:15 +07:00
var realSrc netip . AddrPort
2023-02-15 00:53:53 +07:00
var domain string
2023-02-12 20:50:15 +07:00
useAssign := pktDst == realDst // Use sk_assign instead of modify target ip/port.
if useAssign {
2023-02-13 11:54:04 +07:00
lanWanFlag = consts . LanWanFlag_IsLan
2023-02-12 20:50:15 +07:00
realSrc = src
} else {
2023-02-13 11:54:04 +07:00
lanWanFlag = consts . LanWanFlag_IsWan
2023-02-12 20:50:15 +07:00
// From localhost, so dst IP is src IP.
realSrc = netip . AddrPortFrom ( pktDst . Addr ( ) , src . Port ( ) )
}
2023-02-19 21:16:59 +07:00
// To keep consistency with kernel program, we only sniff DNS request sent to 53.
dnsMessage , natTimeout := ChooseNatTimeout ( data , realDst . Port ( ) == 53 )
2023-01-23 18:54:21 +07:00
// We should cache DNS records and set record TTL to 0, in order to monitor the dns req and resp in real time.
isDns := dnsMessage != nil
2023-02-25 01:38:21 +07:00
if ! isDns {
2023-02-15 00:53:53 +07:00
// Sniff Quic
sniffer := sniffing . NewPacketSniffer ( data )
domain , err = sniffer . SniffQuic ( )
if err != nil && ! sniffing . IsSniffingError ( err ) {
sniffer . Close ( )
return err
}
sniffer . Close ( )
2023-02-04 14:28:48 +07:00
}
2023-01-31 18:54:11 +07:00
2023-02-25 01:38:21 +07:00
// Get outbound.
outboundIndex := consts . OutboundIndex ( routingResult . Outbound )
switch outboundIndex {
case consts . OutboundDirect :
case consts . OutboundMustDirect :
outboundIndex = consts . OutboundDirect
isDns = false // Regard as plain traffic.
case consts . OutboundControlPlaneRouting :
if isDns {
// Routing of DNS packets are managed by DNS controller.
break
2023-02-09 10:40:34 +07:00
}
2023-02-25 01:38:21 +07:00
if outboundIndex , routingResult . Mark , err = c . Route ( realSrc , realDst , domain , consts . L4ProtoType_TCP , routingResult ) ; err != nil {
return err
2023-02-09 10:40:34 +07:00
}
2023-02-25 01:38:21 +07:00
routingResult . Outbound = uint8 ( outboundIndex )
2023-02-19 21:16:59 +07:00
if c . log . IsLevelEnabled ( logrus . TraceLevel ) {
2023-02-25 01:38:21 +07:00
c . log . Tracef ( "outbound: %v => %v" ,
consts . OutboundControlPlaneRouting . String ( ) ,
outboundIndex . String ( ) ,
)
2023-02-19 21:16:59 +07:00
}
2023-02-25 01:38:21 +07:00
default :
2023-02-09 10:40:34 +07:00
}
2023-02-25 01:38:21 +07:00
if isDns {
return c . dnsController . Handle_ ( dnsMessage , & udpRequest {
lanWanFlag : lanWanFlag ,
realSrc : realSrc ,
realDst : realDst ,
src : src ,
lConn : lConn ,
routingResult : routingResult ,
} )
2023-02-12 14:39:00 +07:00
}
2023-02-25 01:38:21 +07:00
if int ( outboundIndex ) >= len ( c . outbounds ) {
return fmt . Errorf ( "outbound %v out of range [0, %v]" , outboundIndex , len ( c . outbounds ) - 1 )
2023-02-08 20:28:08 +07:00
}
2023-02-25 01:38:21 +07:00
outbound := c . outbounds [ outboundIndex ]
2023-02-04 14:28:48 +07:00
2023-02-25 01:38:21 +07:00
// Select dialer from outbound (dialer group).
networkType := & dialer . NetworkType {
L4Proto : consts . L4ProtoStr_UDP ,
IpVersion : consts . IpVersionFromAddr ( realDst . Addr ( ) ) ,
IsDns : true , // UDP relies on DNS check result.
}
dialerForNew , _ , err := outbound . Select ( networkType )
if err != nil {
return fmt . Errorf ( "failed to select dialer from group %v (%v, dns?:%v,from: %v): %w" , outbound . Name , networkType . StringWithoutDns ( ) , isDns , realSrc . String ( ) , err )
}
2023-02-09 10:40:34 +07:00
// Dial and send.
2023-02-17 23:49:35 +07:00
// TODO: Rewritten domain should not use full-cone (such as VMess Packet Addr).
// Maybe we should set up a mapping for UDP: Dialer + Target Domain => Remote Resolved IP.
2023-02-25 01:38:21 +07:00
// However, games may not use QUIC for communication, thus we cannot use domain to dial, which is fine.
2023-02-26 19:33:50 +07:00
dialTarget , dialMode := c . ChooseDialTarget ( outboundIndex , realDst , domain )
2023-02-17 23:49:35 +07:00
2023-02-25 01:38:21 +07:00
// Get udp endpoint.
var ue * UdpEndpoint
retry := 0
getNew :
if retry > MaxRetry {
return fmt . Errorf ( "touch max retry limit" )
}
ue , isNew , err := DefaultUdpEndpointPool . GetOrCreate ( realSrc , & UdpEndpointOptions {
// Handler handles response packets and send it to the client.
Handler : func ( data [ ] byte , from netip . AddrPort ) ( err error ) {
// Do not return conn-unrelated err in this func.
return sendPkt ( data , from , realSrc , src , lConn , lanWanFlag )
} ,
NatTimeout : natTimeout ,
Dialer : dialerForNew ,
Network : MagicNetwork ( "udp" , routingResult . Mark ) ,
Target : dialTarget ,
} )
if err != nil {
return fmt . Errorf ( "failed to GetOrCreate (policy: %v): %w" , outbound . GetSelectionPolicy ( ) , err )
}
2023-02-09 10:40:34 +07:00
2023-02-25 01:38:21 +07:00
// If the udp endpoint has been not alive, remove it from pool and get a new one.
if ! isNew && outbound . GetSelectionPolicy ( ) != consts . DialerSelectionPolicy_Fixed && ! ue . Dialer . MustGetAlive ( networkType ) {
2023-02-09 10:40:34 +07:00
2023-02-25 01:38:21 +07:00
if c . log . IsLevelEnabled ( logrus . DebugLevel ) {
c . log . WithFields ( logrus . Fields {
"src" : RefineSourceToShow ( realSrc , realDst . Addr ( ) , lanWanFlag ) ,
"network" : networkType . String ( ) ,
2023-02-28 20:25:15 +07:00
"dialer" : ue . Dialer . Property ( ) . Name ,
2023-02-25 01:38:21 +07:00
"retry" : retry ,
} ) . Debugln ( "Old udp endpoint was not alive and removed." )
2023-02-09 10:40:34 +07:00
}
2023-02-25 01:38:21 +07:00
_ = DefaultUdpEndpointPool . Remove ( realSrc , ue )
retry ++
goto getNew
}
2023-02-09 10:40:34 +07:00
2023-02-25 01:38:21 +07:00
_ , err = ue . WriteTo ( data , dialTarget )
if err != nil {
if c . log . IsLevelEnabled ( logrus . DebugLevel ) {
c . log . WithFields ( logrus . Fields {
"to" : realDst . String ( ) ,
"domain" : domain ,
"pid" : routingResult . Pid ,
"pname" : ProcessName2String ( routingResult . Pname [ : ] ) ,
"mac" : Mac2String ( routingResult . Mac [ : ] ) ,
"from" : realSrc . String ( ) ,
"network" : networkType . StringWithoutDns ( ) ,
"err" : err . Error ( ) ,
"retry" : retry ,
} ) . Debugln ( "Failed to write UDP packet request. Try to remove old UDP endpoint and retry." )
2023-02-09 10:40:34 +07:00
}
2023-02-25 01:38:21 +07:00
_ = DefaultUdpEndpointPool . Remove ( realSrc , ue )
retry ++
goto getNew
2023-02-09 10:40:34 +07:00
}
// Print log.
2023-02-25 01:38:21 +07:00
// Only print routing for new connection to avoid the log exploded (Quic and BT).
if isNew {
if c . log . IsLevelEnabled ( logrus . InfoLevel ) {
fields := logrus . Fields {
"network" : networkType . StringWithoutDns ( ) ,
2023-02-08 20:32:20 +07:00
"outbound" : outbound . Name ,
2023-02-13 14:27:36 +07:00
"policy" : outbound . GetSelectionPolicy ( ) ,
2023-02-28 20:25:15 +07:00
"dialer" : ue . Dialer . Property ( ) . Name ,
2023-02-15 00:53:53 +07:00
"domain" : domain ,
2023-02-21 15:10:44 +07:00
"pid" : routingResult . Pid ,
"pname" : ProcessName2String ( routingResult . Pname [ : ] ) ,
"mac" : Mac2String ( routingResult . Mac [ : ] ) ,
2023-02-26 19:33:50 +07:00
"dialMode" : dialMode ,
2023-02-25 01:38:21 +07:00
}
c . log . WithFields ( fields ) . Infof ( "%v <-> %v" , RefineSourceToShow ( realSrc , realDst . Addr ( ) , lanWanFlag ) , RefineAddrPortToShow ( realDst ) )
2023-02-08 20:32:20 +07:00
}
2023-01-23 18:54:21 +07:00
}
2023-02-09 10:40:34 +07:00
2023-01-23 18:54:21 +07:00
return nil
}