@@ -24,18 +24,24 @@ import (
24
24
"github.com/metacubex/mihomo/dns"
25
25
"github.com/metacubex/mihomo/log"
26
26
27
+ amnezia "github.com/metacubex/amneziawg-go/device"
27
28
wireguard "github.com/metacubex/sing-wireguard"
29
+ "github.com/metacubex/wireguard-go/device"
28
30
29
31
"github.com/sagernet/sing/common/debug"
30
32
E "github.com/sagernet/sing/common/exceptions"
31
33
M "github.com/sagernet/sing/common/metadata"
32
- "github.com/sagernet/wireguard-go/device"
33
34
)
34
35
36
+ type wireguardGoDevice interface {
37
+ Close ()
38
+ IpcSet (uapiConf string ) error
39
+ }
40
+
35
41
type WireGuard struct {
36
42
* Base
37
43
bind * wireguard.ClientBind
38
- device * device. Device
44
+ device wireguardGoDevice
39
45
tunDevice wireguard.Device
40
46
dialer proxydialer.SingDialer
41
47
resolver * dns.Resolver
@@ -67,6 +73,8 @@ type WireGuardOption struct {
67
73
UDP bool `proxy:"udp,omitempty"`
68
74
PersistentKeepalive int `proxy:"persistent-keepalive,omitempty"`
69
75
76
+ AmneziaWGOption * AmneziaWGOption `proxy:"amnezia-wg-option,omitempty"`
77
+
70
78
Peers []WireGuardPeerOption `proxy:"peers,omitempty"`
71
79
72
80
RemoteDnsResolve bool `proxy:"remote-dns-resolve,omitempty"`
@@ -84,6 +92,18 @@ type WireGuardPeerOption struct {
84
92
AllowedIPs []string `proxy:"allowed-ips,omitempty"`
85
93
}
86
94
95
+ type AmneziaWGOption struct {
96
+ JC int `proxy:"jc"`
97
+ JMin int `proxy:"jmin"`
98
+ JMax int `proxy:"jmax"`
99
+ S1 int `proxy:"s1"`
100
+ S2 int `proxy:"s2"`
101
+ H1 uint32 `proxy:"h1"`
102
+ H2 uint32 `proxy:"h2"`
103
+ H3 uint32 `proxy:"h3"`
104
+ H4 uint32 `proxy:"h4"`
105
+ }
106
+
87
107
type wgSingErrorHandler struct {
88
108
name string
89
109
}
@@ -243,14 +263,19 @@ func NewWireGuard(option WireGuardOption) (*WireGuard, error) {
243
263
if err != nil {
244
264
return nil , E .Cause (err , "create WireGuard device" )
245
265
}
246
- outbound . device = device . NewDevice ( context . Background (), outbound . tunDevice , outbound . bind , & device.Logger {
266
+ logger := & device.Logger {
247
267
Verbosef : func (format string , args ... interface {}) {
248
268
log .SingLogger .Debug (fmt .Sprintf ("[WG](%s) %s" , option .Name , fmt .Sprintf (format , args ... )))
249
269
},
250
270
Errorf : func (format string , args ... interface {}) {
251
271
log .SingLogger .Error (fmt .Sprintf ("[WG](%s) %s" , option .Name , fmt .Sprintf (format , args ... )))
252
272
},
253
- }, option .Workers )
273
+ }
274
+ if option .AmneziaWGOption != nil {
275
+ outbound .device = amnezia .NewDevice (outbound .tunDevice , outbound .bind , logger , option .Workers )
276
+ } else {
277
+ outbound .device = device .NewDevice (outbound .tunDevice , outbound .bind , logger , option .Workers )
278
+ }
254
279
255
280
var has6 bool
256
281
for _ , address := range outbound .localPrefixes {
@@ -367,6 +392,17 @@ func (w *WireGuard) genIpcConf(ctx context.Context, updateOnly bool) (string, er
367
392
ipcConf := ""
368
393
if ! updateOnly {
369
394
ipcConf += "private_key=" + w .option .PrivateKey + "\n "
395
+ if w .option .AmneziaWGOption != nil {
396
+ ipcConf += "jc=" + strconv .Itoa (w .option .AmneziaWGOption .JC ) + "\n "
397
+ ipcConf += "jmin=" + strconv .Itoa (w .option .AmneziaWGOption .JMin ) + "\n "
398
+ ipcConf += "jmax=" + strconv .Itoa (w .option .AmneziaWGOption .JMax ) + "\n "
399
+ ipcConf += "s1=" + strconv .Itoa (w .option .AmneziaWGOption .S1 ) + "\n "
400
+ ipcConf += "s2=" + strconv .Itoa (w .option .AmneziaWGOption .S2 ) + "\n "
401
+ ipcConf += "h1=" + strconv .FormatUint (uint64 (w .option .AmneziaWGOption .H1 ), 10 ) + "\n "
402
+ ipcConf += "h2=" + strconv .FormatUint (uint64 (w .option .AmneziaWGOption .H2 ), 10 ) + "\n "
403
+ ipcConf += "h3=" + strconv .FormatUint (uint64 (w .option .AmneziaWGOption .H3 ), 10 ) + "\n "
404
+ ipcConf += "h4=" + strconv .FormatUint (uint64 (w .option .AmneziaWGOption .H4 ), 10 ) + "\n "
405
+ }
370
406
}
371
407
if len (w .option .Peers ) > 0 {
372
408
for i , peer := range w .option .Peers {
0 commit comments