wgengine: have pinger use all single-IP routes

Fixes #139

Signed-off-by: David Crawshaw <crawshaw@tailscale.com>
This commit is contained in:
David Crawshaw 2020-03-08 07:08:38 -04:00
parent 8ae3ba0cf5
commit ce7f6b2df1
1 changed files with 22 additions and 17 deletions

View File

@ -156,21 +156,21 @@ func newUserspaceEngineAdvanced(logf logger.Logf, tundev tun.Device, routerGen R
// here. // here.
go e.RequestStatus() go e.RequestStatus()
// All nodes have one primary IP address, and it // Ping every single-IP that peer routes.
// is the first entry on the AllowedIPs list. // These synthetic packets are used to traverse NATs.
// var ips []wgcfg.IP
// This code is written defensively in case we ever for _, ipNet := range allowedIPs {
// end up with an empty AllowedIPs list or somehow if ones, bits := ipNet.Mask.Size(); ones == bits && ones != 0 {
// have a subnet as the first entry.
if len(allowedIPs) > 0 {
if ones, bits := allowedIPs[0].Mask.Size(); ones == bits && ones != 0 {
var ip wgcfg.IP var ip wgcfg.IP
copy(ip.Addr[:], allowedIPs[0].IP.To16()) copy(ip.Addr[:], ipNet.IP.To16())
go e.pinger(peerKey, ip) ips = append(ips, ip)
return
} }
} }
logf("ERROR: peer %s has unexpected AllowedIPs: %v", peerKey.ShortString(), allowedIPs) if len(ips) > 0 {
go e.pinger(peerKey, ips)
} else {
logf("[unexpected] peer %s has no single-IP routes: %v", peerKey.ShortString(), allowedIPs)
}
}, },
CreateBind: e.magicConn.CreateBind, CreateBind: e.magicConn.CreateBind,
CreateEndpoint: e.magicConn.CreateEndpoint, CreateEndpoint: e.magicConn.CreateEndpoint,
@ -227,8 +227,8 @@ func newUserspaceEngineAdvanced(logf logger.Logf, tundev tun.Device, routerGen R
// //
// These generated packets are used to ensure we trigger the spray logic in // These generated packets are used to ensure we trigger the spray logic in
// the magicsock package for NAT traversal. // the magicsock package for NAT traversal.
func (e *userspaceEngine) pinger(peerKey wgcfg.Key, ip wgcfg.IP) { func (e *userspaceEngine) pinger(peerKey wgcfg.Key, ips []wgcfg.IP) {
e.logf("generating initial ping traffic to %s (%v)", peerKey.ShortString(), ip) e.logf("generating initial ping traffic to %s (%v)", peerKey.ShortString(), ips)
var srcIP packet.IP var srcIP packet.IP
e.wgLock.Lock() e.wgLock.Lock()
@ -257,7 +257,10 @@ func (e *userspaceEngine) pinger(peerKey wgcfg.Key, ip wgcfg.IP) {
const stopAfter = 3 * time.Second const stopAfter = 3 * time.Second
start := time.Now() start := time.Now()
dstIP := packet.NewIP(ip.IP()) var dstIPs []packet.IP
for _, ip := range ips {
dstIPs = append(dstIPs, packet.NewIP(ip.IP()))
}
payload := []byte("magicsock_spray") // no meaning payload := []byte("magicsock_spray") // no meaning
@ -286,9 +289,11 @@ func (e *userspaceEngine) pinger(peerKey wgcfg.Key, ip wgcfg.IP) {
if time.Since(start) > stopAfter { if time.Since(start) > stopAfter {
return return
} }
b := packet.GenICMP(srcIP, dstIP, ipid, packet.EchoRequest, 0, payload) for _, dstIP := range dstIPs {
b := packet.GenICMP(srcIP, dstIP, ipid, packet.EchoRequest, 0, payload)
e.wgdev.SendPacket(b)
}
ipid++ ipid++
e.wgdev.SendPacket(b)
} }
} }