1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
|
package socket
import (
"encoding/binary"
"errors"
"fmt"
"golang.org/x/sys/unix"
"net"
"strconv"
"syscall"
"unsafe"
)
func be16toh(n uint16) uint16 {
b := (*[2]byte)(unsafe.Pointer(&n))
return binary.BigEndian.Uint16(b[:])
}
func getsockopt(s int, level, name int, ptr unsafe.Pointer, size int) error {
tmpsize := uint32(size)
_, _, errno := unix.Syscall6(unix.SYS_GETSOCKOPT,
uintptr(s),
uintptr(level),
uintptr(name),
uintptr(ptr),
uintptr(unsafe.Pointer(&tmpsize)),
0)
if errno != 0 {
return fmt.Errorf("getsockopt: %w", errno)
}
return nil
}
func ioctl(fd int, req int, ptr unsafe.Pointer) error {
_, _, errno := unix.Syscall(unix.SYS_IOCTL,
uintptr(fd),
uintptr(req),
uintptr(ptr))
if errno != 0 {
return fmt.Errorf("ioctl: %w", errno)
}
return nil
}
func getRawConn(conn net.Conn) (syscall.RawConn, error) {
switch c := conn.(type) {
case *net.TCPConn:
return c.SyscallConn()
default:
return nil, errors.New("unknown connection type")
}
}
func withConnControl(conn net.Conn, f func(fd int) error) (err error) {
var c syscall.RawConn
var ferr error
c, err = getRawConn(conn)
if err != nil {
return
}
err = c.Control(func(fd uintptr) {
ferr = f(int(fd))
})
if ferr != nil {
err = ferr
}
return
}
func getSocketOriginalDst(fd int, sa *unix.RawSockaddrAny) error {
const SO_ORIGINAL_DST = 80
p, n := unsafe.Pointer(sa), int(unsafe.Sizeof(*sa))
family, err := unix.GetsockoptInt(fd, unix.SOL_SOCKET, unix.SO_DOMAIN)
if err != nil {
return err
}
switch family {
case unix.AF_INET6:
err := getsockopt(fd, unix.SOL_IPV6, SO_ORIGINAL_DST, p, n)
if !errors.Is(err, unix.ENOENT) {
return err
}
// skipped check for ipv4 encoded as ipv6 address
fallthrough
case unix.AF_INET:
return getsockopt(fd, unix.SOL_IP, SO_ORIGINAL_DST, p, n)
default:
return errors.New("unknown address family")
}
}
func getConnOriginalAddr(conn net.Conn, addr *string) error {
var sa unix.RawSockaddrAny
f := func(fd int) error {
return getSocketOriginalDst(fd, &sa)
}
if err := withConnControl(conn, f); err != nil {
return fmt.Errorf("get-original-addr: %w", err)
}
var host net.IP
var port uint16
switch sa.Addr.Family {
case unix.AF_INET:
sin := (*unix.RawSockaddrInet4)(unsafe.Pointer(&sa))
host, port = sin.Addr[:], sin.Port
case unix.AF_INET6:
sin := (*unix.RawSockaddrInet6)(unsafe.Pointer(&sa))
host, port = sin.Addr[:], sin.Port
default:
return errors.New("get-original-addr: unknown address family")
}
*addr = net.JoinHostPort(host.String(), strconv.Itoa(int(be16toh(port))))
return nil
}
|