@@ -20,11 +20,13 @@ import (
20
20
)
21
21
22
22
var (
23
- stackOnce sync.Once
24
- ipv4Enabled bool
25
- ipv6Enabled bool
26
- unStrmDgramEnabled bool
27
- rawSocketSess bool
23
+ stackOnce sync.Once
24
+ ipv4Enabled bool
25
+ canListenTCP4OnLoopback bool
26
+ ipv6Enabled bool
27
+ canListenTCP6OnLoopback bool
28
+ unStrmDgramEnabled bool
29
+ rawSocketSess bool
28
30
29
31
aLongTimeAgo = time .Unix (233431200 , 0 )
30
32
neverTimeout = time.Time {}
@@ -37,9 +39,17 @@ func probeStack() {
37
39
if _ , err := RoutedInterface ("ip4" , net .FlagUp ); err == nil {
38
40
ipv4Enabled = true
39
41
}
42
+ if ln , err := net .Listen ("tcp4" , "127.0.0.1:0" ); err == nil {
43
+ ln .Close ()
44
+ canListenTCP4OnLoopback = true
45
+ }
40
46
if _ , err := RoutedInterface ("ip6" , net .FlagUp ); err == nil {
41
47
ipv6Enabled = true
42
48
}
49
+ if ln , err := net .Listen ("tcp6" , "[::1]:0" ); err == nil {
50
+ ln .Close ()
51
+ canListenTCP6OnLoopback = true
52
+ }
43
53
rawSocketSess = supportsRawSocket ()
44
54
switch runtime .GOOS {
45
55
case "aix" :
@@ -152,22 +162,23 @@ func TestableAddress(network, address string) bool {
152
162
// The provided network must be "tcp", "tcp4", "tcp6", "unix" or
153
163
// "unixpacket".
154
164
func NewLocalListener (network string ) (net.Listener , error ) {
165
+ stackOnce .Do (probeStack )
155
166
switch network {
156
167
case "tcp" :
157
- if SupportsIPv4 () {
168
+ if canListenTCP4OnLoopback {
158
169
if ln , err := net .Listen ("tcp4" , "127.0.0.1:0" ); err == nil {
159
170
return ln , nil
160
171
}
161
172
}
162
- if SupportsIPv6 () {
173
+ if canListenTCP6OnLoopback {
163
174
return net .Listen ("tcp6" , "[::1]:0" )
164
175
}
165
176
case "tcp4" :
166
- if SupportsIPv4 () {
177
+ if canListenTCP4OnLoopback {
167
178
return net .Listen ("tcp4" , "127.0.0.1:0" )
168
179
}
169
180
case "tcp6" :
170
- if SupportsIPv6 () {
181
+ if canListenTCP6OnLoopback {
171
182
return net .Listen ("tcp6" , "[::1]:0" )
172
183
}
173
184
case "unix" , "unixpacket" :
@@ -185,22 +196,23 @@ func NewLocalListener(network string) (net.Listener, error) {
185
196
//
186
197
// The provided network must be "udp", "udp4", "udp6" or "unixgram".
187
198
func NewLocalPacketListener (network string ) (net.PacketConn , error ) {
199
+ stackOnce .Do (probeStack )
188
200
switch network {
189
201
case "udp" :
190
- if SupportsIPv4 () {
202
+ if canListenTCP4OnLoopback {
191
203
if c , err := net .ListenPacket ("udp4" , "127.0.0.1:0" ); err == nil {
192
204
return c , nil
193
205
}
194
206
}
195
- if SupportsIPv6 () {
207
+ if canListenTCP6OnLoopback {
196
208
return net .ListenPacket ("udp6" , "[::1]:0" )
197
209
}
198
210
case "udp4" :
199
- if SupportsIPv4 () {
211
+ if canListenTCP4OnLoopback {
200
212
return net .ListenPacket ("udp4" , "127.0.0.1:0" )
201
213
}
202
214
case "udp6" :
203
- if SupportsIPv6 () {
215
+ if canListenTCP6OnLoopback {
204
216
return net .ListenPacket ("udp6" , "[::1]:0" )
205
217
}
206
218
case "unixgram" :
0 commit comments