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-02 20:22:18 +07:00
"bufio"
2023-01-29 06:31:52 +07:00
"encoding/binary"
2023-02-02 20:22:18 +07:00
"errors"
2023-01-28 12:27:54 +07:00
"fmt"
2023-01-23 18:54:21 +07:00
"github.com/cilium/ebpf"
2023-01-24 15:27:19 +07:00
"github.com/v2rayA/dae/common"
2023-02-04 10:38:01 +07:00
"github.com/v2rayA/dae/common/consts"
2023-01-28 12:27:54 +07:00
"github.com/v2rayA/dae/pkg/ebpf_internal"
2023-01-23 18:54:21 +07:00
"net/netip"
2023-02-01 20:15:58 +07:00
"os"
2023-01-28 12:27:54 +07:00
"reflect"
2023-02-01 20:15:58 +07:00
"strings"
2023-01-23 18:54:21 +07:00
)
2023-01-28 10:47:02 +07:00
type _bpfLpmKey struct {
2023-01-23 18:54:21 +07:00
PrefixLen uint32
Data [ 4 ] uint32
}
2023-01-29 06:31:52 +07:00
type _bpfPortRange struct {
PortStart uint16
PortEnd uint16
}
2023-02-04 10:24:03 +07:00
func ( r _bpfPortRange ) Encode ( ) ( b [ 16 ] byte ) {
binary . LittleEndian . PutUint16 ( b [ : 2 ] , r . PortStart )
binary . LittleEndian . PutUint16 ( b [ 2 : ] , r . PortEnd )
return b
2023-01-29 06:31:52 +07:00
}
2023-01-28 10:47:02 +07:00
func ( o * bpfObjects ) newLpmMap ( keys [ ] _bpfLpmKey , values [ ] uint32 ) ( m * ebpf . Map , err error ) {
2023-01-24 15:27:19 +07:00
m , err = ebpf . NewMap ( & ebpf . MapSpec {
Type : ebpf . LPMTrie ,
Flags : o . UnusedLpmType . Flags ( ) ,
MaxEntries : o . UnusedLpmType . MaxEntries ( ) ,
KeySize : o . UnusedLpmType . KeySize ( ) ,
ValueSize : o . UnusedLpmType . ValueSize ( ) ,
} )
2023-01-23 18:54:21 +07:00
if err != nil {
return nil , err
}
2023-01-28 12:27:54 +07:00
if _ , err = BatchUpdate ( m , keys , values , & ebpf . BatchOptions {
2023-01-23 18:54:21 +07:00
ElemFlags : uint64 ( ebpf . UpdateAny ) ,
} ) ; err != nil {
return nil , err
}
return m , nil
}
2023-01-28 10:47:02 +07:00
func cidrToBpfLpmKey ( prefix netip . Prefix ) _bpfLpmKey {
2023-01-23 18:54:21 +07:00
bits := prefix . Bits ( )
if prefix . Addr ( ) . Is4 ( ) {
bits += 96
}
2023-01-24 15:27:19 +07:00
ip := prefix . Addr ( ) . As16 ( )
2023-01-28 10:47:02 +07:00
return _bpfLpmKey {
2023-01-23 18:54:21 +07:00
PrefixLen : uint32 ( bits ) ,
Data : common . Ipv6ByteSliceToUint32Array ( ip [ : ] ) ,
}
}
2023-01-28 12:27:54 +07:00
func BatchUpdate ( m * ebpf . Map , keys interface { } , values interface { } , opts * ebpf . BatchOptions ) ( n int , err error ) {
var old bool
version , e := internal . KernelVersion ( )
2023-02-04 12:37:36 +07:00
if e != nil || version . Less ( consts . UserspaceBatchUpdateFeatureVersion ) {
old = true
}
if m . Type ( ) == ebpf . LPMTrie && version . Less ( consts . UserspaceBatchUpdateLpmTrieFeatureVersion ) {
2023-01-28 12:27:54 +07:00
old = true
}
if ! old {
return m . BatchUpdate ( keys , values , opts )
} else {
vKeys := reflect . ValueOf ( keys )
if vKeys . Kind ( ) != reflect . Slice {
return 0 , fmt . Errorf ( "keys must be slice" )
}
vVals := reflect . ValueOf ( values )
if vVals . Kind ( ) != reflect . Slice {
return 0 , fmt . Errorf ( "values must be slice" )
}
length := vKeys . Len ( )
if vVals . Len ( ) != length {
return 0 , fmt . Errorf ( "keys and values must have same length" )
}
for i := 0 ; i < length ; i ++ {
vKey := vKeys . Index ( i )
vVal := vVals . Index ( i )
if err = m . Update ( vKey . Interface ( ) , vVal . Interface ( ) , ebpf . MapUpdateFlags ( opts . ElemFlags ) ) ; err != nil {
return i , err
}
}
return vKeys . Len ( ) , nil
}
}
2023-02-01 00:34:50 +07:00
2023-02-01 09:24:11 +07:00
func AssignBpfObjects ( to * bpfObjects , from interface { } ) {
2023-02-01 00:34:50 +07:00
vTo := reflect . Indirect ( reflect . ValueOf ( to ) )
vFrom := reflect . Indirect ( reflect . ValueOf ( from ) )
tFrom := vFrom . Type ( )
// programs
for i := 0 ; i < vFrom . NumField ( ) ; i ++ {
fieldFrom := vFrom . Field ( i )
structFieldFrom := tFrom . Field ( i )
if structFieldFrom . Type != reflect . TypeOf ( & ebpf . Program { } ) {
continue
}
fieldTo := vTo . FieldByName ( structFieldFrom . Name )
fieldTo . Set ( fieldFrom )
}
// bpfMaps
vFrom = vFrom . FieldByName ( "bpfMaps" )
tFrom = vFrom . Type ( )
for i := 0 ; i < vFrom . NumField ( ) ; i ++ {
fieldFrom := vFrom . Field ( i )
structFieldFrom := tFrom . Field ( i )
fieldTo := vTo . FieldByName ( structFieldFrom . Name )
fieldTo . Set ( fieldFrom )
}
}
2023-02-02 20:22:18 +07:00
// detectCgroupPath returns the first-found mount point of type cgroup2
// and stores it in the cgroupPath global variable.
// Copied from https://github.com/cilium/ebpf/blob/v0.10.0/examples/cgroup_skb/main.go
func detectCgroupPath ( ) ( string , error ) {
f , err := os . Open ( "/proc/mounts" )
if err != nil {
return "" , err
}
defer f . Close ( )
scanner := bufio . NewScanner ( f )
for scanner . Scan ( ) {
// example fields: cgroup2 /sys/fs/cgroup/unified cgroup2 rw,nosuid,nodev,noexec,relatime 0 0
fields := strings . Split ( scanner . Text ( ) , " " )
if len ( fields ) >= 3 && fields [ 2 ] == "cgroup2" {
return fields [ 1 ] , nil
}
}
return "" , errors . New ( "cgroup2 not mounted" )
}
2023-02-04 10:38:01 +07:00
func ( p bpfIfParams ) CheckVersionRequirement ( version * internal . Version ) ( err error ) {
if ! p . TxL4CksmIp4Offload ||
! p . TxL4CksmIp6Offload {
// Need calc checksum on CPU. And need BPF_F_ADJ_ROOM_NO_CSUM_RESET.
if version . Less ( consts . ChecksumFeatureVersion ) {
return fmt . Errorf ( "your NIC does not support checksum offload and your kernel version %v does not support related BPF features; expect >=%v; upgrade your kernel and try again" , version . String ( ) ,
consts . ChecksumFeatureVersion . String ( ) )
}
}
return nil
}