mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-07 08:07:42 +00:00
wgengine/magicsock: track IP<>node mappings without relying on discokeys.
Updates #3088. Signed-off-by: David Anderson <danderson@tailscale.com> (cherry picked from commit b956139b0ce20ec89950279eae3e285929fd4598)
This commit is contained in:
parent
7fbbaff617
commit
6c0723fbd6
@ -194,15 +194,18 @@ func (m *peerMap) upsertEndpoint(ep *endpoint) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetDiscoKeyForIPPort makes future peer lookups by ipp return the
|
// setNodeKeyForIPPort makes future peer lookups by ipp return the
|
||||||
// same peer info as the lookup by dk.
|
// same endpoint as a lookup by nk.
|
||||||
func (m *peerMap) setDiscoKeyForIPPort(ipp netaddr.IPPort, dk tailcfg.DiscoKey) {
|
//
|
||||||
// Check for a prior mapping for ipp, may need to clean it up.
|
// This should only be called with a fully verified mapping of ipp to
|
||||||
|
// nk, because calling this function defines the endpoint we hand to
|
||||||
|
// WireGuard for packets received from ipp.
|
||||||
|
func (m *peerMap) setNodeKeyForIPPort(ipp netaddr.IPPort, nk tailcfg.NodeKey) {
|
||||||
if pi := m.byIPPort[ipp]; pi != nil {
|
if pi := m.byIPPort[ipp]; pi != nil {
|
||||||
delete(pi.ipPorts, ipp)
|
delete(pi.ipPorts, ipp)
|
||||||
delete(m.byIPPort, ipp)
|
delete(m.byIPPort, ipp)
|
||||||
}
|
}
|
||||||
if pi, ok := m.byDiscoKey[dk]; ok {
|
if pi, ok := m.byNodeKey[nk]; ok {
|
||||||
pi.ipPorts[ipp] = true
|
pi.ipPorts[ipp] = true
|
||||||
m.byIPPort[ipp] = pi
|
m.byIPPort[ipp] = pi
|
||||||
}
|
}
|
||||||
@ -844,10 +847,10 @@ func (c *Conn) callNetInfoCallbackLocked(ni *tailcfg.NetInfo) {
|
|||||||
// discoKey. It's used in tests to enable receiving of packets from
|
// discoKey. It's used in tests to enable receiving of packets from
|
||||||
// addr without having to spin up the entire active discovery
|
// addr without having to spin up the entire active discovery
|
||||||
// machinery.
|
// machinery.
|
||||||
func (c *Conn) addValidDiscoPathForTest(discoKey tailcfg.DiscoKey, addr netaddr.IPPort) {
|
func (c *Conn) addValidDiscoPathForTest(nodeKey tailcfg.NodeKey, addr netaddr.IPPort) {
|
||||||
c.mu.Lock()
|
c.mu.Lock()
|
||||||
defer c.mu.Unlock()
|
defer c.mu.Unlock()
|
||||||
c.peerMap.setDiscoKeyForIPPort(addr, discoKey)
|
c.peerMap.setNodeKeyForIPPort(addr, nodeKey)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Conn) SetNetInfoCallback(fn func(*tailcfg.NetInfo)) {
|
func (c *Conn) SetNetInfoCallback(fn func(*tailcfg.NetInfo)) {
|
||||||
@ -1940,10 +1943,21 @@ func (c *Conn) handlePingLocked(dm *disco.Ping, src netaddr.IPPort, di *discoInf
|
|||||||
likelyHeartBeat := src == di.lastPingFrom && time.Since(di.lastPingTime) < 5*time.Second
|
likelyHeartBeat := src == di.lastPingFrom && time.Since(di.lastPingTime) < 5*time.Second
|
||||||
di.lastPingFrom = src
|
di.lastPingFrom = src
|
||||||
di.lastPingTime = time.Now()
|
di.lastPingTime = time.Now()
|
||||||
|
isDerp := src.IP() == derpMagicIPAddr
|
||||||
|
|
||||||
|
// If we can figure out with certainty which node key this disco
|
||||||
|
// message is for, eagerly update our IP<>node and disco<>node
|
||||||
|
// mappings to make p2p path discovery faster in simple
|
||||||
|
// cases. Without this, disco would still work, but would be
|
||||||
|
// reliant on DERP call-me-maybe to establish the disco<>node
|
||||||
|
// mapping, and on subsequent disco handlePongLocked to establish
|
||||||
|
// the IP<>disco mapping.
|
||||||
if nk, ok := c.unambiguousNodeKeyOfPingLocked(dm, di.discoKey, derpNodeSrc); ok {
|
if nk, ok := c.unambiguousNodeKeyOfPingLocked(dm, di.discoKey, derpNodeSrc); ok {
|
||||||
di.setNodeKey(nk)
|
di.setNodeKey(nk)
|
||||||
|
if !isDerp {
|
||||||
|
c.peerMap.setNodeKeyForIPPort(src, nk)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
isDerp := src.IP() == derpMagicIPAddr
|
|
||||||
|
|
||||||
// If we got a ping over DERP, then derpNodeSrc is non-zero and we reply
|
// If we got a ping over DERP, then derpNodeSrc is non-zero and we reply
|
||||||
// over DERP (in which case ipDst is also a DERP address).
|
// over DERP (in which case ipDst is also a DERP address).
|
||||||
@ -1960,7 +1974,6 @@ func (c *Conn) handlePingLocked(dm *disco.Ping, src netaddr.IPPort, di *discoInf
|
|||||||
numNodes = 1
|
numNodes = 1
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
c.setAddrToDiscoLocked(src, di.discoKey)
|
|
||||||
c.peerMap.forEachEndpointWithDiscoKey(di.discoKey, func(ep *endpoint) {
|
c.peerMap.forEachEndpointWithDiscoKey(di.discoKey, func(ep *endpoint) {
|
||||||
ep.addCandidateEndpoint(src)
|
ep.addCandidateEndpoint(src)
|
||||||
numNodes++
|
numNodes++
|
||||||
@ -2035,22 +2048,6 @@ func (c *Conn) enqueueCallMeMaybe(derpAddr netaddr.IPPort, de *endpoint) {
|
|||||||
go de.sendDiscoMessage(derpAddr, &disco.CallMeMaybe{MyNumber: eps}, discoLog)
|
go de.sendDiscoMessage(derpAddr, &disco.CallMeMaybe{MyNumber: eps}, discoLog)
|
||||||
}
|
}
|
||||||
|
|
||||||
// setAddrToDiscoLocked records that newk is at src.
|
|
||||||
//
|
|
||||||
// c.mu must be held.
|
|
||||||
func (c *Conn) setAddrToDiscoLocked(src netaddr.IPPort, newk tailcfg.DiscoKey) {
|
|
||||||
oldEp, ok := c.peerMap.endpointForIPPort(src)
|
|
||||||
if !ok {
|
|
||||||
c.logf("[v1] magicsock: disco: adding mapping of %v to %v", src, newk.ShortString())
|
|
||||||
} else if oldEp.discoKey != newk {
|
|
||||||
c.logf("[v1] magicsock: disco: changing mapping of %v from %x=>%x", src, oldEp.discoKey.ShortString(), newk.ShortString())
|
|
||||||
} else {
|
|
||||||
// No change
|
|
||||||
return
|
|
||||||
}
|
|
||||||
c.peerMap.setDiscoKeyForIPPort(src, newk)
|
|
||||||
}
|
|
||||||
|
|
||||||
// discoInfoLocked returns the previous or new discoInfo for k.
|
// discoInfoLocked returns the previous or new discoInfo for k.
|
||||||
//
|
//
|
||||||
// c.mu must be held.
|
// c.mu must be held.
|
||||||
@ -3689,7 +3686,7 @@ func (de *endpoint) handlePongConnLocked(m *disco.Pong, di *discoInfo, src netad
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
de.c.setAddrToDiscoLocked(src, de.discoKey)
|
de.c.peerMap.setNodeKeyForIPPort(src, de.publicKey)
|
||||||
|
|
||||||
st.addPongReplyLocked(pongReply{
|
st.addPongReplyLocked(pongReply{
|
||||||
latency: latency,
|
latency: latency,
|
||||||
|
@ -1248,7 +1248,7 @@ func addTestEndpoint(tb testing.TB, conn *Conn, sendConn net.PacketConn) (tailcf
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
tb.Fatal(err)
|
tb.Fatal(err)
|
||||||
}
|
}
|
||||||
conn.addValidDiscoPathForTest(discoKey, netaddr.MustParseIPPort(sendConn.LocalAddr().String()))
|
conn.addValidDiscoPathForTest(nodeKey, netaddr.MustParseIPPort(sendConn.LocalAddr().String()))
|
||||||
return nodeKey, discoKey
|
return nodeKey, discoKey
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user