From 78dc21fcce943061beac2a52d0ce2ddf70a5927d Mon Sep 17 00:00:00 2001 From: Joseph Henry Date: Wed, 15 Mar 2023 13:24:07 -0700 Subject: [PATCH] Prevent path-learning loops --- node/Peer.cpp | 89 +++++++++++++++++++++++---------------------------- 1 file changed, 40 insertions(+), 49 deletions(-) diff --git a/node/Peer.cpp b/node/Peer.cpp index 28351c28d..f73359fdf 100644 --- a/node/Peer.cpp +++ b/node/Peer.cpp @@ -109,6 +109,13 @@ void Peer::received( havePath = true; break; } + // If same address on same interface then don't learn unless existing path isn't alive (prevents learning loop) + if (_paths[i].p->address().ipsEqual(path->address()) && _paths[i].p->localSocket() == path->localSocket()) { + if (_paths[i].p->alive(now)) { + havePath = true; + break; + } + } } else { break; } @@ -116,69 +123,37 @@ void Peer::received( } if ( (!havePath) && RR->node->shouldUsePathForZeroTierTraffic(tPtr,_id.address(),path->localSocket(),path->address()) ) { - - /** - * First, fill all free slots before attempting to replace a path - * - If the above fails, attempt to replace the path that has been dead the longest - * - If there are no free slots, and no dead paths (unlikely), then replace old path most similar to new path - * - If all of the above fails to yield a suitable replacement. Replace first path found to have lower `(quality / priority)` - */ - if (verb == Packet::VERB_OK) { Mutex::Lock _l(_paths_m); + unsigned int oldestPathIdx = ZT_MAX_PEER_NETWORK_PATHS; + unsigned int oldestPathAge = 0; unsigned int replacePath = ZT_MAX_PEER_NETWORK_PATHS; - uint64_t maxScore = 0; - uint64_t currScore; - long replacePathQuality = 0; - bool foundFreeSlot = false; for(unsigned int i=0;ialive(now)) { - currScore = _paths[i].p->age(now) / 1000; + // Keep track of oldest path as a last resort option + unsigned int currAge = _paths[i].p->age(now); + if (currAge > oldestPathAge) { + oldestPathAge = currAge; + oldestPathIdx = i; } - // Reward as similarity increases if (_paths[i].p->address().ipsEqual(path->address())) { - currScore++; - if (_paths[i].p->address().port() == path->address().port()) { - currScore++; - if (_paths[i].p->localSocket() == path->localSocket()) { - currScore++; // max score (3) + if (_paths[i].p->localSocket() == path->localSocket()) { + if (!_paths[i].p->alive(now)) { + replacePath = i; + break; } } } - // If best so far, mark for replacement - if (currScore > maxScore) { - maxScore = currScore; - replacePath = i; - } } else { - foundFreeSlot = true; replacePath = i; break; } } - if (!foundFreeSlot) { - if (maxScore > 3) { - // Do nothing. We found a dead path and have already marked it as a candidate - } - // If we couldn't find a replacement by matching, replacing a dead path, or taking a free slot, then replace by quality - else if (maxScore == 0) { - for(unsigned int i=0;iquality(now) / _paths[i].priority; - if (q > replacePathQuality) { - replacePathQuality = q; - replacePath = i; - } - } - } - } - } + // If we didn't find a good candidate then resort to replacing oldest path + replacePath = (replacePath == ZT_MAX_PEER_NETWORK_PATHS) ? oldestPathIdx : replacePath; if (replacePath != ZT_MAX_PEER_NETWORK_PATHS) { RR->t->peerLearnedNewPath(tPtr, networkId, *this, path, packetId); _paths[replacePath].lr = now; @@ -540,11 +515,15 @@ unsigned int Peer::doPingAndKeepalive(void *tPtr,int64_t now) // let those old links expire. long maxPriority = 0; for(unsigned int i=0;isent(now); sent |= (_paths[i].p->address().ss_family == AF_INET) ? 0x1 : 0x2; } - } else { + } + else { _paths[i] = _PeerPath(); + deletionOccurred = true; } - } else break; + } + if (!_paths[i].p || deletionOccurred) { + for(unsigned int j=i;j