mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-03-06 14:19:59 -05:00
Make addrman's bucket placement deterministic.
Give each address a single fixed location in the new and tried tables, which become simple fixed-size arrays instead of sets and vectors. This prevents attackers from having an advantages by inserting an address multiple times. This change was suggested as Countermeasure 1 in Eclipse Attacks on Bitcoin’s Peer-to-Peer Network, Ethan Heilman, Alison Kendler, Aviv Zohar, Sharon Goldberg. ePrint Archive Report 2015/263. March 2015. It is also more efficient.
This commit is contained in:
parent
b23add5521
commit
e6b343d880
2 changed files with 231 additions and 206 deletions
268
src/addrman.cpp
268
src/addrman.cpp
|
@ -14,12 +14,12 @@ int CAddrInfo::GetTriedBucket(const uint256& nKey) const
|
||||||
{
|
{
|
||||||
CDataStream ss1(SER_GETHASH, 0);
|
CDataStream ss1(SER_GETHASH, 0);
|
||||||
std::vector<unsigned char> vchKey = GetKey();
|
std::vector<unsigned char> vchKey = GetKey();
|
||||||
ss1 << ((unsigned char)32) << nKey << vchKey;
|
ss1 << nKey << vchKey;
|
||||||
uint64_t hash1 = Hash(ss1.begin(), ss1.end()).GetCheapHash();
|
uint64_t hash1 = Hash(ss1.begin(), ss1.end()).GetCheapHash();
|
||||||
|
|
||||||
CDataStream ss2(SER_GETHASH, 0);
|
CDataStream ss2(SER_GETHASH, 0);
|
||||||
std::vector<unsigned char> vchGroupKey = GetGroup();
|
std::vector<unsigned char> vchGroupKey = GetGroup();
|
||||||
ss2 << ((unsigned char)32) << nKey << vchGroupKey << (hash1 % ADDRMAN_TRIED_BUCKETS_PER_GROUP);
|
ss2 << nKey << vchGroupKey << (hash1 % ADDRMAN_TRIED_BUCKETS_PER_GROUP);
|
||||||
uint64_t hash2 = Hash(ss2.begin(), ss2.end()).GetCheapHash();
|
uint64_t hash2 = Hash(ss2.begin(), ss2.end()).GetCheapHash();
|
||||||
return hash2 % ADDRMAN_TRIED_BUCKET_COUNT;
|
return hash2 % ADDRMAN_TRIED_BUCKET_COUNT;
|
||||||
}
|
}
|
||||||
|
@ -29,15 +29,24 @@ int CAddrInfo::GetNewBucket(const uint256& nKey, const CNetAddr& src) const
|
||||||
CDataStream ss1(SER_GETHASH, 0);
|
CDataStream ss1(SER_GETHASH, 0);
|
||||||
std::vector<unsigned char> vchGroupKey = GetGroup();
|
std::vector<unsigned char> vchGroupKey = GetGroup();
|
||||||
std::vector<unsigned char> vchSourceGroupKey = src.GetGroup();
|
std::vector<unsigned char> vchSourceGroupKey = src.GetGroup();
|
||||||
ss1 << ((unsigned char)32) << nKey << vchGroupKey << vchSourceGroupKey;
|
ss1 << nKey << vchGroupKey << vchSourceGroupKey;
|
||||||
uint64_t hash1 = Hash(ss1.begin(), ss1.end()).GetCheapHash();
|
uint64_t hash1 = Hash(ss1.begin(), ss1.end()).GetCheapHash();
|
||||||
|
|
||||||
CDataStream ss2(SER_GETHASH, 0);
|
CDataStream ss2(SER_GETHASH, 0);
|
||||||
ss2 << ((unsigned char)32) << nKey << vchSourceGroupKey << (hash1 % ADDRMAN_NEW_BUCKETS_PER_SOURCE_GROUP);
|
ss2 << nKey << vchSourceGroupKey << (hash1 % ADDRMAN_NEW_BUCKETS_PER_SOURCE_GROUP);
|
||||||
uint64_t hash2 = Hash(ss2.begin(), ss2.end()).GetCheapHash();
|
uint64_t hash2 = Hash(ss2.begin(), ss2.end()).GetCheapHash();
|
||||||
return hash2 % ADDRMAN_NEW_BUCKET_COUNT;
|
return hash2 % ADDRMAN_NEW_BUCKET_COUNT;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int CAddrInfo::GetBucketPosition(const uint256 &nKey, bool fNew, int nBucket) const
|
||||||
|
{
|
||||||
|
CDataStream ss1(SER_GETHASH, 0);
|
||||||
|
std::vector<unsigned char> vchKey = GetKey();
|
||||||
|
ss1 << nKey << (fNew ? 'N' : 'K') << nBucket << vchKey;
|
||||||
|
uint64_t hash1 = Hash(ss1.begin(), ss1.end()).GetCheapHash();
|
||||||
|
return hash1 % ADDRMAN_BUCKET_SIZE;
|
||||||
|
}
|
||||||
|
|
||||||
bool CAddrInfo::IsTerrible(int64_t nNow) const
|
bool CAddrInfo::IsTerrible(int64_t nNow) const
|
||||||
{
|
{
|
||||||
if (nLastTry && nLastTry >= nNow - 60) // never remove things tried in the last minute
|
if (nLastTry && nLastTry >= nNow - 60) // never remove things tried in the last minute
|
||||||
|
@ -128,85 +137,44 @@ void CAddrMan::SwapRandom(unsigned int nRndPos1, unsigned int nRndPos2)
|
||||||
vRandom[nRndPos2] = nId1;
|
vRandom[nRndPos2] = nId1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int CAddrMan::SelectTried(int nKBucket)
|
void CAddrMan::Delete(int nId)
|
||||||
{
|
{
|
||||||
std::vector<int>& vTried = vvTried[nKBucket];
|
assert(mapInfo.count(nId) != 0);
|
||||||
|
CAddrInfo& info = mapInfo[nId];
|
||||||
|
assert(!info.fInTried);
|
||||||
|
assert(info.nRefCount == 0);
|
||||||
|
|
||||||
// randomly shuffle the first few elements (using the entire list)
|
SwapRandom(info.nRandomPos, vRandom.size() - 1);
|
||||||
// find the least recently tried among them
|
vRandom.pop_back();
|
||||||
int64_t nOldest = -1;
|
mapAddr.erase(info);
|
||||||
int nOldestPos = -1;
|
mapInfo.erase(nId);
|
||||||
for (unsigned int i = 0; i < ADDRMAN_TRIED_ENTRIES_INSPECT_ON_EVICT && i < vTried.size(); i++) {
|
nNew--;
|
||||||
int nPos = GetRandInt(vTried.size() - i) + i;
|
|
||||||
int nTemp = vTried[nPos];
|
|
||||||
vTried[nPos] = vTried[i];
|
|
||||||
vTried[i] = nTemp;
|
|
||||||
assert(nOldest == -1 || mapInfo.count(nTemp) == 1);
|
|
||||||
if (nOldest == -1 || mapInfo[nTemp].nLastSuccess < mapInfo[nOldest].nLastSuccess) {
|
|
||||||
nOldest = nTemp;
|
|
||||||
nOldestPos = nPos;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nOldestPos;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int CAddrMan::ShrinkNew(int nUBucket)
|
void CAddrMan::ClearNew(int nUBucket, int nUBucketPos)
|
||||||
{
|
{
|
||||||
assert(nUBucket >= 0 && (unsigned int)nUBucket < vvNew.size());
|
// if there is an entry in the specified bucket, delete it.
|
||||||
std::set<int>& vNew = vvNew[nUBucket];
|
if (vvNew[nUBucket][nUBucketPos] != -1) {
|
||||||
|
int nIdDelete = vvNew[nUBucket][nUBucketPos];
|
||||||
// first look for deletable items
|
CAddrInfo& infoDelete = mapInfo[nIdDelete];
|
||||||
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++) {
|
assert(infoDelete.nRefCount > 0);
|
||||||
assert(mapInfo.count(*it));
|
infoDelete.nRefCount--;
|
||||||
CAddrInfo& info = mapInfo[*it];
|
vvNew[nUBucket][nUBucketPos] = -1;
|
||||||
if (info.IsTerrible()) {
|
if (infoDelete.nRefCount == 0) {
|
||||||
if (--info.nRefCount == 0) {
|
Delete(nIdDelete);
|
||||||
SwapRandom(info.nRandomPos, vRandom.size() - 1);
|
|
||||||
vRandom.pop_back();
|
|
||||||
mapAddr.erase(info);
|
|
||||||
mapInfo.erase(*it);
|
|
||||||
nNew--;
|
|
||||||
}
|
|
||||||
vNew.erase(it);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// otherwise, select four randomly, and pick the oldest of those to replace
|
|
||||||
int n[4] = {GetRandInt(vNew.size()), GetRandInt(vNew.size()), GetRandInt(vNew.size()), GetRandInt(vNew.size())};
|
|
||||||
int nI = 0;
|
|
||||||
int nOldest = -1;
|
|
||||||
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++) {
|
|
||||||
if (nI == n[0] || nI == n[1] || nI == n[2] || nI == n[3]) {
|
|
||||||
assert(nOldest == -1 || mapInfo.count(*it) == 1);
|
|
||||||
if (nOldest == -1 || mapInfo[*it].nTime < mapInfo[nOldest].nTime)
|
|
||||||
nOldest = *it;
|
|
||||||
}
|
|
||||||
nI++;
|
|
||||||
}
|
|
||||||
assert(mapInfo.count(nOldest) == 1);
|
|
||||||
CAddrInfo& info = mapInfo[nOldest];
|
|
||||||
if (--info.nRefCount == 0) {
|
|
||||||
SwapRandom(info.nRandomPos, vRandom.size() - 1);
|
|
||||||
vRandom.pop_back();
|
|
||||||
mapAddr.erase(info);
|
|
||||||
mapInfo.erase(nOldest);
|
|
||||||
nNew--;
|
|
||||||
}
|
|
||||||
vNew.erase(nOldest);
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
void CAddrMan::MakeTried(CAddrInfo& info, int nId)
|
||||||
{
|
{
|
||||||
assert(vvNew[nOrigin].count(nId) == 1);
|
|
||||||
|
|
||||||
// remove the entry from all new buckets
|
// remove the entry from all new buckets
|
||||||
for (std::vector<std::set<int> >::iterator it = vvNew.begin(); it != vvNew.end(); it++) {
|
for (int bucket = 0; bucket < ADDRMAN_NEW_BUCKET_COUNT; bucket++) {
|
||||||
if ((*it).erase(nId))
|
int pos = info.GetBucketPosition(nKey, true, bucket);
|
||||||
|
if (vvNew[bucket][pos] == nId) {
|
||||||
|
vvNew[bucket][pos] = -1;
|
||||||
info.nRefCount--;
|
info.nRefCount--;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
nNew--;
|
nNew--;
|
||||||
|
|
||||||
|
@ -214,44 +182,36 @@ void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
||||||
|
|
||||||
// which tried bucket to move the entry to
|
// which tried bucket to move the entry to
|
||||||
int nKBucket = info.GetTriedBucket(nKey);
|
int nKBucket = info.GetTriedBucket(nKey);
|
||||||
std::vector<int>& vTried = vvTried[nKBucket];
|
int nKBucketPos = info.GetBucketPosition(nKey, false, nKBucket);
|
||||||
|
|
||||||
// first check whether there is place to just add it
|
// first make space to add it (the existing tried entry there is moved to new, deleting whatever is there).
|
||||||
if (vTried.size() < ADDRMAN_TRIED_BUCKET_SIZE) {
|
if (vvTried[nKBucket][nKBucketPos] != -1) {
|
||||||
vTried.push_back(nId);
|
// find an item to evict
|
||||||
nTried++;
|
int nIdEvict = vvTried[nKBucket][nKBucketPos];
|
||||||
info.fInTried = true;
|
assert(mapInfo.count(nIdEvict) == 1);
|
||||||
return;
|
CAddrInfo& infoOld = mapInfo[nIdEvict];
|
||||||
|
|
||||||
|
// Remove the to-be-evicted item from the tried set.
|
||||||
|
infoOld.fInTried = false;
|
||||||
|
vvTried[nKBucket][nKBucketPos] = -1;
|
||||||
|
nTried--;
|
||||||
|
|
||||||
|
// find which new bucket it belongs to
|
||||||
|
int nUBucket = infoOld.GetNewBucket(nKey);
|
||||||
|
int nUBucketPos = infoOld.GetBucketPosition(nKey, true, nUBucket);
|
||||||
|
ClearNew(nUBucket, nUBucketPos);
|
||||||
|
assert(vvNew[nUBucket][nUBucketPos] == -1);
|
||||||
|
|
||||||
|
// Enter it into the new set again.
|
||||||
|
infoOld.nRefCount = 1;
|
||||||
|
vvNew[nUBucket][nUBucketPos] = nIdEvict;
|
||||||
|
nNew++;
|
||||||
}
|
}
|
||||||
|
assert(vvTried[nKBucket][nKBucketPos] == -1);
|
||||||
|
|
||||||
// otherwise, find an item to evict
|
vvTried[nKBucket][nKBucketPos] = nId;
|
||||||
int nPos = SelectTried(nKBucket);
|
nTried++;
|
||||||
|
|
||||||
// find which new bucket it belongs to
|
|
||||||
assert(mapInfo.count(vTried[nPos]) == 1);
|
|
||||||
int nUBucket = mapInfo[vTried[nPos]].GetNewBucket(nKey);
|
|
||||||
std::set<int>& vNew = vvNew[nUBucket];
|
|
||||||
|
|
||||||
// remove the to-be-replaced tried entry from the tried set
|
|
||||||
CAddrInfo& infoOld = mapInfo[vTried[nPos]];
|
|
||||||
infoOld.fInTried = false;
|
|
||||||
infoOld.nRefCount = 1;
|
|
||||||
// do not update nTried, as we are going to move something else there immediately
|
|
||||||
|
|
||||||
// check whether there is place in that one,
|
|
||||||
if (vNew.size() < ADDRMAN_NEW_BUCKET_SIZE) {
|
|
||||||
// if so, move it back there
|
|
||||||
vNew.insert(vTried[nPos]);
|
|
||||||
} else {
|
|
||||||
// otherwise, move it to the new bucket nId came from (there is certainly place there)
|
|
||||||
vvNew[nOrigin].insert(vTried[nPos]);
|
|
||||||
}
|
|
||||||
nNew++;
|
|
||||||
|
|
||||||
vTried[nPos] = nId;
|
|
||||||
// we just overwrote an entry in vTried; no need to update nTried
|
|
||||||
info.fInTried = true;
|
info.fInTried = true;
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CAddrMan::Good_(const CService& addr, int64_t nTime)
|
void CAddrMan::Good_(const CService& addr, int64_t nTime)
|
||||||
|
@ -281,12 +241,12 @@ void CAddrMan::Good_(const CService& addr, int64_t nTime)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// find a bucket it is in now
|
// find a bucket it is in now
|
||||||
int nRnd = GetRandInt(vvNew.size());
|
int nRnd = GetRandInt(ADDRMAN_NEW_BUCKET_COUNT);
|
||||||
int nUBucket = -1;
|
int nUBucket = -1;
|
||||||
for (unsigned int n = 0; n < vvNew.size(); n++) {
|
for (unsigned int n = 0; n < ADDRMAN_NEW_BUCKET_COUNT; n++) {
|
||||||
int nB = (n + nRnd) % vvNew.size();
|
int nB = (n + nRnd) % ADDRMAN_NEW_BUCKET_COUNT;
|
||||||
std::set<int>& vNew = vvNew[nB];
|
int nBpos = info.GetBucketPosition(nKey, true, nB);
|
||||||
if (vNew.count(nId)) {
|
if (vvNew[nB][nBpos] == nId) {
|
||||||
nUBucket = nB;
|
nUBucket = nB;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -300,7 +260,7 @@ void CAddrMan::Good_(const CService& addr, int64_t nTime)
|
||||||
LogPrint("addrman", "Moving %s to tried\n", addr.ToString());
|
LogPrint("addrman", "Moving %s to tried\n", addr.ToString());
|
||||||
|
|
||||||
// move nId to the tried tables
|
// move nId to the tried tables
|
||||||
MakeTried(info, nId, nUBucket);
|
MakeTried(info, nId);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CAddrMan::Add_(const CAddress& addr, const CNetAddr& source, int64_t nTimePenalty)
|
bool CAddrMan::Add_(const CAddress& addr, const CNetAddr& source, int64_t nTimePenalty)
|
||||||
|
@ -348,12 +308,25 @@ bool CAddrMan::Add_(const CAddress& addr, const CNetAddr& source, int64_t nTimeP
|
||||||
}
|
}
|
||||||
|
|
||||||
int nUBucket = pinfo->GetNewBucket(nKey, source);
|
int nUBucket = pinfo->GetNewBucket(nKey, source);
|
||||||
std::set<int>& vNew = vvNew[nUBucket];
|
int nUBucketPos = pinfo->GetBucketPosition(nKey, true, nUBucket);
|
||||||
if (!vNew.count(nId)) {
|
if (vvNew[nUBucket][nUBucketPos] != nId) {
|
||||||
pinfo->nRefCount++;
|
bool fInsert = vvNew[nUBucket][nUBucketPos] == -1;
|
||||||
if (vNew.size() == ADDRMAN_NEW_BUCKET_SIZE)
|
if (!fInsert) {
|
||||||
ShrinkNew(nUBucket);
|
CAddrInfo& infoExisting = mapInfo[vvNew[nUBucket][nUBucketPos]];
|
||||||
vvNew[nUBucket].insert(nId);
|
if (infoExisting.IsTerrible() || (infoExisting.nRefCount > 1 && pinfo->nRefCount == 0)) {
|
||||||
|
// Overwrite the existing new table entry.
|
||||||
|
fInsert = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (fInsert) {
|
||||||
|
ClearNew(nUBucket, nUBucketPos);
|
||||||
|
pinfo->nRefCount++;
|
||||||
|
vvNew[nUBucket][nUBucketPos] = nId;
|
||||||
|
} else {
|
||||||
|
if (pinfo->nRefCount == 0) {
|
||||||
|
Delete(nId);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return fNew;
|
return fNew;
|
||||||
}
|
}
|
||||||
|
@ -388,13 +361,13 @@ CAddress CAddrMan::Select_(int nUnkBias)
|
||||||
// use a tried node
|
// use a tried node
|
||||||
double fChanceFactor = 1.0;
|
double fChanceFactor = 1.0;
|
||||||
while (1) {
|
while (1) {
|
||||||
int nKBucket = GetRandInt(vvTried.size());
|
int nKBucket = GetRandInt(ADDRMAN_TRIED_BUCKET_COUNT);
|
||||||
std::vector<int>& vTried = vvTried[nKBucket];
|
int nKBucketPos = GetRandInt(ADDRMAN_BUCKET_SIZE);
|
||||||
if (vTried.size() == 0)
|
if (vvTried[nKBucket][nKBucketPos] == -1)
|
||||||
continue;
|
continue;
|
||||||
int nPos = GetRandInt(vTried.size());
|
int nId = vvTried[nKBucket][nKBucketPos];
|
||||||
assert(mapInfo.count(vTried[nPos]) == 1);
|
assert(mapInfo.count(nId) == 1);
|
||||||
CAddrInfo& info = mapInfo[vTried[nPos]];
|
CAddrInfo& info = mapInfo[nId];
|
||||||
if (GetRandInt(1 << 30) < fChanceFactor * info.GetChance() * (1 << 30))
|
if (GetRandInt(1 << 30) < fChanceFactor * info.GetChance() * (1 << 30))
|
||||||
return info;
|
return info;
|
||||||
fChanceFactor *= 1.2;
|
fChanceFactor *= 1.2;
|
||||||
|
@ -403,16 +376,13 @@ CAddress CAddrMan::Select_(int nUnkBias)
|
||||||
// use a new node
|
// use a new node
|
||||||
double fChanceFactor = 1.0;
|
double fChanceFactor = 1.0;
|
||||||
while (1) {
|
while (1) {
|
||||||
int nUBucket = GetRandInt(vvNew.size());
|
int nUBucket = GetRandInt(ADDRMAN_NEW_BUCKET_COUNT);
|
||||||
std::set<int>& vNew = vvNew[nUBucket];
|
int nUBucketPos = GetRandInt(ADDRMAN_BUCKET_SIZE);
|
||||||
if (vNew.size() == 0)
|
if (vvNew[nUBucket][nUBucketPos] == -1)
|
||||||
continue;
|
continue;
|
||||||
int nPos = GetRandInt(vNew.size());
|
int nId = vvNew[nUBucket][nUBucketPos];
|
||||||
std::set<int>::iterator it = vNew.begin();
|
assert(mapInfo.count(nId) == 1);
|
||||||
while (nPos--)
|
CAddrInfo& info = mapInfo[nId];
|
||||||
it++;
|
|
||||||
assert(mapInfo.count(*it) == 1);
|
|
||||||
CAddrInfo& info = mapInfo[*it];
|
|
||||||
if (GetRandInt(1 << 30) < fChanceFactor * info.GetChance() * (1 << 30))
|
if (GetRandInt(1 << 30) < fChanceFactor * info.GetChance() * (1 << 30))
|
||||||
return info;
|
return info;
|
||||||
fChanceFactor *= 1.2;
|
fChanceFactor *= 1.2;
|
||||||
|
@ -460,22 +430,30 @@ int CAddrMan::Check_()
|
||||||
if (mapNew.size() != nNew)
|
if (mapNew.size() != nNew)
|
||||||
return -10;
|
return -10;
|
||||||
|
|
||||||
for (int n = 0; n < vvTried.size(); n++) {
|
for (int n = 0; n < ADDRMAN_TRIED_BUCKET_COUNT; n++) {
|
||||||
std::vector<int>& vTried = vvTried[n];
|
for (int i = 0; i < ADDRMAN_BUCKET_SIZE; i++) {
|
||||||
for (std::vector<int>::iterator it = vTried.begin(); it != vTried.end(); it++) {
|
if (vvTried[n][i] != -1) {
|
||||||
if (!setTried.count(*it))
|
if (!setTried.count(vvTried[n][i]))
|
||||||
return -11;
|
return -11;
|
||||||
setTried.erase(*it);
|
if (mapInfo[vvTried[n][i]].GetTriedBucket(nKey) != n)
|
||||||
|
return -17;
|
||||||
|
if (mapInfo[vvTried[n][i]].GetBucketPosition(nKey, false, n) != i)
|
||||||
|
return -18;
|
||||||
|
setTried.erase(vvTried[n][i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int n = 0; n < vvNew.size(); n++) {
|
for (int n = 0; n < ADDRMAN_NEW_BUCKET_COUNT; n++) {
|
||||||
std::set<int>& vNew = vvNew[n];
|
for (int i = 0; i < ADDRMAN_BUCKET_SIZE; i++) {
|
||||||
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++) {
|
if (vvNew[n][i] != -1) {
|
||||||
if (!mapNew.count(*it))
|
if (!mapNew.count(vvNew[n][i]))
|
||||||
return -12;
|
return -12;
|
||||||
if (--mapNew[*it] == 0)
|
if (mapInfo[vvNew[n][i]].GetBucketPosition(nKey, true, n) != i)
|
||||||
mapNew.erase(*it);
|
return -19;
|
||||||
|
if (--mapNew[vvNew[n][i]] == 0)
|
||||||
|
mapNew.erase(vvNew[n][i]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
169
src/addrman.h
169
src/addrman.h
|
@ -10,7 +10,6 @@
|
||||||
#include "random.h"
|
#include "random.h"
|
||||||
#include "sync.h"
|
#include "sync.h"
|
||||||
#include "timedata.h"
|
#include "timedata.h"
|
||||||
#include "uint256.h"
|
|
||||||
#include "util.h"
|
#include "util.h"
|
||||||
|
|
||||||
#include <map>
|
#include <map>
|
||||||
|
@ -91,6 +90,9 @@ public:
|
||||||
return GetNewBucket(nKey, source);
|
return GetNewBucket(nKey, source);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//! Calculate in which position of a bucket to store this entry.
|
||||||
|
int GetBucketPosition(const uint256 &nKey, bool fNew, int nBucket) const;
|
||||||
|
|
||||||
//! Determine whether the statistics about this entry are bad enough so that it can just be deleted
|
//! Determine whether the statistics about this entry are bad enough so that it can just be deleted
|
||||||
bool IsTerrible(int64_t nNow = GetAdjustedTime()) const;
|
bool IsTerrible(int64_t nNow = GetAdjustedTime()) const;
|
||||||
|
|
||||||
|
@ -128,14 +130,11 @@ public:
|
||||||
//! total number of buckets for tried addresses
|
//! total number of buckets for tried addresses
|
||||||
#define ADDRMAN_TRIED_BUCKET_COUNT 64
|
#define ADDRMAN_TRIED_BUCKET_COUNT 64
|
||||||
|
|
||||||
//! maximum allowed number of entries in buckets for tried addresses
|
|
||||||
#define ADDRMAN_TRIED_BUCKET_SIZE 64
|
|
||||||
|
|
||||||
//! total number of buckets for new addresses
|
//! total number of buckets for new addresses
|
||||||
#define ADDRMAN_NEW_BUCKET_COUNT 256
|
#define ADDRMAN_NEW_BUCKET_COUNT 256
|
||||||
|
|
||||||
//! maximum allowed number of entries in buckets for new addresses
|
//! maximum allowed number of entries in buckets for new and tried addresses
|
||||||
#define ADDRMAN_NEW_BUCKET_SIZE 64
|
#define ADDRMAN_BUCKET_SIZE 64
|
||||||
|
|
||||||
//! over how many buckets entries with tried addresses from a single group (/16 for IPv4) are spread
|
//! over how many buckets entries with tried addresses from a single group (/16 for IPv4) are spread
|
||||||
#define ADDRMAN_TRIED_BUCKETS_PER_GROUP 4
|
#define ADDRMAN_TRIED_BUCKETS_PER_GROUP 4
|
||||||
|
@ -146,9 +145,6 @@ public:
|
||||||
//! in how many buckets for entries with new addresses a single address may occur
|
//! in how many buckets for entries with new addresses a single address may occur
|
||||||
#define ADDRMAN_NEW_BUCKETS_PER_ADDRESS 4
|
#define ADDRMAN_NEW_BUCKETS_PER_ADDRESS 4
|
||||||
|
|
||||||
//! how many entries in a bucket with tried addresses are inspected, when selecting one to replace
|
|
||||||
#define ADDRMAN_TRIED_ENTRIES_INSPECT_ON_EVICT 4
|
|
||||||
|
|
||||||
//! how old addresses can maximally be
|
//! how old addresses can maximally be
|
||||||
#define ADDRMAN_HORIZON_DAYS 30
|
#define ADDRMAN_HORIZON_DAYS 30
|
||||||
|
|
||||||
|
@ -195,13 +191,13 @@ private:
|
||||||
int nTried;
|
int nTried;
|
||||||
|
|
||||||
//! list of "tried" buckets
|
//! list of "tried" buckets
|
||||||
std::vector<std::vector<int> > vvTried;
|
int vvTried[ADDRMAN_TRIED_BUCKET_COUNT][ADDRMAN_BUCKET_SIZE];
|
||||||
|
|
||||||
//! number of (unique) "new" entries
|
//! number of (unique) "new" entries
|
||||||
int nNew;
|
int nNew;
|
||||||
|
|
||||||
//! list of "new" buckets
|
//! list of "new" buckets
|
||||||
std::vector<std::set<int> > vvNew;
|
int vvNew[ADDRMAN_NEW_BUCKET_COUNT][ADDRMAN_BUCKET_SIZE];
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
|
@ -215,17 +211,14 @@ protected:
|
||||||
//! Swap two elements in vRandom.
|
//! Swap two elements in vRandom.
|
||||||
void SwapRandom(unsigned int nRandomPos1, unsigned int nRandomPos2);
|
void SwapRandom(unsigned int nRandomPos1, unsigned int nRandomPos2);
|
||||||
|
|
||||||
//! Return position in given bucket to replace.
|
|
||||||
int SelectTried(int nKBucket);
|
|
||||||
|
|
||||||
//! Remove an element from a "new" bucket.
|
|
||||||
//! This is the only place where actual deletions occur.
|
|
||||||
//! Elements are never deleted while in the "tried" table, only possibly evicted back to the "new" table.
|
|
||||||
int ShrinkNew(int nUBucket);
|
|
||||||
|
|
||||||
//! Move an entry from the "new" table(s) to the "tried" table
|
//! Move an entry from the "new" table(s) to the "tried" table
|
||||||
//! @pre vvUnkown[nOrigin].count(nId) != 0
|
void MakeTried(CAddrInfo& info, int nId);
|
||||||
void MakeTried(CAddrInfo& info, int nId, int nOrigin);
|
|
||||||
|
//! Delete an entry. It must not be in tried, and have refcount 0.
|
||||||
|
void Delete(int nId);
|
||||||
|
|
||||||
|
//! Clear a position in a "new" table. This is the only place where entries are actually deleted.
|
||||||
|
void ClearNew(int nUBucket, int nUBucketPos);
|
||||||
|
|
||||||
//! Mark an entry "good", possibly moving it from "new" to "tried".
|
//! Mark an entry "good", possibly moving it from "new" to "tried".
|
||||||
void Good_(const CService &addr, int64_t nTime);
|
void Good_(const CService &addr, int64_t nTime);
|
||||||
|
@ -254,17 +247,21 @@ protected:
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
* serialized format:
|
* serialized format:
|
||||||
* * version byte (currently 0)
|
* * version byte (currently 1)
|
||||||
* * nKey
|
* * 0x20 + nKey (serialized as if it were a vector, for backward compatibility)
|
||||||
* * nNew
|
* * nNew
|
||||||
* * nTried
|
* * nTried
|
||||||
* * number of "new" buckets
|
* * number of "new" buckets XOR 2**30
|
||||||
* * all nNew addrinfos in vvNew
|
* * all nNew addrinfos in vvNew
|
||||||
* * all nTried addrinfos in vvTried
|
* * all nTried addrinfos in vvTried
|
||||||
* * for each bucket:
|
* * for each bucket:
|
||||||
* * number of elements
|
* * number of elements
|
||||||
* * for each element: index
|
* * for each element: index
|
||||||
*
|
*
|
||||||
|
* 2**30 is xorred with the number of buckets to make addrman deserializer v0 detect it
|
||||||
|
* as incompatible. This is necessary because it did not check the version number on
|
||||||
|
* deserialization.
|
||||||
|
*
|
||||||
* Notice that vvTried, mapAddr and vVector are never encoded explicitly;
|
* Notice that vvTried, mapAddr and vVector are never encoded explicitly;
|
||||||
* they are instead reconstructed from the other information.
|
* they are instead reconstructed from the other information.
|
||||||
*
|
*
|
||||||
|
@ -276,49 +273,53 @@ public:
|
||||||
*
|
*
|
||||||
* We don't use ADD_SERIALIZE_METHODS since the serialization and deserialization code has
|
* We don't use ADD_SERIALIZE_METHODS since the serialization and deserialization code has
|
||||||
* very little in common.
|
* very little in common.
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
template<typename Stream>
|
template<typename Stream>
|
||||||
void Serialize(Stream &s, int nType, int nVersionDummy) const
|
void Serialize(Stream &s, int nType, int nVersionDummy) const
|
||||||
{
|
{
|
||||||
LOCK(cs);
|
LOCK(cs);
|
||||||
|
|
||||||
unsigned char nVersion = 0;
|
unsigned char nVersion = 1;
|
||||||
s << nVersion;
|
s << nVersion;
|
||||||
s << ((unsigned char)32);
|
s << ((unsigned char)32);
|
||||||
s << nKey;
|
s << nKey;
|
||||||
s << nNew;
|
s << nNew;
|
||||||
s << nTried;
|
s << nTried;
|
||||||
|
|
||||||
int nUBuckets = ADDRMAN_NEW_BUCKET_COUNT;
|
int nUBuckets = ADDRMAN_NEW_BUCKET_COUNT ^ (1 << 30);
|
||||||
s << nUBuckets;
|
s << nUBuckets;
|
||||||
std::map<int, int> mapUnkIds;
|
std::map<int, int> mapUnkIds;
|
||||||
int nIds = 0;
|
int nIds = 0;
|
||||||
for (std::map<int, CAddrInfo>::const_iterator it = mapInfo.begin(); it != mapInfo.end(); it++) {
|
for (std::map<int, CAddrInfo>::const_iterator it = mapInfo.begin(); it != mapInfo.end(); it++) {
|
||||||
if (nIds == nNew) break; // this means nNew was wrong, oh ow
|
|
||||||
mapUnkIds[(*it).first] = nIds;
|
mapUnkIds[(*it).first] = nIds;
|
||||||
const CAddrInfo &info = (*it).second;
|
const CAddrInfo &info = (*it).second;
|
||||||
if (info.nRefCount) {
|
if (info.nRefCount) {
|
||||||
|
assert(nIds != nNew); // this means nNew was wrong, oh ow
|
||||||
s << info;
|
s << info;
|
||||||
nIds++;
|
nIds++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
nIds = 0;
|
nIds = 0;
|
||||||
for (std::map<int, CAddrInfo>::const_iterator it = mapInfo.begin(); it != mapInfo.end(); it++) {
|
for (std::map<int, CAddrInfo>::const_iterator it = mapInfo.begin(); it != mapInfo.end(); it++) {
|
||||||
if (nIds == nTried) break; // this means nTried was wrong, oh ow
|
|
||||||
const CAddrInfo &info = (*it).second;
|
const CAddrInfo &info = (*it).second;
|
||||||
if (info.fInTried) {
|
if (info.fInTried) {
|
||||||
|
assert(nIds != nTried); // this means nTried was wrong, oh ow
|
||||||
s << info;
|
s << info;
|
||||||
nIds++;
|
nIds++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (std::vector<std::set<int> >::const_iterator it = vvNew.begin(); it != vvNew.end(); it++) {
|
for (int bucket = 0; bucket < ADDRMAN_NEW_BUCKET_COUNT; bucket++) {
|
||||||
const std::set<int> &vNew = (*it);
|
int nSize = 0;
|
||||||
int nSize = vNew.size();
|
for (int i = 0; i < ADDRMAN_BUCKET_SIZE; i++) {
|
||||||
|
if (vvNew[bucket][i] != -1)
|
||||||
|
nSize++;
|
||||||
|
}
|
||||||
s << nSize;
|
s << nSize;
|
||||||
for (std::set<int>::const_iterator it2 = vNew.begin(); it2 != vNew.end(); it2++) {
|
for (int i = 0; i < ADDRMAN_BUCKET_SIZE; i++) {
|
||||||
int nIndex = mapUnkIds[*it2];
|
if (vvNew[bucket][i] != -1) {
|
||||||
s << nIndex;
|
int nIndex = mapUnkIds[vvNew[bucket][i]];
|
||||||
|
s << nIndex;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -328,67 +329,97 @@ public:
|
||||||
{
|
{
|
||||||
LOCK(cs);
|
LOCK(cs);
|
||||||
|
|
||||||
|
Clear();
|
||||||
|
|
||||||
unsigned char nVersion;
|
unsigned char nVersion;
|
||||||
s >> nVersion;
|
s >> nVersion;
|
||||||
unsigned char nKeySize;
|
unsigned char nKeySize;
|
||||||
s >> nKeySize;
|
s >> nKeySize;
|
||||||
if (nKeySize != 32) throw std::ios_base::failure("Incorrect keysize in addrman");
|
if (nKeySize != 32) throw std::ios_base::failure("Incorrect keysize in addrman deserialization");
|
||||||
s >> nKey;
|
s >> nKey;
|
||||||
s >> nNew;
|
s >> nNew;
|
||||||
s >> nTried;
|
s >> nTried;
|
||||||
|
|
||||||
int nUBuckets = 0;
|
int nUBuckets = 0;
|
||||||
s >> nUBuckets;
|
s >> nUBuckets;
|
||||||
nIdCount = 0;
|
if (nVersion != 0) {
|
||||||
mapInfo.clear();
|
nUBuckets ^= (1 << 30);
|
||||||
mapAddr.clear();
|
}
|
||||||
vRandom.clear();
|
|
||||||
vvTried = std::vector<std::vector<int> >(ADDRMAN_TRIED_BUCKET_COUNT, std::vector<int>(0));
|
// Deserialize entries from the new table.
|
||||||
vvNew = std::vector<std::set<int> >(ADDRMAN_NEW_BUCKET_COUNT, std::set<int>());
|
|
||||||
for (int n = 0; n < nNew; n++) {
|
for (int n = 0; n < nNew; n++) {
|
||||||
CAddrInfo &info = mapInfo[n];
|
CAddrInfo &info = mapInfo[n];
|
||||||
s >> info;
|
s >> info;
|
||||||
mapAddr[info] = n;
|
mapAddr[info] = n;
|
||||||
info.nRandomPos = vRandom.size();
|
info.nRandomPos = vRandom.size();
|
||||||
vRandom.push_back(n);
|
vRandom.push_back(n);
|
||||||
if (nUBuckets != ADDRMAN_NEW_BUCKET_COUNT) {
|
if (nVersion != 1 || nUBuckets != ADDRMAN_NEW_BUCKET_COUNT) {
|
||||||
vvNew[info.GetNewBucket(nKey)].insert(n);
|
// In case the new table data cannot be used (nVersion unknown, or bucket count wrong),
|
||||||
info.nRefCount++;
|
// immediately try to give them a reference based on their primary source address.
|
||||||
|
int nUBucket = info.GetNewBucket(nKey);
|
||||||
|
int nUBucketPos = info.GetBucketPosition(nKey, true, nUBucket);
|
||||||
|
if (vvNew[nUBucket][nUBucketPos] == -1) {
|
||||||
|
vvNew[nUBucket][nUBucketPos] = n;
|
||||||
|
info.nRefCount++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
nIdCount = nNew;
|
nIdCount = nNew;
|
||||||
|
|
||||||
|
// Deserialize entries from the tried table.
|
||||||
int nLost = 0;
|
int nLost = 0;
|
||||||
for (int n = 0; n < nTried; n++) {
|
for (int n = 0; n < nTried; n++) {
|
||||||
CAddrInfo info;
|
CAddrInfo info;
|
||||||
s >> info;
|
s >> info;
|
||||||
std::vector<int> &vTried = vvTried[info.GetTriedBucket(nKey)];
|
int nKBucket = info.GetTriedBucket(nKey);
|
||||||
if (vTried.size() < ADDRMAN_TRIED_BUCKET_SIZE) {
|
int nKBucketPos = info.GetBucketPosition(nKey, false, nKBucket);
|
||||||
|
if (vvTried[nKBucket][nKBucketPos] == -1) {
|
||||||
info.nRandomPos = vRandom.size();
|
info.nRandomPos = vRandom.size();
|
||||||
info.fInTried = true;
|
info.fInTried = true;
|
||||||
vRandom.push_back(nIdCount);
|
vRandom.push_back(nIdCount);
|
||||||
mapInfo[nIdCount] = info;
|
mapInfo[nIdCount] = info;
|
||||||
mapAddr[info] = nIdCount;
|
mapAddr[info] = nIdCount;
|
||||||
vTried.push_back(nIdCount);
|
vvTried[nKBucket][nKBucketPos] = nIdCount;
|
||||||
nIdCount++;
|
nIdCount++;
|
||||||
} else {
|
} else {
|
||||||
nLost++;
|
nLost++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
nTried -= nLost;
|
nTried -= nLost;
|
||||||
for (int b = 0; b < nUBuckets; b++) {
|
|
||||||
std::set<int> &vNew = vvNew[b];
|
// Deserialize positions in the new table (if possible).
|
||||||
|
for (int bucket = 0; bucket < nUBuckets; bucket++) {
|
||||||
int nSize = 0;
|
int nSize = 0;
|
||||||
s >> nSize;
|
s >> nSize;
|
||||||
for (int n = 0; n < nSize; n++) {
|
for (int n = 0; n < nSize; n++) {
|
||||||
int nIndex = 0;
|
int nIndex = 0;
|
||||||
s >> nIndex;
|
s >> nIndex;
|
||||||
CAddrInfo &info = mapInfo[nIndex];
|
if (nIndex >= 0 && nIndex < nNew) {
|
||||||
if (nUBuckets == ADDRMAN_NEW_BUCKET_COUNT && info.nRefCount < ADDRMAN_NEW_BUCKETS_PER_ADDRESS) {
|
CAddrInfo &info = mapInfo[nIndex];
|
||||||
info.nRefCount++;
|
int nUBucketPos = info.GetBucketPosition(nKey, true, bucket);
|
||||||
vNew.insert(nIndex);
|
if (nVersion == 1 && nUBuckets == ADDRMAN_NEW_BUCKET_COUNT && vvNew[bucket][nUBucketPos] == -1 && info.nRefCount < ADDRMAN_NEW_BUCKETS_PER_ADDRESS) {
|
||||||
|
info.nRefCount++;
|
||||||
|
vvNew[bucket][nUBucketPos] = nIndex;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Prune new entries with refcount 0 (as a result of collisions).
|
||||||
|
int nLostUnk = 0;
|
||||||
|
for (std::map<int, CAddrInfo>::const_iterator it = mapInfo.begin(); it != mapInfo.end(); ) {
|
||||||
|
if (it->second.fInTried == false && it->second.nRefCount == 0) {
|
||||||
|
std::map<int, CAddrInfo>::const_iterator itCopy = it++;
|
||||||
|
Delete(itCopy->first);
|
||||||
|
nLostUnk++;
|
||||||
|
} else {
|
||||||
|
it++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (nLost + nLostUnk > 0) {
|
||||||
|
LogPrint("addrman", "addrman lost %i new and %i tried addresses due to collisions\n", nLostUnk, nLost);
|
||||||
|
}
|
||||||
|
|
||||||
|
Check();
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned int GetSerializeSize(int nType, int nVersion) const
|
unsigned int GetSerializeSize(int nType, int nVersion) const
|
||||||
|
@ -396,18 +427,34 @@ public:
|
||||||
return (CSizeComputer(nType, nVersion) << *this).size();
|
return (CSizeComputer(nType, nVersion) << *this).size();
|
||||||
}
|
}
|
||||||
|
|
||||||
CAddrMan() : vRandom(0), vvTried(ADDRMAN_TRIED_BUCKET_COUNT, std::vector<int>(0)), vvNew(ADDRMAN_NEW_BUCKET_COUNT, std::set<int>())
|
void Clear()
|
||||||
{
|
{
|
||||||
nKey = GetRandHash();
|
std::vector<int>().swap(vRandom);
|
||||||
|
nKey = GetRandHash();
|
||||||
|
for (size_t bucket = 0; bucket < ADDRMAN_NEW_BUCKET_COUNT; bucket++) {
|
||||||
|
for (size_t entry = 0; entry < ADDRMAN_BUCKET_SIZE; entry++) {
|
||||||
|
vvNew[bucket][entry] = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (size_t bucket = 0; bucket < ADDRMAN_TRIED_BUCKET_COUNT; bucket++) {
|
||||||
|
for (size_t entry = 0; entry < ADDRMAN_BUCKET_SIZE; entry++) {
|
||||||
|
vvTried[bucket][entry] = -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
nIdCount = 0;
|
nIdCount = 0;
|
||||||
nTried = 0;
|
nTried = 0;
|
||||||
nNew = 0;
|
nNew = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
CAddrMan()
|
||||||
|
{
|
||||||
|
Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
~CAddrMan()
|
~CAddrMan()
|
||||||
{
|
{
|
||||||
nKey.SetNull();
|
nKey.SetNull();
|
||||||
}
|
}
|
||||||
|
|
||||||
//! Return the number of (unique) addresses in all tables.
|
//! Return the number of (unique) addresses in all tables.
|
||||||
|
|
Loading…
Add table
Reference in a new issue