forked from FH0/tproxy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtcp_linux.go
97 lines (82 loc) · 1.85 KB
/
tcp_linux.go
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
package tproxy
import (
"encoding/binary"
"net"
"syscall"
"golang.org/x/sys/unix"
)
//revive:disable
// SO_ORIGINAL_DST syscall and unix not contain
const SO_ORIGINAL_DST = 80
//revive:enable
type tcpConn struct {
net.Conn
daddr net.Addr
}
// LocalAddr returns the local network address, if known.
func (t *tcpConn) LocalAddr() net.Addr {
return t.daddr
}
type tcpListener struct {
net.Listener
}
// Accept like net package
func (t *tcpListener) Accept() (net.Conn, error) {
conn, err := t.Listener.Accept()
if err != nil {
return nil, err
}
file, err := conn.(*net.TCPConn).File()
if err != nil {
return nil, err
}
fd := file.Fd()
var daddr net.Addr
mreq, err := syscall.GetsockoptIPv6Mreq(int(fd), syscall.IPPROTO_IP, SO_ORIGINAL_DST)
if err != nil {
mtuInfo, err := syscall.GetsockoptIPv6MTUInfo(int(fd), syscall.IPPROTO_IPV6, SO_ORIGINAL_DST)
if err != nil {
daddr = conn.LocalAddr()
} else {
daddr = &net.TCPAddr{
IP: mtuInfo.Addr.Addr[:],
Port: int(binary.BigEndian.Uint16([]byte{byte(mtuInfo.Addr.Port), byte(mtuInfo.Addr.Port >> 8)})),
}
}
} else {
daddr = &net.TCPAddr{
IP: mreq.Multiaddr[4:8],
Port: int(binary.BigEndian.Uint16(mreq.Multiaddr[2:])),
}
}
return &tcpConn{
Conn: conn,
daddr: daddr,
}, nil
}
// ListenTCP address is [::], dual stack
func ListenTCP(addr string) (_ net.Listener, err error) {
listener := &tcpListener{}
listener.Listener, err = net.Listen("tcp", addr)
if err != nil {
return
}
file, err := listener.Listener.(*net.TCPListener).File()
if err != nil {
return
}
fd := int(file.Fd())
err = syscall.SetsockoptInt(fd, unix.SOL_IP, unix.IP_TRANSPARENT, 1)
if err != nil {
return
}
err = syscall.SetsockoptInt(fd, unix.SOL_IPV6, unix.IPV6_TRANSPARENT, 1)
if err != nil {
return
}
err = file.Close()
if err != nil {
return
}
return listener, err
}