mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-26 19:23:26 -03:00
Merge pull request #4933
20e01b1
Apply clang-format on some infrequently-updated files (Pieter Wuille)
This commit is contained in:
commit
934b153a2c
30 changed files with 845 additions and 742 deletions
225
src/addrman.cpp
225
src/addrman.cpp
|
@ -9,7 +9,7 @@
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
int CAddrInfo::GetTriedBucket(const std::vector<unsigned char> &nKey) const
|
int CAddrInfo::GetTriedBucket(const std::vector<unsigned char>& nKey) const
|
||||||
{
|
{
|
||||||
CDataStream ss1(SER_GETHASH, 0);
|
CDataStream ss1(SER_GETHASH, 0);
|
||||||
std::vector<unsigned char> vchKey = GetKey();
|
std::vector<unsigned char> vchKey = GetKey();
|
||||||
|
@ -23,7 +23,7 @@ int CAddrInfo::GetTriedBucket(const std::vector<unsigned char> &nKey) const
|
||||||
return hash2 % ADDRMAN_TRIED_BUCKET_COUNT;
|
return hash2 % ADDRMAN_TRIED_BUCKET_COUNT;
|
||||||
}
|
}
|
||||||
|
|
||||||
int CAddrInfo::GetNewBucket(const std::vector<unsigned char> &nKey, const CNetAddr& src) const
|
int CAddrInfo::GetNewBucket(const std::vector<unsigned char>& 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();
|
||||||
|
@ -39,19 +39,19 @@ int CAddrInfo::GetNewBucket(const std::vector<unsigned char> &nKey, const CNetAd
|
||||||
|
|
||||||
bool CAddrInfo::IsTerrible(int64_t nNow) const
|
bool CAddrInfo::IsTerrible(int64_t nNow) const
|
||||||
{
|
{
|
||||||
if (nLastTry && nLastTry >= nNow-60) // never remove things tried the last minute
|
if (nLastTry && nLastTry >= nNow - 60) // never remove things tried the last minute
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (nTime > nNow + 10*60) // came in a flying DeLorean
|
if (nTime > nNow + 10 * 60) // came in a flying DeLorean
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (nTime==0 || nNow-nTime > ADDRMAN_HORIZON_DAYS*24*60*60) // not seen in recent history
|
if (nTime == 0 || nNow - nTime > ADDRMAN_HORIZON_DAYS * 24 * 60 * 60) // not seen in recent history
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (nLastSuccess==0 && nAttempts>=ADDRMAN_RETRIES) // tried N times and never a success
|
if (nLastSuccess == 0 && nAttempts >= ADDRMAN_RETRIES) // tried N times and never a success
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (nNow-nLastSuccess > ADDRMAN_MIN_FAIL_DAYS*24*60*60 && nAttempts>=ADDRMAN_MAX_FAILURES) // N successive failures in the last week
|
if (nNow - nLastSuccess > ADDRMAN_MIN_FAIL_DAYS * 24 * 60 * 60 && nAttempts >= ADDRMAN_MAX_FAILURES) // N successive failures in the last week
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
@ -64,23 +64,25 @@ double CAddrInfo::GetChance(int64_t nNow) const
|
||||||
int64_t nSinceLastSeen = nNow - nTime;
|
int64_t nSinceLastSeen = nNow - nTime;
|
||||||
int64_t nSinceLastTry = nNow - nLastTry;
|
int64_t nSinceLastTry = nNow - nLastTry;
|
||||||
|
|
||||||
if (nSinceLastSeen < 0) nSinceLastSeen = 0;
|
if (nSinceLastSeen < 0)
|
||||||
if (nSinceLastTry < 0) nSinceLastTry = 0;
|
nSinceLastSeen = 0;
|
||||||
|
if (nSinceLastTry < 0)
|
||||||
|
nSinceLastTry = 0;
|
||||||
|
|
||||||
fChance *= 600.0 / (600.0 + nSinceLastSeen);
|
fChance *= 600.0 / (600.0 + nSinceLastSeen);
|
||||||
|
|
||||||
// deprioritize very recent attempts away
|
// deprioritize very recent attempts away
|
||||||
if (nSinceLastTry < 60*10)
|
if (nSinceLastTry < 60 * 10)
|
||||||
fChance *= 0.01;
|
fChance *= 0.01;
|
||||||
|
|
||||||
// deprioritize 50% after each failed attempt
|
// deprioritize 50% after each failed attempt
|
||||||
for (int n=0; n<nAttempts; n++)
|
for (int n = 0; n < nAttempts; n++)
|
||||||
fChance /= 1.5;
|
fChance /= 1.5;
|
||||||
|
|
||||||
return fChance;
|
return fChance;
|
||||||
}
|
}
|
||||||
|
|
||||||
CAddrInfo* CAddrMan::Find(const CNetAddr& addr, int *pnId)
|
CAddrInfo* CAddrMan::Find(const CNetAddr& addr, int* pnId)
|
||||||
{
|
{
|
||||||
std::map<CNetAddr, int>::iterator it = mapAddr.find(addr);
|
std::map<CNetAddr, int>::iterator it = mapAddr.find(addr);
|
||||||
if (it == mapAddr.end())
|
if (it == mapAddr.end())
|
||||||
|
@ -93,7 +95,7 @@ CAddrInfo* CAddrMan::Find(const CNetAddr& addr, int *pnId)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
CAddrInfo* CAddrMan::Create(const CAddress &addr, const CNetAddr &addrSource, int *pnId)
|
CAddrInfo* CAddrMan::Create(const CAddress& addr, const CNetAddr& addrSource, int* pnId)
|
||||||
{
|
{
|
||||||
int nId = nIdCount++;
|
int nId = nIdCount++;
|
||||||
mapInfo[nId] = CAddrInfo(addr, addrSource);
|
mapInfo[nId] = CAddrInfo(addr, addrSource);
|
||||||
|
@ -127,22 +129,21 @@ void CAddrMan::SwapRandom(unsigned int nRndPos1, unsigned int nRndPos2)
|
||||||
|
|
||||||
int CAddrMan::SelectTried(int nKBucket)
|
int CAddrMan::SelectTried(int nKBucket)
|
||||||
{
|
{
|
||||||
std::vector<int> &vTried = vvTried[nKBucket];
|
std::vector<int>& vTried = vvTried[nKBucket];
|
||||||
|
|
||||||
// random shuffle the first few elements (using the entire list)
|
// random shuffle the first few elements (using the entire list)
|
||||||
// find the least recently tried among them
|
// find the least recently tried among them
|
||||||
int64_t nOldest = -1;
|
int64_t nOldest = -1;
|
||||||
int nOldestPos = -1;
|
int nOldestPos = -1;
|
||||||
for (unsigned int i = 0; i < ADDRMAN_TRIED_ENTRIES_INSPECT_ON_EVICT && i < vTried.size(); i++)
|
for (unsigned int i = 0; i < ADDRMAN_TRIED_ENTRIES_INSPECT_ON_EVICT && i < vTried.size(); i++) {
|
||||||
{
|
|
||||||
int nPos = GetRandInt(vTried.size() - i) + i;
|
int nPos = GetRandInt(vTried.size() - i) + i;
|
||||||
int nTemp = vTried[nPos];
|
int nTemp = vTried[nPos];
|
||||||
vTried[nPos] = vTried[i];
|
vTried[nPos] = vTried[i];
|
||||||
vTried[i] = nTemp;
|
vTried[i] = nTemp;
|
||||||
assert(nOldest == -1 || mapInfo.count(nTemp) == 1);
|
assert(nOldest == -1 || mapInfo.count(nTemp) == 1);
|
||||||
if (nOldest == -1 || mapInfo[nTemp].nLastSuccess < mapInfo[nOldest].nLastSuccess) {
|
if (nOldest == -1 || mapInfo[nTemp].nLastSuccess < mapInfo[nOldest].nLastSuccess) {
|
||||||
nOldest = nTemp;
|
nOldest = nTemp;
|
||||||
nOldestPos = nPos;
|
nOldestPos = nPos;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -152,18 +153,15 @@ int CAddrMan::SelectTried(int nKBucket)
|
||||||
int CAddrMan::ShrinkNew(int nUBucket)
|
int CAddrMan::ShrinkNew(int nUBucket)
|
||||||
{
|
{
|
||||||
assert(nUBucket >= 0 && (unsigned int)nUBucket < vvNew.size());
|
assert(nUBucket >= 0 && (unsigned int)nUBucket < vvNew.size());
|
||||||
std::set<int> &vNew = vvNew[nUBucket];
|
std::set<int>& vNew = vvNew[nUBucket];
|
||||||
|
|
||||||
// first look for deletable items
|
// first look for deletable items
|
||||||
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++)
|
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++) {
|
||||||
{
|
|
||||||
assert(mapInfo.count(*it));
|
assert(mapInfo.count(*it));
|
||||||
CAddrInfo &info = mapInfo[*it];
|
CAddrInfo& info = mapInfo[*it];
|
||||||
if (info.IsTerrible())
|
if (info.IsTerrible()) {
|
||||||
{
|
if (--info.nRefCount == 0) {
|
||||||
if (--info.nRefCount == 0)
|
SwapRandom(info.nRandomPos, vRandom.size() - 1);
|
||||||
{
|
|
||||||
SwapRandom(info.nRandomPos, vRandom.size()-1);
|
|
||||||
vRandom.pop_back();
|
vRandom.pop_back();
|
||||||
mapAddr.erase(info);
|
mapAddr.erase(info);
|
||||||
mapInfo.erase(*it);
|
mapInfo.erase(*it);
|
||||||
|
@ -178,10 +176,8 @@ int CAddrMan::ShrinkNew(int nUBucket)
|
||||||
int n[4] = {GetRandInt(vNew.size()), GetRandInt(vNew.size()), GetRandInt(vNew.size()), GetRandInt(vNew.size())};
|
int n[4] = {GetRandInt(vNew.size()), GetRandInt(vNew.size()), GetRandInt(vNew.size()), GetRandInt(vNew.size())};
|
||||||
int nI = 0;
|
int nI = 0;
|
||||||
int nOldest = -1;
|
int nOldest = -1;
|
||||||
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++)
|
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]) {
|
||||||
if (nI == n[0] || nI == n[1] || nI == n[2] || nI == n[3])
|
|
||||||
{
|
|
||||||
assert(nOldest == -1 || mapInfo.count(*it) == 1);
|
assert(nOldest == -1 || mapInfo.count(*it) == 1);
|
||||||
if (nOldest == -1 || mapInfo[*it].nTime < mapInfo[nOldest].nTime)
|
if (nOldest == -1 || mapInfo[*it].nTime < mapInfo[nOldest].nTime)
|
||||||
nOldest = *it;
|
nOldest = *it;
|
||||||
|
@ -189,10 +185,9 @@ int CAddrMan::ShrinkNew(int nUBucket)
|
||||||
nI++;
|
nI++;
|
||||||
}
|
}
|
||||||
assert(mapInfo.count(nOldest) == 1);
|
assert(mapInfo.count(nOldest) == 1);
|
||||||
CAddrInfo &info = mapInfo[nOldest];
|
CAddrInfo& info = mapInfo[nOldest];
|
||||||
if (--info.nRefCount == 0)
|
if (--info.nRefCount == 0) {
|
||||||
{
|
SwapRandom(info.nRandomPos, vRandom.size() - 1);
|
||||||
SwapRandom(info.nRandomPos, vRandom.size()-1);
|
|
||||||
vRandom.pop_back();
|
vRandom.pop_back();
|
||||||
mapAddr.erase(info);
|
mapAddr.erase(info);
|
||||||
mapInfo.erase(nOldest);
|
mapInfo.erase(nOldest);
|
||||||
|
@ -208,8 +203,7 @@ void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
||||||
assert(vvNew[nOrigin].count(nId) == 1);
|
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 (std::vector<std::set<int> >::iterator it = vvNew.begin(); it != vvNew.end(); it++) {
|
||||||
{
|
|
||||||
if ((*it).erase(nId))
|
if ((*it).erase(nId))
|
||||||
info.nRefCount--;
|
info.nRefCount--;
|
||||||
}
|
}
|
||||||
|
@ -219,11 +213,10 @@ void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
||||||
|
|
||||||
// what tried bucket to move the entry to
|
// what tried bucket to move the entry to
|
||||||
int nKBucket = info.GetTriedBucket(nKey);
|
int nKBucket = info.GetTriedBucket(nKey);
|
||||||
std::vector<int> &vTried = vvTried[nKBucket];
|
std::vector<int>& vTried = vvTried[nKBucket];
|
||||||
|
|
||||||
// first check whether there is place to just add it
|
// first check whether there is place to just add it
|
||||||
if (vTried.size() < ADDRMAN_TRIED_BUCKET_SIZE)
|
if (vTried.size() < ADDRMAN_TRIED_BUCKET_SIZE) {
|
||||||
{
|
|
||||||
vTried.push_back(nId);
|
vTried.push_back(nId);
|
||||||
nTried++;
|
nTried++;
|
||||||
info.fInTried = true;
|
info.fInTried = true;
|
||||||
|
@ -236,7 +229,7 @@ void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
||||||
// find which new bucket it belongs to
|
// find which new bucket it belongs to
|
||||||
assert(mapInfo.count(vTried[nPos]) == 1);
|
assert(mapInfo.count(vTried[nPos]) == 1);
|
||||||
int nUBucket = mapInfo[vTried[nPos]].GetNewBucket(nKey);
|
int nUBucket = mapInfo[vTried[nPos]].GetNewBucket(nKey);
|
||||||
std::set<int> &vNew = vvNew[nUBucket];
|
std::set<int>& vNew = vvNew[nUBucket];
|
||||||
|
|
||||||
// remove the to-be-replaced tried entry from the tried set
|
// remove the to-be-replaced tried entry from the tried set
|
||||||
CAddrInfo& infoOld = mapInfo[vTried[nPos]];
|
CAddrInfo& infoOld = mapInfo[vTried[nPos]];
|
||||||
|
@ -245,8 +238,7 @@ void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
||||||
// do not update nTried, as we are going to move something else there immediately
|
// do not update nTried, as we are going to move something else there immediately
|
||||||
|
|
||||||
// check whether there is place in that one,
|
// check whether there is place in that one,
|
||||||
if (vNew.size() < ADDRMAN_NEW_BUCKET_SIZE)
|
if (vNew.size() < ADDRMAN_NEW_BUCKET_SIZE) {
|
||||||
{
|
|
||||||
// if so, move it back there
|
// if so, move it back there
|
||||||
vNew.insert(vTried[nPos]);
|
vNew.insert(vTried[nPos]);
|
||||||
} else {
|
} else {
|
||||||
|
@ -261,16 +253,16 @@ void CAddrMan::MakeTried(CAddrInfo& info, int nId, int nOrigin)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CAddrMan::Good_(const CService &addr, int64_t nTime)
|
void CAddrMan::Good_(const CService& addr, int64_t nTime)
|
||||||
{
|
{
|
||||||
int nId;
|
int nId;
|
||||||
CAddrInfo *pinfo = Find(addr, &nId);
|
CAddrInfo* pinfo = Find(addr, &nId);
|
||||||
|
|
||||||
// if not found, bail out
|
// if not found, bail out
|
||||||
if (!pinfo)
|
if (!pinfo)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
CAddrInfo &info = *pinfo;
|
CAddrInfo& info = *pinfo;
|
||||||
|
|
||||||
// check whether we are talking about the exact same CService (including same port)
|
// check whether we are talking about the exact same CService (including same port)
|
||||||
if (info != addr)
|
if (info != addr)
|
||||||
|
@ -289,12 +281,10 @@ void CAddrMan::Good_(const CService &addr, int64_t nTime)
|
||||||
// find a bucket it is in now
|
// find a bucket it is in now
|
||||||
int nRnd = GetRandInt(vvNew.size());
|
int nRnd = GetRandInt(vvNew.size());
|
||||||
int nUBucket = -1;
|
int nUBucket = -1;
|
||||||
for (unsigned int n = 0; n < vvNew.size(); n++)
|
for (unsigned int n = 0; n < vvNew.size(); n++) {
|
||||||
{
|
int nB = (n + nRnd) % vvNew.size();
|
||||||
int nB = (n+nRnd) % vvNew.size();
|
std::set<int>& vNew = vvNew[nB];
|
||||||
std::set<int> &vNew = vvNew[nB];
|
if (vNew.count(nId)) {
|
||||||
if (vNew.count(nId))
|
|
||||||
{
|
|
||||||
nUBucket = nB;
|
nUBucket = nB;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -302,7 +292,8 @@ void CAddrMan::Good_(const CService &addr, int64_t nTime)
|
||||||
|
|
||||||
// if no bucket is found, something bad happened;
|
// if no bucket is found, something bad happened;
|
||||||
// TODO: maybe re-add the node, but for now, just bail out
|
// TODO: maybe re-add the node, but for now, just bail out
|
||||||
if (nUBucket == -1) return;
|
if (nUBucket == -1)
|
||||||
|
return;
|
||||||
|
|
||||||
LogPrint("addrman", "Moving %s to tried\n", addr.ToString());
|
LogPrint("addrman", "Moving %s to tried\n", addr.ToString());
|
||||||
|
|
||||||
|
@ -310,17 +301,16 @@ void CAddrMan::Good_(const CService &addr, int64_t nTime)
|
||||||
MakeTried(info, nId, nUBucket);
|
MakeTried(info, nId, nUBucket);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CAddrMan::Add_(const CAddress &addr, const CNetAddr& source, int64_t nTimePenalty)
|
bool CAddrMan::Add_(const CAddress& addr, const CNetAddr& source, int64_t nTimePenalty)
|
||||||
{
|
{
|
||||||
if (!addr.IsRoutable())
|
if (!addr.IsRoutable())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
bool fNew = false;
|
bool fNew = false;
|
||||||
int nId;
|
int nId;
|
||||||
CAddrInfo *pinfo = Find(addr, &nId);
|
CAddrInfo* pinfo = Find(addr, &nId);
|
||||||
|
|
||||||
if (pinfo)
|
if (pinfo) {
|
||||||
{
|
|
||||||
// periodically update nTime
|
// periodically update nTime
|
||||||
bool fCurrentlyOnline = (GetAdjustedTime() - addr.nTime < 24 * 60 * 60);
|
bool fCurrentlyOnline = (GetAdjustedTime() - addr.nTime < 24 * 60 * 60);
|
||||||
int64_t nUpdateInterval = (fCurrentlyOnline ? 60 * 60 : 24 * 60 * 60);
|
int64_t nUpdateInterval = (fCurrentlyOnline ? 60 * 60 : 24 * 60 * 60);
|
||||||
|
@ -344,7 +334,7 @@ bool CAddrMan::Add_(const CAddress &addr, const CNetAddr& source, int64_t nTimeP
|
||||||
|
|
||||||
// stochastic test: previous nRefCount == N: 2^N times harder to increase it
|
// stochastic test: previous nRefCount == N: 2^N times harder to increase it
|
||||||
int nFactor = 1;
|
int nFactor = 1;
|
||||||
for (int n=0; n<pinfo->nRefCount; n++)
|
for (int n = 0; n < pinfo->nRefCount; n++)
|
||||||
nFactor *= 2;
|
nFactor *= 2;
|
||||||
if (nFactor > 1 && (GetRandInt(nFactor) != 0))
|
if (nFactor > 1 && (GetRandInt(nFactor) != 0))
|
||||||
return false;
|
return false;
|
||||||
|
@ -356,9 +346,8 @@ 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];
|
std::set<int>& vNew = vvNew[nUBucket];
|
||||||
if (!vNew.count(nId))
|
if (!vNew.count(nId)) {
|
||||||
{
|
|
||||||
pinfo->nRefCount++;
|
pinfo->nRefCount++;
|
||||||
if (vNew.size() == ADDRMAN_NEW_BUCKET_SIZE)
|
if (vNew.size() == ADDRMAN_NEW_BUCKET_SIZE)
|
||||||
ShrinkNew(nUBucket);
|
ShrinkNew(nUBucket);
|
||||||
|
@ -367,15 +356,15 @@ bool CAddrMan::Add_(const CAddress &addr, const CNetAddr& source, int64_t nTimeP
|
||||||
return fNew;
|
return fNew;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CAddrMan::Attempt_(const CService &addr, int64_t nTime)
|
void CAddrMan::Attempt_(const CService& addr, int64_t nTime)
|
||||||
{
|
{
|
||||||
CAddrInfo *pinfo = Find(addr);
|
CAddrInfo* pinfo = Find(addr);
|
||||||
|
|
||||||
// if not found, bail out
|
// if not found, bail out
|
||||||
if (!pinfo)
|
if (!pinfo)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
CAddrInfo &info = *pinfo;
|
CAddrInfo& info = *pinfo;
|
||||||
|
|
||||||
// check whether we are talking about the exact same CService (including same port)
|
// check whether we are talking about the exact same CService (including same port)
|
||||||
if (info != addr)
|
if (info != addr)
|
||||||
|
@ -393,37 +382,36 @@ CAddress CAddrMan::Select_(int nUnkBias)
|
||||||
|
|
||||||
double nCorTried = sqrt(nTried) * (100.0 - nUnkBias);
|
double nCorTried = sqrt(nTried) * (100.0 - nUnkBias);
|
||||||
double nCorNew = sqrt(nNew) * nUnkBias;
|
double nCorNew = sqrt(nNew) * nUnkBias;
|
||||||
if ((nCorTried + nCorNew)*GetRandInt(1<<30)/(1<<30) < nCorTried)
|
if ((nCorTried + nCorNew) * GetRandInt(1 << 30) / (1 << 30) < nCorTried) {
|
||||||
{
|
|
||||||
// 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(vvTried.size());
|
||||||
std::vector<int> &vTried = vvTried[nKBucket];
|
std::vector<int>& vTried = vvTried[nKBucket];
|
||||||
if (vTried.size() == 0) continue;
|
if (vTried.size() == 0)
|
||||||
|
continue;
|
||||||
int nPos = GetRandInt(vTried.size());
|
int nPos = GetRandInt(vTried.size());
|
||||||
assert(mapInfo.count(vTried[nPos]) == 1);
|
assert(mapInfo.count(vTried[nPos]) == 1);
|
||||||
CAddrInfo &info = mapInfo[vTried[nPos]];
|
CAddrInfo& info = mapInfo[vTried[nPos]];
|
||||||
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;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// 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(vvNew.size());
|
||||||
std::set<int> &vNew = vvNew[nUBucket];
|
std::set<int>& vNew = vvNew[nUBucket];
|
||||||
if (vNew.size() == 0) continue;
|
if (vNew.size() == 0)
|
||||||
|
continue;
|
||||||
int nPos = GetRandInt(vNew.size());
|
int nPos = GetRandInt(vNew.size());
|
||||||
std::set<int>::iterator it = vNew.begin();
|
std::set<int>::iterator it = vNew.begin();
|
||||||
while (nPos--)
|
while (nPos--)
|
||||||
it++;
|
it++;
|
||||||
assert(mapInfo.count(*it) == 1);
|
assert(mapInfo.count(*it) == 1);
|
||||||
CAddrInfo &info = mapInfo[*it];
|
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;
|
||||||
}
|
}
|
||||||
|
@ -436,69 +424,76 @@ int CAddrMan::Check_()
|
||||||
std::set<int> setTried;
|
std::set<int> setTried;
|
||||||
std::map<int, int> mapNew;
|
std::map<int, int> mapNew;
|
||||||
|
|
||||||
if (vRandom.size() != nTried + nNew) return -7;
|
if (vRandom.size() != nTried + nNew)
|
||||||
|
return -7;
|
||||||
|
|
||||||
for (std::map<int, CAddrInfo>::iterator it = mapInfo.begin(); it != mapInfo.end(); it++)
|
for (std::map<int, CAddrInfo>::iterator it = mapInfo.begin(); it != mapInfo.end(); it++) {
|
||||||
{
|
|
||||||
int n = (*it).first;
|
int n = (*it).first;
|
||||||
CAddrInfo &info = (*it).second;
|
CAddrInfo& info = (*it).second;
|
||||||
if (info.fInTried)
|
if (info.fInTried) {
|
||||||
{
|
if (!info.nLastSuccess)
|
||||||
|
return -1;
|
||||||
if (!info.nLastSuccess) return -1;
|
if (info.nRefCount)
|
||||||
if (info.nRefCount) return -2;
|
return -2;
|
||||||
setTried.insert(n);
|
setTried.insert(n);
|
||||||
} else {
|
} else {
|
||||||
if (info.nRefCount < 0 || info.nRefCount > ADDRMAN_NEW_BUCKETS_PER_ADDRESS) return -3;
|
if (info.nRefCount < 0 || info.nRefCount > ADDRMAN_NEW_BUCKETS_PER_ADDRESS)
|
||||||
if (!info.nRefCount) return -4;
|
return -3;
|
||||||
|
if (!info.nRefCount)
|
||||||
|
return -4;
|
||||||
mapNew[n] = info.nRefCount;
|
mapNew[n] = info.nRefCount;
|
||||||
}
|
}
|
||||||
if (mapAddr[info] != n) return -5;
|
if (mapAddr[info] != n)
|
||||||
if (info.nRandomPos<0 || info.nRandomPos>=vRandom.size() || vRandom[info.nRandomPos] != n) return -14;
|
return -5;
|
||||||
if (info.nLastTry < 0) return -6;
|
if (info.nRandomPos < 0 || info.nRandomPos >= vRandom.size() || vRandom[info.nRandomPos] != n)
|
||||||
if (info.nLastSuccess < 0) return -8;
|
return -14;
|
||||||
|
if (info.nLastTry < 0)
|
||||||
|
return -6;
|
||||||
|
if (info.nLastSuccess < 0)
|
||||||
|
return -8;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (setTried.size() != nTried) return -9;
|
if (setTried.size() != nTried)
|
||||||
if (mapNew.size() != nNew) return -10;
|
return -9;
|
||||||
|
if (mapNew.size() != nNew)
|
||||||
|
return -10;
|
||||||
|
|
||||||
for (int n=0; n<vvTried.size(); n++)
|
for (int n = 0; n < vvTried.size(); n++) {
|
||||||
{
|
std::vector<int>& vTried = vvTried[n];
|
||||||
std::vector<int> &vTried = vvTried[n];
|
for (std::vector<int>::iterator it = vTried.begin(); it != vTried.end(); it++) {
|
||||||
for (std::vector<int>::iterator it = vTried.begin(); it != vTried.end(); it++)
|
if (!setTried.count(*it))
|
||||||
{
|
return -11;
|
||||||
if (!setTried.count(*it)) return -11;
|
|
||||||
setTried.erase(*it);
|
setTried.erase(*it);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int n=0; n<vvNew.size(); n++)
|
for (int n = 0; n < vvNew.size(); n++) {
|
||||||
{
|
std::set<int>& vNew = vvNew[n];
|
||||||
std::set<int> &vNew = vvNew[n];
|
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++) {
|
||||||
for (std::set<int>::iterator it = vNew.begin(); it != vNew.end(); it++)
|
if (!mapNew.count(*it))
|
||||||
{
|
return -12;
|
||||||
if (!mapNew.count(*it)) return -12;
|
|
||||||
if (--mapNew[*it] == 0)
|
if (--mapNew[*it] == 0)
|
||||||
mapNew.erase(*it);
|
mapNew.erase(*it);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (setTried.size()) return -13;
|
if (setTried.size())
|
||||||
if (mapNew.size()) return -15;
|
return -13;
|
||||||
|
if (mapNew.size())
|
||||||
|
return -15;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
void CAddrMan::GetAddr_(std::vector<CAddress> &vAddr)
|
void CAddrMan::GetAddr_(std::vector<CAddress>& vAddr)
|
||||||
{
|
{
|
||||||
unsigned int nNodes = ADDRMAN_GETADDR_MAX_PCT * vRandom.size() / 100;
|
unsigned int nNodes = ADDRMAN_GETADDR_MAX_PCT * vRandom.size() / 100;
|
||||||
if (nNodes > ADDRMAN_GETADDR_MAX)
|
if (nNodes > ADDRMAN_GETADDR_MAX)
|
||||||
nNodes = ADDRMAN_GETADDR_MAX;
|
nNodes = ADDRMAN_GETADDR_MAX;
|
||||||
|
|
||||||
// gather a list of random nodes, skipping those of low quality
|
// gather a list of random nodes, skipping those of low quality
|
||||||
for (unsigned int n = 0; n < vRandom.size(); n++)
|
for (unsigned int n = 0; n < vRandom.size(); n++) {
|
||||||
{
|
|
||||||
if (vAddr.size() >= nNodes)
|
if (vAddr.size() >= nNodes)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -512,15 +507,15 @@ void CAddrMan::GetAddr_(std::vector<CAddress> &vAddr)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void CAddrMan::Connected_(const CService &addr, int64_t nTime)
|
void CAddrMan::Connected_(const CService& addr, int64_t nTime)
|
||||||
{
|
{
|
||||||
CAddrInfo *pinfo = Find(addr);
|
CAddrInfo* pinfo = Find(addr);
|
||||||
|
|
||||||
// if not found, bail out
|
// if not found, bail out
|
||||||
if (!pinfo)
|
if (!pinfo)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
CAddrInfo &info = *pinfo;
|
CAddrInfo& info = *pinfo;
|
||||||
|
|
||||||
// check whether we are talking about the exact same CService (including same port)
|
// check whether we are talking about the exact same CService (including same port)
|
||||||
if (info != addr)
|
if (info != addr)
|
||||||
|
|
|
@ -37,13 +37,13 @@ static inline size_t GetSystemPageSize()
|
||||||
page_size = sSysInfo.dwPageSize;
|
page_size = sSysInfo.dwPageSize;
|
||||||
#elif defined(PAGESIZE) // defined in limits.h
|
#elif defined(PAGESIZE) // defined in limits.h
|
||||||
page_size = PAGESIZE;
|
page_size = PAGESIZE;
|
||||||
#else // assume some POSIX OS
|
#else // assume some POSIX OS
|
||||||
page_size = sysconf(_SC_PAGESIZE);
|
page_size = sysconf(_SC_PAGESIZE);
|
||||||
#endif
|
#endif
|
||||||
return page_size;
|
return page_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MemoryPageLocker::Lock(const void *addr, size_t len)
|
bool MemoryPageLocker::Lock(const void* addr, size_t len)
|
||||||
{
|
{
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
return VirtualLock(const_cast<void*>(addr), len) != 0;
|
return VirtualLock(const_cast<void*>(addr), len) != 0;
|
||||||
|
@ -52,7 +52,7 @@ bool MemoryPageLocker::Lock(const void *addr, size_t len)
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MemoryPageLocker::Unlock(const void *addr, size_t len)
|
bool MemoryPageLocker::Unlock(const void* addr, size_t len)
|
||||||
{
|
{
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
return VirtualUnlock(const_cast<void*>(addr), len) != 0;
|
return VirtualUnlock(const_cast<void*>(addr), len) != 0;
|
||||||
|
@ -64,4 +64,3 @@ bool MemoryPageLocker::Unlock(const void *addr, size_t len)
|
||||||
LockedPageManager::LockedPageManager() : LockedPageManagerBase<MemoryPageLocker>(GetSystemPageSize())
|
LockedPageManager::LockedPageManager() : LockedPageManagerBase<MemoryPageLocker>(GetSystemPageSize())
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -26,14 +26,14 @@
|
||||||
* small objects that span up to a few pages, mostly smaller than a page. To support large allocations,
|
* small objects that span up to a few pages, mostly smaller than a page. To support large allocations,
|
||||||
* something like an interval tree would be the preferred data structure.
|
* something like an interval tree would be the preferred data structure.
|
||||||
*/
|
*/
|
||||||
template <class Locker> class LockedPageManagerBase
|
template <class Locker>
|
||||||
|
class LockedPageManagerBase
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
LockedPageManagerBase(size_t page_size):
|
LockedPageManagerBase(size_t page_size) : page_size(page_size)
|
||||||
page_size(page_size)
|
|
||||||
{
|
{
|
||||||
// Determine bitmask for extracting page from address
|
// Determine bitmask for extracting page from address
|
||||||
assert(!(page_size & (page_size-1))); // size must be power of two
|
assert(!(page_size & (page_size - 1))); // size must be power of two
|
||||||
page_mask = ~(page_size - 1);
|
page_mask = ~(page_size - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,22 +44,21 @@ public:
|
||||||
|
|
||||||
|
|
||||||
// For all pages in affected range, increase lock count
|
// For all pages in affected range, increase lock count
|
||||||
void LockRange(void *p, size_t size)
|
void LockRange(void* p, size_t size)
|
||||||
{
|
{
|
||||||
boost::mutex::scoped_lock lock(mutex);
|
boost::mutex::scoped_lock lock(mutex);
|
||||||
if(!size) return;
|
if (!size)
|
||||||
|
return;
|
||||||
const size_t base_addr = reinterpret_cast<size_t>(p);
|
const size_t base_addr = reinterpret_cast<size_t>(p);
|
||||||
const size_t start_page = base_addr & page_mask;
|
const size_t start_page = base_addr & page_mask;
|
||||||
const size_t end_page = (base_addr + size - 1) & page_mask;
|
const size_t end_page = (base_addr + size - 1) & page_mask;
|
||||||
for(size_t page = start_page; page <= end_page; page += page_size)
|
for (size_t page = start_page; page <= end_page; page += page_size) {
|
||||||
{
|
|
||||||
Histogram::iterator it = histogram.find(page);
|
Histogram::iterator it = histogram.find(page);
|
||||||
if(it == histogram.end()) // Newly locked page
|
if (it == histogram.end()) // Newly locked page
|
||||||
{
|
{
|
||||||
locker.Lock(reinterpret_cast<void*>(page), page_size);
|
locker.Lock(reinterpret_cast<void*>(page), page_size);
|
||||||
histogram.insert(std::make_pair(page, 1));
|
histogram.insert(std::make_pair(page, 1));
|
||||||
}
|
} else // Page was already locked; increase counter
|
||||||
else // Page was already locked; increase counter
|
|
||||||
{
|
{
|
||||||
it->second += 1;
|
it->second += 1;
|
||||||
}
|
}
|
||||||
|
@ -67,20 +66,20 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
// For all pages in affected range, decrease lock count
|
// For all pages in affected range, decrease lock count
|
||||||
void UnlockRange(void *p, size_t size)
|
void UnlockRange(void* p, size_t size)
|
||||||
{
|
{
|
||||||
boost::mutex::scoped_lock lock(mutex);
|
boost::mutex::scoped_lock lock(mutex);
|
||||||
if(!size) return;
|
if (!size)
|
||||||
|
return;
|
||||||
const size_t base_addr = reinterpret_cast<size_t>(p);
|
const size_t base_addr = reinterpret_cast<size_t>(p);
|
||||||
const size_t start_page = base_addr & page_mask;
|
const size_t start_page = base_addr & page_mask;
|
||||||
const size_t end_page = (base_addr + size - 1) & page_mask;
|
const size_t end_page = (base_addr + size - 1) & page_mask;
|
||||||
for(size_t page = start_page; page <= end_page; page += page_size)
|
for (size_t page = start_page; page <= end_page; page += page_size) {
|
||||||
{
|
|
||||||
Histogram::iterator it = histogram.find(page);
|
Histogram::iterator it = histogram.find(page);
|
||||||
assert(it != histogram.end()); // Cannot unlock an area that was not locked
|
assert(it != histogram.end()); // Cannot unlock an area that was not locked
|
||||||
// Decrease counter for page, when it is zero, the page will be unlocked
|
// Decrease counter for page, when it is zero, the page will be unlocked
|
||||||
it->second -= 1;
|
it->second -= 1;
|
||||||
if(it->second == 0) // Nothing on the page anymore that keeps it locked
|
if (it->second == 0) // Nothing on the page anymore that keeps it locked
|
||||||
{
|
{
|
||||||
// Unlock page and remove the count from histogram
|
// Unlock page and remove the count from histogram
|
||||||
locker.Unlock(reinterpret_cast<void*>(page), page_size);
|
locker.Unlock(reinterpret_cast<void*>(page), page_size);
|
||||||
|
@ -101,7 +100,7 @@ private:
|
||||||
boost::mutex mutex;
|
boost::mutex mutex;
|
||||||
size_t page_size, page_mask;
|
size_t page_size, page_mask;
|
||||||
// map of page base address to lock count
|
// map of page base address to lock count
|
||||||
typedef std::map<size_t,int> Histogram;
|
typedef std::map<size_t, int> Histogram;
|
||||||
Histogram histogram;
|
Histogram histogram;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -116,11 +115,11 @@ public:
|
||||||
/** Lock memory pages.
|
/** Lock memory pages.
|
||||||
* addr and len must be a multiple of the system page size
|
* addr and len must be a multiple of the system page size
|
||||||
*/
|
*/
|
||||||
bool Lock(const void *addr, size_t len);
|
bool Lock(const void* addr, size_t len);
|
||||||
/** Unlock memory pages.
|
/** Unlock memory pages.
|
||||||
* addr and len must be a multiple of the system page size
|
* addr and len must be a multiple of the system page size
|
||||||
*/
|
*/
|
||||||
bool Unlock(const void *addr, size_t len);
|
bool Unlock(const void* addr, size_t len);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -134,7 +133,7 @@ public:
|
||||||
* secure_allocator are created. So instead of having LockedPageManager also be
|
* secure_allocator are created. So instead of having LockedPageManager also be
|
||||||
* static-initialized, it is created on demand.
|
* static-initialized, it is created on demand.
|
||||||
*/
|
*/
|
||||||
class LockedPageManager: public LockedPageManagerBase<MemoryPageLocker>
|
class LockedPageManager : public LockedPageManagerBase<MemoryPageLocker>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
static LockedPageManager& Instance()
|
static LockedPageManager& Instance()
|
||||||
|
@ -165,11 +164,15 @@ private:
|
||||||
// Functions for directly locking/unlocking memory objects.
|
// Functions for directly locking/unlocking memory objects.
|
||||||
// Intended for non-dynamically allocated structures.
|
// Intended for non-dynamically allocated structures.
|
||||||
//
|
//
|
||||||
template<typename T> void LockObject(const T &t) {
|
template <typename T>
|
||||||
|
void LockObject(const T& t)
|
||||||
|
{
|
||||||
LockedPageManager::Instance().LockRange((void*)(&t), sizeof(T));
|
LockedPageManager::Instance().LockRange((void*)(&t), sizeof(T));
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename T> void UnlockObject(const T &t) {
|
template <typename T>
|
||||||
|
void UnlockObject(const T& t)
|
||||||
|
{
|
||||||
OPENSSL_cleanse((void*)(&t), sizeof(T));
|
OPENSSL_cleanse((void*)(&t), sizeof(T));
|
||||||
LockedPageManager::Instance().UnlockRange((void*)(&t), sizeof(T));
|
LockedPageManager::Instance().UnlockRange((void*)(&t), sizeof(T));
|
||||||
}
|
}
|
||||||
|
@ -178,13 +181,12 @@ template<typename T> void UnlockObject(const T &t) {
|
||||||
// Allocator that locks its contents from being paged
|
// Allocator that locks its contents from being paged
|
||||||
// out of memory and clears its contents before deletion.
|
// out of memory and clears its contents before deletion.
|
||||||
//
|
//
|
||||||
template<typename T>
|
template <typename T>
|
||||||
struct secure_allocator : public std::allocator<T>
|
struct secure_allocator : public std::allocator<T> {
|
||||||
{
|
|
||||||
// MSVC8 default copy constructor is broken
|
// MSVC8 default copy constructor is broken
|
||||||
typedef std::allocator<T> base;
|
typedef std::allocator<T> base;
|
||||||
typedef typename base::size_type size_type;
|
typedef typename base::size_type size_type;
|
||||||
typedef typename base::difference_type difference_type;
|
typedef typename base::difference_type difference_type;
|
||||||
typedef typename base::pointer pointer;
|
typedef typename base::pointer pointer;
|
||||||
typedef typename base::const_pointer const_pointer;
|
typedef typename base::const_pointer const_pointer;
|
||||||
typedef typename base::reference reference;
|
typedef typename base::reference reference;
|
||||||
|
@ -193,14 +195,18 @@ struct secure_allocator : public std::allocator<T>
|
||||||
secure_allocator() throw() {}
|
secure_allocator() throw() {}
|
||||||
secure_allocator(const secure_allocator& a) throw() : base(a) {}
|
secure_allocator(const secure_allocator& a) throw() : base(a) {}
|
||||||
template <typename U>
|
template <typename U>
|
||||||
secure_allocator(const secure_allocator<U>& a) throw() : base(a) {}
|
secure_allocator(const secure_allocator<U>& a) throw() : base(a)
|
||||||
~secure_allocator() throw() {}
|
|
||||||
template<typename _Other> struct rebind
|
|
||||||
{ typedef secure_allocator<_Other> other; };
|
|
||||||
|
|
||||||
T* allocate(std::size_t n, const void *hint = 0)
|
|
||||||
{
|
{
|
||||||
T *p;
|
}
|
||||||
|
~secure_allocator() throw() {}
|
||||||
|
template <typename _Other>
|
||||||
|
struct rebind {
|
||||||
|
typedef secure_allocator<_Other> other;
|
||||||
|
};
|
||||||
|
|
||||||
|
T* allocate(std::size_t n, const void* hint = 0)
|
||||||
|
{
|
||||||
|
T* p;
|
||||||
p = std::allocator<T>::allocate(n, hint);
|
p = std::allocator<T>::allocate(n, hint);
|
||||||
if (p != NULL)
|
if (p != NULL)
|
||||||
LockedPageManager::Instance().LockRange(p, sizeof(T) * n);
|
LockedPageManager::Instance().LockRange(p, sizeof(T) * n);
|
||||||
|
@ -209,8 +215,7 @@ struct secure_allocator : public std::allocator<T>
|
||||||
|
|
||||||
void deallocate(T* p, std::size_t n)
|
void deallocate(T* p, std::size_t n)
|
||||||
{
|
{
|
||||||
if (p != NULL)
|
if (p != NULL) {
|
||||||
{
|
|
||||||
OPENSSL_cleanse(p, sizeof(T) * n);
|
OPENSSL_cleanse(p, sizeof(T) * n);
|
||||||
LockedPageManager::Instance().UnlockRange(p, sizeof(T) * n);
|
LockedPageManager::Instance().UnlockRange(p, sizeof(T) * n);
|
||||||
}
|
}
|
||||||
|
@ -222,13 +227,12 @@ struct secure_allocator : public std::allocator<T>
|
||||||
//
|
//
|
||||||
// Allocator that clears its contents before deletion.
|
// Allocator that clears its contents before deletion.
|
||||||
//
|
//
|
||||||
template<typename T>
|
template <typename T>
|
||||||
struct zero_after_free_allocator : public std::allocator<T>
|
struct zero_after_free_allocator : public std::allocator<T> {
|
||||||
{
|
|
||||||
// MSVC8 default copy constructor is broken
|
// MSVC8 default copy constructor is broken
|
||||||
typedef std::allocator<T> base;
|
typedef std::allocator<T> base;
|
||||||
typedef typename base::size_type size_type;
|
typedef typename base::size_type size_type;
|
||||||
typedef typename base::difference_type difference_type;
|
typedef typename base::difference_type difference_type;
|
||||||
typedef typename base::pointer pointer;
|
typedef typename base::pointer pointer;
|
||||||
typedef typename base::const_pointer const_pointer;
|
typedef typename base::const_pointer const_pointer;
|
||||||
typedef typename base::reference reference;
|
typedef typename base::reference reference;
|
||||||
|
@ -237,10 +241,14 @@ struct zero_after_free_allocator : public std::allocator<T>
|
||||||
zero_after_free_allocator() throw() {}
|
zero_after_free_allocator() throw() {}
|
||||||
zero_after_free_allocator(const zero_after_free_allocator& a) throw() : base(a) {}
|
zero_after_free_allocator(const zero_after_free_allocator& a) throw() : base(a) {}
|
||||||
template <typename U>
|
template <typename U>
|
||||||
zero_after_free_allocator(const zero_after_free_allocator<U>& a) throw() : base(a) {}
|
zero_after_free_allocator(const zero_after_free_allocator<U>& a) throw() : base(a)
|
||||||
|
{
|
||||||
|
}
|
||||||
~zero_after_free_allocator() throw() {}
|
~zero_after_free_allocator() throw() {}
|
||||||
template<typename _Other> struct rebind
|
template <typename _Other>
|
||||||
{ typedef zero_after_free_allocator<_Other> other; };
|
struct rebind {
|
||||||
|
typedef zero_after_free_allocator<_Other> other;
|
||||||
|
};
|
||||||
|
|
||||||
void deallocate(T* p, std::size_t n)
|
void deallocate(T* p, std::size_t n)
|
||||||
{
|
{
|
||||||
|
|
129
src/base58.cpp
129
src/base58.cpp
|
@ -18,7 +18,8 @@
|
||||||
/* All alphanumeric characters except for "0", "I", "O", and "l" */
|
/* All alphanumeric characters except for "0", "I", "O", and "l" */
|
||||||
static const char* pszBase58 = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz";
|
static const char* pszBase58 = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz";
|
||||||
|
|
||||||
bool DecodeBase58(const char *psz, std::vector<unsigned char>& vch) {
|
bool DecodeBase58(const char* psz, std::vector<unsigned char>& vch)
|
||||||
|
{
|
||||||
// Skip leading spaces.
|
// Skip leading spaces.
|
||||||
while (*psz && isspace(*psz))
|
while (*psz && isspace(*psz))
|
||||||
psz++;
|
psz++;
|
||||||
|
@ -33,7 +34,7 @@ bool DecodeBase58(const char *psz, std::vector<unsigned char>& vch) {
|
||||||
// Process the characters.
|
// Process the characters.
|
||||||
while (*psz && !isspace(*psz)) {
|
while (*psz && !isspace(*psz)) {
|
||||||
// Decode base58 character
|
// Decode base58 character
|
||||||
const char *ch = strchr(pszBase58, *psz);
|
const char* ch = strchr(pszBase58, *psz);
|
||||||
if (ch == NULL)
|
if (ch == NULL)
|
||||||
return false;
|
return false;
|
||||||
// Apply "b256 = b256 * 58 + ch".
|
// Apply "b256 = b256 * 58 + ch".
|
||||||
|
@ -59,11 +60,12 @@ bool DecodeBase58(const char *psz, std::vector<unsigned char>& vch) {
|
||||||
vch.reserve(zeroes + (b256.end() - it));
|
vch.reserve(zeroes + (b256.end() - it));
|
||||||
vch.assign(zeroes, 0x00);
|
vch.assign(zeroes, 0x00);
|
||||||
while (it != b256.end())
|
while (it != b256.end())
|
||||||
vch.push_back(*(it++));
|
vch.push_back(*(it++));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string EncodeBase58(const unsigned char* pbegin, const unsigned char* pend) {
|
std::string EncodeBase58(const unsigned char* pbegin, const unsigned char* pend)
|
||||||
|
{
|
||||||
// Skip & count leading zeroes.
|
// Skip & count leading zeroes.
|
||||||
int zeroes = 0;
|
int zeroes = 0;
|
||||||
while (pbegin != pend && *pbegin == 0) {
|
while (pbegin != pend && *pbegin == 0) {
|
||||||
|
@ -97,15 +99,18 @@ std::string EncodeBase58(const unsigned char* pbegin, const unsigned char* pend)
|
||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string EncodeBase58(const std::vector<unsigned char>& vch) {
|
std::string EncodeBase58(const std::vector<unsigned char>& vch)
|
||||||
|
{
|
||||||
return EncodeBase58(&vch[0], &vch[0] + vch.size());
|
return EncodeBase58(&vch[0], &vch[0] + vch.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DecodeBase58(const std::string& str, std::vector<unsigned char>& vchRet) {
|
bool DecodeBase58(const std::string& str, std::vector<unsigned char>& vchRet)
|
||||||
|
{
|
||||||
return DecodeBase58(str.c_str(), vchRet);
|
return DecodeBase58(str.c_str(), vchRet);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string EncodeBase58Check(const std::vector<unsigned char>& vchIn) {
|
std::string EncodeBase58Check(const std::vector<unsigned char>& vchIn)
|
||||||
|
{
|
||||||
// add 4-byte hash check to the end
|
// add 4-byte hash check to the end
|
||||||
std::vector<unsigned char> vch(vchIn);
|
std::vector<unsigned char> vch(vchIn);
|
||||||
uint256 hash = Hash(vch.begin(), vch.end());
|
uint256 hash = Hash(vch.begin(), vch.end());
|
||||||
|
@ -113,45 +118,49 @@ std::string EncodeBase58Check(const std::vector<unsigned char>& vchIn) {
|
||||||
return EncodeBase58(vch);
|
return EncodeBase58(vch);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DecodeBase58Check(const char* psz, std::vector<unsigned char>& vchRet) {
|
bool DecodeBase58Check(const char* psz, std::vector<unsigned char>& vchRet)
|
||||||
|
{
|
||||||
if (!DecodeBase58(psz, vchRet) ||
|
if (!DecodeBase58(psz, vchRet) ||
|
||||||
(vchRet.size() < 4))
|
(vchRet.size() < 4)) {
|
||||||
{
|
|
||||||
vchRet.clear();
|
vchRet.clear();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// re-calculate the checksum, insure it matches the included 4-byte checksum
|
// re-calculate the checksum, insure it matches the included 4-byte checksum
|
||||||
uint256 hash = Hash(vchRet.begin(), vchRet.end()-4);
|
uint256 hash = Hash(vchRet.begin(), vchRet.end() - 4);
|
||||||
if (memcmp(&hash, &vchRet.end()[-4], 4) != 0)
|
if (memcmp(&hash, &vchRet.end()[-4], 4) != 0) {
|
||||||
{
|
|
||||||
vchRet.clear();
|
vchRet.clear();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
vchRet.resize(vchRet.size()-4);
|
vchRet.resize(vchRet.size() - 4);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DecodeBase58Check(const std::string& str, std::vector<unsigned char>& vchRet) {
|
bool DecodeBase58Check(const std::string& str, std::vector<unsigned char>& vchRet)
|
||||||
|
{
|
||||||
return DecodeBase58Check(str.c_str(), vchRet);
|
return DecodeBase58Check(str.c_str(), vchRet);
|
||||||
}
|
}
|
||||||
|
|
||||||
CBase58Data::CBase58Data() {
|
CBase58Data::CBase58Data()
|
||||||
|
{
|
||||||
vchVersion.clear();
|
vchVersion.clear();
|
||||||
vchData.clear();
|
vchData.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
void CBase58Data::SetData(const std::vector<unsigned char> &vchVersionIn, const void* pdata, size_t nSize) {
|
void CBase58Data::SetData(const std::vector<unsigned char>& vchVersionIn, const void* pdata, size_t nSize)
|
||||||
|
{
|
||||||
vchVersion = vchVersionIn;
|
vchVersion = vchVersionIn;
|
||||||
vchData.resize(nSize);
|
vchData.resize(nSize);
|
||||||
if (!vchData.empty())
|
if (!vchData.empty())
|
||||||
memcpy(&vchData[0], pdata, nSize);
|
memcpy(&vchData[0], pdata, nSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CBase58Data::SetData(const std::vector<unsigned char> &vchVersionIn, const unsigned char *pbegin, const unsigned char *pend) {
|
void CBase58Data::SetData(const std::vector<unsigned char>& vchVersionIn, const unsigned char* pbegin, const unsigned char* pend)
|
||||||
|
{
|
||||||
SetData(vchVersionIn, (void*)pbegin, pend - pbegin);
|
SetData(vchVersionIn, (void*)pbegin, pend - pbegin);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBase58Data::SetString(const char* psz, unsigned int nVersionBytes) {
|
bool CBase58Data::SetString(const char* psz, unsigned int nVersionBytes)
|
||||||
|
{
|
||||||
std::vector<unsigned char> vchTemp;
|
std::vector<unsigned char> vchTemp;
|
||||||
bool rc58 = DecodeBase58Check(psz, vchTemp);
|
bool rc58 = DecodeBase58Check(psz, vchTemp);
|
||||||
if ((!rc58) || (vchTemp.size() < nVersionBytes)) {
|
if ((!rc58) || (vchTemp.size() < nVersionBytes)) {
|
||||||
|
@ -167,65 +176,80 @@ bool CBase58Data::SetString(const char* psz, unsigned int nVersionBytes) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBase58Data::SetString(const std::string& str) {
|
bool CBase58Data::SetString(const std::string& str)
|
||||||
|
{
|
||||||
return SetString(str.c_str());
|
return SetString(str.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string CBase58Data::ToString() const {
|
std::string CBase58Data::ToString() const
|
||||||
|
{
|
||||||
std::vector<unsigned char> vch = vchVersion;
|
std::vector<unsigned char> vch = vchVersion;
|
||||||
vch.insert(vch.end(), vchData.begin(), vchData.end());
|
vch.insert(vch.end(), vchData.begin(), vchData.end());
|
||||||
return EncodeBase58Check(vch);
|
return EncodeBase58Check(vch);
|
||||||
}
|
}
|
||||||
|
|
||||||
int CBase58Data::CompareTo(const CBase58Data& b58) const {
|
int CBase58Data::CompareTo(const CBase58Data& b58) const
|
||||||
if (vchVersion < b58.vchVersion) return -1;
|
{
|
||||||
if (vchVersion > b58.vchVersion) return 1;
|
if (vchVersion < b58.vchVersion)
|
||||||
if (vchData < b58.vchData) return -1;
|
return -1;
|
||||||
if (vchData > b58.vchData) return 1;
|
if (vchVersion > b58.vchVersion)
|
||||||
|
return 1;
|
||||||
|
if (vchData < b58.vchData)
|
||||||
|
return -1;
|
||||||
|
if (vchData > b58.vchData)
|
||||||
|
return 1;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace {
|
namespace
|
||||||
|
{
|
||||||
|
class CBitcoinAddressVisitor : public boost::static_visitor<bool>
|
||||||
|
{
|
||||||
|
private:
|
||||||
|
CBitcoinAddress* addr;
|
||||||
|
|
||||||
class CBitcoinAddressVisitor : public boost::static_visitor<bool> {
|
public:
|
||||||
private:
|
CBitcoinAddressVisitor(CBitcoinAddress* addrIn) : addr(addrIn) {}
|
||||||
CBitcoinAddress *addr;
|
|
||||||
public:
|
|
||||||
CBitcoinAddressVisitor(CBitcoinAddress *addrIn) : addr(addrIn) { }
|
|
||||||
|
|
||||||
bool operator()(const CKeyID &id) const { return addr->Set(id); }
|
bool operator()(const CKeyID& id) const { return addr->Set(id); }
|
||||||
bool operator()(const CScriptID &id) const { return addr->Set(id); }
|
bool operator()(const CScriptID& id) const { return addr->Set(id); }
|
||||||
bool operator()(const CNoDestination &no) const { return false; }
|
bool operator()(const CNoDestination& no) const { return false; }
|
||||||
};
|
};
|
||||||
|
|
||||||
} // anon namespace
|
} // anon namespace
|
||||||
|
|
||||||
bool CBitcoinAddress::Set(const CKeyID &id) {
|
bool CBitcoinAddress::Set(const CKeyID& id)
|
||||||
|
{
|
||||||
SetData(Params().Base58Prefix(CChainParams::PUBKEY_ADDRESS), &id, 20);
|
SetData(Params().Base58Prefix(CChainParams::PUBKEY_ADDRESS), &id, 20);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinAddress::Set(const CScriptID &id) {
|
bool CBitcoinAddress::Set(const CScriptID& id)
|
||||||
|
{
|
||||||
SetData(Params().Base58Prefix(CChainParams::SCRIPT_ADDRESS), &id, 20);
|
SetData(Params().Base58Prefix(CChainParams::SCRIPT_ADDRESS), &id, 20);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinAddress::Set(const CTxDestination &dest) {
|
bool CBitcoinAddress::Set(const CTxDestination& dest)
|
||||||
|
{
|
||||||
return boost::apply_visitor(CBitcoinAddressVisitor(this), dest);
|
return boost::apply_visitor(CBitcoinAddressVisitor(this), dest);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinAddress::IsValid() const {
|
bool CBitcoinAddress::IsValid() const
|
||||||
|
{
|
||||||
return IsValid(Params());
|
return IsValid(Params());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinAddress::IsValid(const CChainParams ¶ms) const {
|
bool CBitcoinAddress::IsValid(const CChainParams& params) const
|
||||||
|
{
|
||||||
bool fCorrectSize = vchData.size() == 20;
|
bool fCorrectSize = vchData.size() == 20;
|
||||||
bool fKnownVersion = vchVersion == params.Base58Prefix(CChainParams::PUBKEY_ADDRESS) ||
|
bool fKnownVersion = vchVersion == params.Base58Prefix(CChainParams::PUBKEY_ADDRESS) ||
|
||||||
vchVersion == params.Base58Prefix(CChainParams::SCRIPT_ADDRESS);
|
vchVersion == params.Base58Prefix(CChainParams::SCRIPT_ADDRESS);
|
||||||
return fCorrectSize && fKnownVersion;
|
return fCorrectSize && fKnownVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
CTxDestination CBitcoinAddress::Get() const {
|
CTxDestination CBitcoinAddress::Get() const
|
||||||
|
{
|
||||||
if (!IsValid())
|
if (!IsValid())
|
||||||
return CNoDestination();
|
return CNoDestination();
|
||||||
uint160 id;
|
uint160 id;
|
||||||
|
@ -238,7 +262,8 @@ CTxDestination CBitcoinAddress::Get() const {
|
||||||
return CNoDestination();
|
return CNoDestination();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinAddress::GetKeyID(CKeyID &keyID) const {
|
bool CBitcoinAddress::GetKeyID(CKeyID& keyID) const
|
||||||
|
{
|
||||||
if (!IsValid() || vchVersion != Params().Base58Prefix(CChainParams::PUBKEY_ADDRESS))
|
if (!IsValid() || vchVersion != Params().Base58Prefix(CChainParams::PUBKEY_ADDRESS))
|
||||||
return false;
|
return false;
|
||||||
uint160 id;
|
uint160 id;
|
||||||
|
@ -247,33 +272,39 @@ bool CBitcoinAddress::GetKeyID(CKeyID &keyID) const {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinAddress::IsScript() const {
|
bool CBitcoinAddress::IsScript() const
|
||||||
|
{
|
||||||
return IsValid() && vchVersion == Params().Base58Prefix(CChainParams::SCRIPT_ADDRESS);
|
return IsValid() && vchVersion == Params().Base58Prefix(CChainParams::SCRIPT_ADDRESS);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CBitcoinSecret::SetKey(const CKey& vchSecret) {
|
void CBitcoinSecret::SetKey(const CKey& vchSecret)
|
||||||
|
{
|
||||||
assert(vchSecret.IsValid());
|
assert(vchSecret.IsValid());
|
||||||
SetData(Params().Base58Prefix(CChainParams::SECRET_KEY), vchSecret.begin(), vchSecret.size());
|
SetData(Params().Base58Prefix(CChainParams::SECRET_KEY), vchSecret.begin(), vchSecret.size());
|
||||||
if (vchSecret.IsCompressed())
|
if (vchSecret.IsCompressed())
|
||||||
vchData.push_back(1);
|
vchData.push_back(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
CKey CBitcoinSecret::GetKey() {
|
CKey CBitcoinSecret::GetKey()
|
||||||
|
{
|
||||||
CKey ret;
|
CKey ret;
|
||||||
ret.Set(&vchData[0], &vchData[32], vchData.size() > 32 && vchData[32] == 1);
|
ret.Set(&vchData[0], &vchData[32], vchData.size() > 32 && vchData[32] == 1);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinSecret::IsValid() const {
|
bool CBitcoinSecret::IsValid() const
|
||||||
|
{
|
||||||
bool fExpectedFormat = vchData.size() == 32 || (vchData.size() == 33 && vchData[32] == 1);
|
bool fExpectedFormat = vchData.size() == 32 || (vchData.size() == 33 && vchData[32] == 1);
|
||||||
bool fCorrectVersion = vchVersion == Params().Base58Prefix(CChainParams::SECRET_KEY);
|
bool fCorrectVersion = vchVersion == Params().Base58Prefix(CChainParams::SECRET_KEY);
|
||||||
return fExpectedFormat && fCorrectVersion;
|
return fExpectedFormat && fCorrectVersion;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinSecret::SetString(const char* pszSecret) {
|
bool CBitcoinSecret::SetString(const char* pszSecret)
|
||||||
|
{
|
||||||
return CBase58Data::SetString(pszSecret) && IsValid();
|
return CBase58Data::SetString(pszSecret) && IsValid();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CBitcoinSecret::SetString(const std::string& strSecret) {
|
bool CBitcoinSecret::SetString(const std::string& strSecret)
|
||||||
|
{
|
||||||
return SetString(strSecret.c_str());
|
return SetString(strSecret.c_str());
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,9 +17,11 @@ using namespace boost::assign;
|
||||||
// Main network
|
// Main network
|
||||||
//
|
//
|
||||||
|
|
||||||
class CBaseMainParams : public CBaseChainParams {
|
class CBaseMainParams : public CBaseChainParams
|
||||||
|
{
|
||||||
public:
|
public:
|
||||||
CBaseMainParams() {
|
CBaseMainParams()
|
||||||
|
{
|
||||||
networkID = CBaseChainParams::MAIN;
|
networkID = CBaseChainParams::MAIN;
|
||||||
nRPCPort = 8332;
|
nRPCPort = 8332;
|
||||||
}
|
}
|
||||||
|
@ -29,9 +31,11 @@ static CBaseMainParams mainParams;
|
||||||
//
|
//
|
||||||
// Testnet (v3)
|
// Testnet (v3)
|
||||||
//
|
//
|
||||||
class CBaseTestNetParams : public CBaseMainParams {
|
class CBaseTestNetParams : public CBaseMainParams
|
||||||
|
{
|
||||||
public:
|
public:
|
||||||
CBaseTestNetParams() {
|
CBaseTestNetParams()
|
||||||
|
{
|
||||||
networkID = CBaseChainParams::TESTNET;
|
networkID = CBaseChainParams::TESTNET;
|
||||||
nRPCPort = 18332;
|
nRPCPort = 18332;
|
||||||
strDataDir = "testnet3";
|
strDataDir = "testnet3";
|
||||||
|
@ -42,40 +46,45 @@ static CBaseTestNetParams testNetParams;
|
||||||
//
|
//
|
||||||
// Regression test
|
// Regression test
|
||||||
//
|
//
|
||||||
class CBaseRegTestParams : public CBaseTestNetParams {
|
class CBaseRegTestParams : public CBaseTestNetParams
|
||||||
|
{
|
||||||
public:
|
public:
|
||||||
CBaseRegTestParams() {
|
CBaseRegTestParams()
|
||||||
|
{
|
||||||
networkID = CBaseChainParams::REGTEST;
|
networkID = CBaseChainParams::REGTEST;
|
||||||
strDataDir = "regtest";
|
strDataDir = "regtest";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
static CBaseRegTestParams regTestParams;
|
static CBaseRegTestParams regTestParams;
|
||||||
|
|
||||||
static CBaseChainParams *pCurrentBaseParams = 0;
|
static CBaseChainParams* pCurrentBaseParams = 0;
|
||||||
|
|
||||||
const CBaseChainParams &BaseParams() {
|
const CBaseChainParams& BaseParams()
|
||||||
|
{
|
||||||
assert(pCurrentBaseParams);
|
assert(pCurrentBaseParams);
|
||||||
return *pCurrentBaseParams;
|
return *pCurrentBaseParams;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SelectBaseParams(CBaseChainParams::Network network) {
|
void SelectBaseParams(CBaseChainParams::Network network)
|
||||||
|
{
|
||||||
switch (network) {
|
switch (network) {
|
||||||
case CBaseChainParams::MAIN:
|
case CBaseChainParams::MAIN:
|
||||||
pCurrentBaseParams = &mainParams;
|
pCurrentBaseParams = &mainParams;
|
||||||
break;
|
break;
|
||||||
case CBaseChainParams::TESTNET:
|
case CBaseChainParams::TESTNET:
|
||||||
pCurrentBaseParams = &testNetParams;
|
pCurrentBaseParams = &testNetParams;
|
||||||
break;
|
break;
|
||||||
case CBaseChainParams::REGTEST:
|
case CBaseChainParams::REGTEST:
|
||||||
pCurrentBaseParams = ®TestParams;
|
pCurrentBaseParams = ®TestParams;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
assert(false && "Unimplemented network");
|
assert(false && "Unimplemented network");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SelectBaseParamsFromCommandLine() {
|
bool SelectBaseParamsFromCommandLine()
|
||||||
|
{
|
||||||
bool fRegTest = GetBoolArg("-regtest", false);
|
bool fRegTest = GetBoolArg("-regtest", false);
|
||||||
bool fTestNet = GetBoolArg("-testnet", false);
|
bool fTestNet = GetBoolArg("-testnet", false);
|
||||||
|
|
||||||
|
@ -93,6 +102,7 @@ bool SelectBaseParamsFromCommandLine() {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool AreBaseParamsConfigured() {
|
bool AreBaseParamsConfigured()
|
||||||
|
{
|
||||||
return pCurrentBaseParams != NULL;
|
return pCurrentBaseParams != NULL;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,6 +26,7 @@ public:
|
||||||
const std::string& DataDir() const { return strDataDir; }
|
const std::string& DataDir() const { return strDataDir; }
|
||||||
int RPCPort() const { return nRPCPort; }
|
int RPCPort() const { return nRPCPort; }
|
||||||
Network NetworkID() const { return networkID; }
|
Network NetworkID() const { return networkID; }
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
CBaseChainParams() {}
|
CBaseChainParams() {}
|
||||||
|
|
||||||
|
@ -38,7 +39,7 @@ protected:
|
||||||
* Return the currently selected parameters. This won't change after app startup
|
* Return the currently selected parameters. This won't change after app startup
|
||||||
* outside of the unit tests.
|
* outside of the unit tests.
|
||||||
*/
|
*/
|
||||||
const CBaseChainParams &BaseParams();
|
const CBaseChainParams& BaseParams();
|
||||||
|
|
||||||
/** Sets the params returned by Params() to those for the given network. */
|
/** Sets the params returned by Params() to those for the given network. */
|
||||||
void SelectBaseParams(CBaseChainParams::Network network);
|
void SelectBaseParams(CBaseChainParams::Network network);
|
||||||
|
|
|
@ -13,20 +13,20 @@ class uint256;
|
||||||
/** Block-chain checkpoints are compiled-in sanity checks.
|
/** Block-chain checkpoints are compiled-in sanity checks.
|
||||||
* They are updated every release or three.
|
* They are updated every release or three.
|
||||||
*/
|
*/
|
||||||
namespace Checkpoints {
|
namespace Checkpoints
|
||||||
|
{
|
||||||
|
// Returns true if block passes checkpoint checks
|
||||||
|
bool CheckBlock(int nHeight, const uint256& hash);
|
||||||
|
|
||||||
// Returns true if block passes checkpoint checks
|
// Return conservative estimate of total number of blocks, 0 if unknown
|
||||||
bool CheckBlock(int nHeight, const uint256& hash);
|
int GetTotalBlocksEstimate();
|
||||||
|
|
||||||
// Return conservative estimate of total number of blocks, 0 if unknown
|
// Returns last CBlockIndex* in mapBlockIndex that is a checkpoint
|
||||||
int GetTotalBlocksEstimate();
|
CBlockIndex* GetLastCheckpoint();
|
||||||
|
|
||||||
// Returns last CBlockIndex* in mapBlockIndex that is a checkpoint
|
double GuessVerificationProgress(CBlockIndex* pindex, bool fSigchecks = true);
|
||||||
CBlockIndex* GetLastCheckpoint();
|
|
||||||
|
|
||||||
double GuessVerificationProgress(CBlockIndex *pindex, bool fSigchecks = true);
|
extern bool fEnabled;
|
||||||
|
|
||||||
extern bool fEnabled;
|
|
||||||
|
|
||||||
} //namespace Checkpoints
|
} //namespace Checkpoints
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,8 @@
|
||||||
#include <boost/thread/locks.hpp>
|
#include <boost/thread/locks.hpp>
|
||||||
#include <boost/thread/mutex.hpp>
|
#include <boost/thread/mutex.hpp>
|
||||||
|
|
||||||
template<typename T> class CCheckQueueControl;
|
template <typename T>
|
||||||
|
class CCheckQueueControl;
|
||||||
|
|
||||||
/** Queue for verifications that have to be performed.
|
/** Queue for verifications that have to be performed.
|
||||||
* The verifications are represented by a type T, which must provide an
|
* The verifications are represented by a type T, which must provide an
|
||||||
|
@ -24,7 +25,9 @@ template<typename T> class CCheckQueueControl;
|
||||||
* the master is done adding work, it temporarily joins the worker pool
|
* the master is done adding work, it temporarily joins the worker pool
|
||||||
* as an N'th worker, until all jobs are done.
|
* as an N'th worker, until all jobs are done.
|
||||||
*/
|
*/
|
||||||
template<typename T> class CCheckQueue {
|
template <typename T>
|
||||||
|
class CCheckQueue
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
// Mutex to protect the inner state
|
// Mutex to protect the inner state
|
||||||
boost::mutex mutex;
|
boost::mutex mutex;
|
||||||
|
@ -60,8 +63,9 @@ private:
|
||||||
unsigned int nBatchSize;
|
unsigned int nBatchSize;
|
||||||
|
|
||||||
// Internal function that does bulk of the verification work.
|
// Internal function that does bulk of the verification work.
|
||||||
bool Loop(bool fMaster = false) {
|
bool Loop(bool fMaster = false)
|
||||||
boost::condition_variable &cond = fMaster ? condMaster : condWorker;
|
{
|
||||||
|
boost::condition_variable& cond = fMaster ? condMaster : condWorker;
|
||||||
std::vector<T> vChecks;
|
std::vector<T> vChecks;
|
||||||
vChecks.reserve(nBatchSize);
|
vChecks.reserve(nBatchSize);
|
||||||
unsigned int nNow = 0;
|
unsigned int nNow = 0;
|
||||||
|
@ -103,41 +107,43 @@ private:
|
||||||
nNow = std::max(1U, std::min(nBatchSize, (unsigned int)queue.size() / (nTotal + nIdle + 1)));
|
nNow = std::max(1U, std::min(nBatchSize, (unsigned int)queue.size() / (nTotal + nIdle + 1)));
|
||||||
vChecks.resize(nNow);
|
vChecks.resize(nNow);
|
||||||
for (unsigned int i = 0; i < nNow; i++) {
|
for (unsigned int i = 0; i < nNow; i++) {
|
||||||
// We want the lock on the mutex to be as short as possible, so swap jobs from the global
|
// We want the lock on the mutex to be as short as possible, so swap jobs from the global
|
||||||
// queue to the local batch vector instead of copying.
|
// queue to the local batch vector instead of copying.
|
||||||
vChecks[i].swap(queue.back());
|
vChecks[i].swap(queue.back());
|
||||||
queue.pop_back();
|
queue.pop_back();
|
||||||
}
|
}
|
||||||
// Check whether we need to do work at all
|
// Check whether we need to do work at all
|
||||||
fOk = fAllOk;
|
fOk = fAllOk;
|
||||||
}
|
}
|
||||||
// execute work
|
// execute work
|
||||||
BOOST_FOREACH(T &check, vChecks)
|
BOOST_FOREACH (T& check, vChecks)
|
||||||
if (fOk)
|
if (fOk)
|
||||||
fOk = check();
|
fOk = check();
|
||||||
vChecks.clear();
|
vChecks.clear();
|
||||||
} while(true);
|
} while (true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Create a new check queue
|
// Create a new check queue
|
||||||
CCheckQueue(unsigned int nBatchSizeIn) :
|
CCheckQueue(unsigned int nBatchSizeIn) : nIdle(0), nTotal(0), fAllOk(true), nTodo(0), fQuit(false), nBatchSize(nBatchSizeIn) {}
|
||||||
nIdle(0), nTotal(0), fAllOk(true), nTodo(0), fQuit(false), nBatchSize(nBatchSizeIn) {}
|
|
||||||
|
|
||||||
// Worker thread
|
// Worker thread
|
||||||
void Thread() {
|
void Thread()
|
||||||
|
{
|
||||||
Loop();
|
Loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wait until execution finishes, and return whether all evaluations where succesful.
|
// Wait until execution finishes, and return whether all evaluations where succesful.
|
||||||
bool Wait() {
|
bool Wait()
|
||||||
|
{
|
||||||
return Loop(true);
|
return Loop(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add a batch of checks to the queue
|
// Add a batch of checks to the queue
|
||||||
void Add(std::vector<T> &vChecks) {
|
void Add(std::vector<T>& vChecks)
|
||||||
|
{
|
||||||
boost::unique_lock<boost::mutex> lock(mutex);
|
boost::unique_lock<boost::mutex> lock(mutex);
|
||||||
BOOST_FOREACH(T &check, vChecks) {
|
BOOST_FOREACH (T& check, vChecks) {
|
||||||
queue.push_back(T());
|
queue.push_back(T());
|
||||||
check.swap(queue.back());
|
check.swap(queue.back());
|
||||||
}
|
}
|
||||||
|
@ -148,7 +154,8 @@ public:
|
||||||
condWorker.notify_all();
|
condWorker.notify_all();
|
||||||
}
|
}
|
||||||
|
|
||||||
~CCheckQueue() {
|
~CCheckQueue()
|
||||||
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
friend class CCheckQueueControl<T>;
|
friend class CCheckQueueControl<T>;
|
||||||
|
@ -157,13 +164,16 @@ public:
|
||||||
/** RAII-style controller object for a CCheckQueue that guarantees the passed
|
/** RAII-style controller object for a CCheckQueue that guarantees the passed
|
||||||
* queue is finished before continuing.
|
* queue is finished before continuing.
|
||||||
*/
|
*/
|
||||||
template<typename T> class CCheckQueueControl {
|
template <typename T>
|
||||||
|
class CCheckQueueControl
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
CCheckQueue<T> *pqueue;
|
CCheckQueue<T>* pqueue;
|
||||||
bool fDone;
|
bool fDone;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CCheckQueueControl(CCheckQueue<T> *pqueueIn) : pqueue(pqueueIn), fDone(false) {
|
CCheckQueueControl(CCheckQueue<T>* pqueueIn) : pqueue(pqueueIn), fDone(false)
|
||||||
|
{
|
||||||
// passed queue is supposed to be unused, or NULL
|
// passed queue is supposed to be unused, or NULL
|
||||||
if (pqueue != NULL) {
|
if (pqueue != NULL) {
|
||||||
assert(pqueue->nTotal == pqueue->nIdle);
|
assert(pqueue->nTotal == pqueue->nIdle);
|
||||||
|
@ -172,7 +182,8 @@ public:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Wait() {
|
bool Wait()
|
||||||
|
{
|
||||||
if (pqueue == NULL)
|
if (pqueue == NULL)
|
||||||
return true;
|
return true;
|
||||||
bool fRet = pqueue->Wait();
|
bool fRet = pqueue->Wait();
|
||||||
|
@ -180,12 +191,14 @@ public:
|
||||||
return fRet;
|
return fRet;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Add(std::vector<T> &vChecks) {
|
void Add(std::vector<T>& vChecks)
|
||||||
|
{
|
||||||
if (pqueue != NULL)
|
if (pqueue != NULL)
|
||||||
pqueue->Add(vChecks);
|
pqueue->Add(vChecks);
|
||||||
}
|
}
|
||||||
|
|
||||||
~CCheckQueueControl() {
|
~CCheckQueueControl()
|
||||||
|
{
|
||||||
if (!fDone)
|
if (!fDone)
|
||||||
Wait();
|
Wait();
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,13 +13,13 @@
|
||||||
//
|
//
|
||||||
|
|
||||||
// These need to be macros, as version.cpp's and bitcoin*-res.rc's voodoo requires it
|
// These need to be macros, as version.cpp's and bitcoin*-res.rc's voodoo requires it
|
||||||
#define CLIENT_VERSION_MAJOR 0
|
#define CLIENT_VERSION_MAJOR 0
|
||||||
#define CLIENT_VERSION_MINOR 9
|
#define CLIENT_VERSION_MINOR 9
|
||||||
#define CLIENT_VERSION_REVISION 99
|
#define CLIENT_VERSION_REVISION 99
|
||||||
#define CLIENT_VERSION_BUILD 0
|
#define CLIENT_VERSION_BUILD 0
|
||||||
|
|
||||||
// Set to true for release, false for prerelease or test build
|
// Set to true for release, false for prerelease or test build
|
||||||
#define CLIENT_VERSION_IS_RELEASE false
|
#define CLIENT_VERSION_IS_RELEASE false
|
||||||
|
|
||||||
// Copyright year (2009-this)
|
// Copyright year (2009-this)
|
||||||
// Todo: update this when changing our copyright comments in the source
|
// Todo: update this when changing our copyright comments in the source
|
||||||
|
@ -33,6 +33,6 @@
|
||||||
#define DO_STRINGIZE(X) #X
|
#define DO_STRINGIZE(X) #X
|
||||||
|
|
||||||
// Copyright string used in Windows .rc files
|
// Copyright string used in Windows .rc files
|
||||||
#define COPYRIGHT_STR "2009-" STRINGIZE(COPYRIGHT_YEAR) " The Bitcoin Core Developers"
|
#define COPYRIGHT_STR "2009-" STRINGIZE(COPYRIGHT_YEAR) " The Bitcoin Core Developers"
|
||||||
|
|
||||||
#endif // CLIENTVERSION_H
|
#endif // CLIENTVERSION_H
|
||||||
|
|
|
@ -57,7 +57,6 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::set<COutPoint> setSelected;
|
std::set<COutPoint> setSelected;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // COINCONTROL_H
|
#endif // COINCONTROL_H
|
||||||
|
|
144
src/db.cpp
144
src/db.cpp
|
@ -30,7 +30,6 @@ using namespace boost;
|
||||||
unsigned int nWalletDBUpdated;
|
unsigned int nWalletDBUpdated;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// CDB
|
// CDB
|
||||||
//
|
//
|
||||||
|
@ -94,15 +93,15 @@ bool CDBEnv::Open(const boost::filesystem::path& pathIn)
|
||||||
dbenv.set_flags(DB_TXN_WRITE_NOSYNC, 1);
|
dbenv.set_flags(DB_TXN_WRITE_NOSYNC, 1);
|
||||||
dbenv.log_set_config(DB_LOG_AUTO_REMOVE, 1);
|
dbenv.log_set_config(DB_LOG_AUTO_REMOVE, 1);
|
||||||
int ret = dbenv.open(path.string().c_str(),
|
int ret = dbenv.open(path.string().c_str(),
|
||||||
DB_CREATE |
|
DB_CREATE |
|
||||||
DB_INIT_LOCK |
|
DB_INIT_LOCK |
|
||||||
DB_INIT_LOG |
|
DB_INIT_LOG |
|
||||||
DB_INIT_MPOOL |
|
DB_INIT_MPOOL |
|
||||||
DB_INIT_TXN |
|
DB_INIT_TXN |
|
||||||
DB_THREAD |
|
DB_THREAD |
|
||||||
DB_RECOVER |
|
DB_RECOVER |
|
||||||
nEnvFlags,
|
nEnvFlags,
|
||||||
S_IRUSR | S_IWUSR);
|
S_IRUSR | S_IWUSR);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
return error("CDBEnv::Open : Error %d opening database environment: %s\n", ret, DbEnv::strerror(ret));
|
return error("CDBEnv::Open : Error %d opening database environment: %s\n", ret, DbEnv::strerror(ret));
|
||||||
|
|
||||||
|
@ -121,21 +120,21 @@ void CDBEnv::MakeMock()
|
||||||
LogPrint("db", "CDBEnv::MakeMock\n");
|
LogPrint("db", "CDBEnv::MakeMock\n");
|
||||||
|
|
||||||
dbenv.set_cachesize(1, 0, 1);
|
dbenv.set_cachesize(1, 0, 1);
|
||||||
dbenv.set_lg_bsize(10485760*4);
|
dbenv.set_lg_bsize(10485760 * 4);
|
||||||
dbenv.set_lg_max(10485760);
|
dbenv.set_lg_max(10485760);
|
||||||
dbenv.set_lk_max_locks(10000);
|
dbenv.set_lk_max_locks(10000);
|
||||||
dbenv.set_lk_max_objects(10000);
|
dbenv.set_lk_max_objects(10000);
|
||||||
dbenv.set_flags(DB_AUTO_COMMIT, 1);
|
dbenv.set_flags(DB_AUTO_COMMIT, 1);
|
||||||
dbenv.log_set_config(DB_LOG_IN_MEMORY, 1);
|
dbenv.log_set_config(DB_LOG_IN_MEMORY, 1);
|
||||||
int ret = dbenv.open(NULL,
|
int ret = dbenv.open(NULL,
|
||||||
DB_CREATE |
|
DB_CREATE |
|
||||||
DB_INIT_LOCK |
|
DB_INIT_LOCK |
|
||||||
DB_INIT_LOG |
|
DB_INIT_LOG |
|
||||||
DB_INIT_MPOOL |
|
DB_INIT_MPOOL |
|
||||||
DB_INIT_TXN |
|
DB_INIT_TXN |
|
||||||
DB_THREAD |
|
DB_THREAD |
|
||||||
DB_PRIVATE,
|
DB_PRIVATE,
|
||||||
S_IRUSR | S_IWUSR);
|
S_IRUSR | S_IWUSR);
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
throw runtime_error(strprintf("CDBEnv::MakeMock : Error %d opening database environment.", ret));
|
throw runtime_error(strprintf("CDBEnv::MakeMock : Error %d opening database environment.", ret));
|
||||||
|
|
||||||
|
@ -160,30 +159,27 @@ CDBEnv::VerifyResult CDBEnv::Verify(std::string strFile, bool (*recoverFunc)(CDB
|
||||||
return (fRecovered ? RECOVER_OK : RECOVER_FAIL);
|
return (fRecovered ? RECOVER_OK : RECOVER_FAIL);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CDBEnv::Salvage(std::string strFile, bool fAggressive,
|
bool CDBEnv::Salvage(std::string strFile, bool fAggressive, std::vector<CDBEnv::KeyValPair>& vResult)
|
||||||
std::vector<CDBEnv::KeyValPair >& vResult)
|
|
||||||
{
|
{
|
||||||
LOCK(cs_db);
|
LOCK(cs_db);
|
||||||
assert(mapFileUseCount.count(strFile) == 0);
|
assert(mapFileUseCount.count(strFile) == 0);
|
||||||
|
|
||||||
u_int32_t flags = DB_SALVAGE;
|
u_int32_t flags = DB_SALVAGE;
|
||||||
if (fAggressive) flags |= DB_AGGRESSIVE;
|
if (fAggressive)
|
||||||
|
flags |= DB_AGGRESSIVE;
|
||||||
|
|
||||||
stringstream strDump;
|
stringstream strDump;
|
||||||
|
|
||||||
Db db(&dbenv, 0);
|
Db db(&dbenv, 0);
|
||||||
int result = db.verify(strFile.c_str(), NULL, &strDump, flags);
|
int result = db.verify(strFile.c_str(), NULL, &strDump, flags);
|
||||||
if (result == DB_VERIFY_BAD)
|
if (result == DB_VERIFY_BAD) {
|
||||||
{
|
|
||||||
LogPrintf("CDBEnv::Salvage : Database salvage found errors, all data may not be recoverable.\n");
|
LogPrintf("CDBEnv::Salvage : Database salvage found errors, all data may not be recoverable.\n");
|
||||||
if (!fAggressive)
|
if (!fAggressive) {
|
||||||
{
|
|
||||||
LogPrintf("CDBEnv::Salvage : Rerun with aggressive mode to ignore errors and continue.\n");
|
LogPrintf("CDBEnv::Salvage : Rerun with aggressive mode to ignore errors and continue.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (result != 0 && result != DB_VERIFY_BAD)
|
if (result != 0 && result != DB_VERIFY_BAD) {
|
||||||
{
|
|
||||||
LogPrintf("CDBEnv::Salvage : Database salvage failed with result %d.\n", result);
|
LogPrintf("CDBEnv::Salvage : Database salvage failed with result %d.\n", result);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -201,13 +197,11 @@ bool CDBEnv::Salvage(std::string strFile, bool fAggressive,
|
||||||
getline(strDump, strLine); // Skip past header
|
getline(strDump, strLine); // Skip past header
|
||||||
|
|
||||||
std::string keyHex, valueHex;
|
std::string keyHex, valueHex;
|
||||||
while (!strDump.eof() && keyHex != "DATA=END")
|
while (!strDump.eof() && keyHex != "DATA=END") {
|
||||||
{
|
|
||||||
getline(strDump, keyHex);
|
getline(strDump, keyHex);
|
||||||
if (keyHex != "DATA_END")
|
if (keyHex != "DATA_END") {
|
||||||
{
|
|
||||||
getline(strDump, valueHex);
|
getline(strDump, valueHex);
|
||||||
vResult.push_back(make_pair(ParseHex(keyHex),ParseHex(valueHex)));
|
vResult.push_back(make_pair(ParseHex(keyHex), ParseHex(valueHex)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -224,8 +218,7 @@ void CDBEnv::CheckpointLSN(const std::string& strFile)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
CDB::CDB(const std::string& strFilename, const char* pszMode) :
|
CDB::CDB(const std::string& strFilename, const char* pszMode) : pdb(NULL), activeTxn(NULL)
|
||||||
pdb(NULL), activeTxn(NULL)
|
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
fReadOnly = (!strchr(pszMode, '+') && !strchr(pszMode, 'w'));
|
fReadOnly = (!strchr(pszMode, '+') && !strchr(pszMode, 'w'));
|
||||||
|
@ -245,28 +238,25 @@ CDB::CDB(const std::string& strFilename, const char* pszMode) :
|
||||||
strFile = strFilename;
|
strFile = strFilename;
|
||||||
++bitdb.mapFileUseCount[strFile];
|
++bitdb.mapFileUseCount[strFile];
|
||||||
pdb = bitdb.mapDb[strFile];
|
pdb = bitdb.mapDb[strFile];
|
||||||
if (pdb == NULL)
|
if (pdb == NULL) {
|
||||||
{
|
|
||||||
pdb = new Db(&bitdb.dbenv, 0);
|
pdb = new Db(&bitdb.dbenv, 0);
|
||||||
|
|
||||||
bool fMockDb = bitdb.IsMock();
|
bool fMockDb = bitdb.IsMock();
|
||||||
if (fMockDb)
|
if (fMockDb) {
|
||||||
{
|
DbMpoolFile* mpf = pdb->get_mpf();
|
||||||
DbMpoolFile*mpf = pdb->get_mpf();
|
|
||||||
ret = mpf->set_flags(DB_MPOOL_NOFILE, 1);
|
ret = mpf->set_flags(DB_MPOOL_NOFILE, 1);
|
||||||
if (ret != 0)
|
if (ret != 0)
|
||||||
throw runtime_error(strprintf("CDB : Failed to configure for no temp file backing for database %s", strFile));
|
throw runtime_error(strprintf("CDB : Failed to configure for no temp file backing for database %s", strFile));
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = pdb->open(NULL, // Txn pointer
|
ret = pdb->open(NULL, // Txn pointer
|
||||||
fMockDb ? NULL : strFile.c_str(), // Filename
|
fMockDb ? NULL : strFile.c_str(), // Filename
|
||||||
fMockDb ? strFile.c_str() : "main", // Logical db name
|
fMockDb ? strFile.c_str() : "main", // Logical db name
|
||||||
DB_BTREE, // Database type
|
DB_BTREE, // Database type
|
||||||
nFlags, // Flags
|
nFlags, // Flags
|
||||||
0);
|
0);
|
||||||
|
|
||||||
if (ret != 0)
|
if (ret != 0) {
|
||||||
{
|
|
||||||
delete pdb;
|
delete pdb;
|
||||||
pdb = NULL;
|
pdb = NULL;
|
||||||
--bitdb.mapFileUseCount[strFile];
|
--bitdb.mapFileUseCount[strFile];
|
||||||
|
@ -274,8 +264,7 @@ CDB::CDB(const std::string& strFilename, const char* pszMode) :
|
||||||
throw runtime_error(strprintf("CDB : Error %d, can't open database %s", ret, strFile));
|
throw runtime_error(strprintf("CDB : Error %d, can't open database %s", ret, strFile));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fCreate && !Exists(string("version")))
|
if (fCreate && !Exists(string("version"))) {
|
||||||
{
|
|
||||||
bool fTmp = fReadOnly;
|
bool fTmp = fReadOnly;
|
||||||
fReadOnly = false;
|
fReadOnly = false;
|
||||||
WriteVersion(CLIENT_VERSION);
|
WriteVersion(CLIENT_VERSION);
|
||||||
|
@ -297,7 +286,7 @@ void CDB::Flush()
|
||||||
if (fReadOnly)
|
if (fReadOnly)
|
||||||
nMinutes = 1;
|
nMinutes = 1;
|
||||||
|
|
||||||
bitdb.dbenv.txn_checkpoint(nMinutes ? GetArg("-dblogsize", 100)*1024 : 0, nMinutes, 0);
|
bitdb.dbenv.txn_checkpoint(nMinutes ? GetArg("-dblogsize", 100) * 1024 : 0, nMinutes, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CDB::Close()
|
void CDB::Close()
|
||||||
|
@ -321,8 +310,7 @@ void CDBEnv::CloseDb(const string& strFile)
|
||||||
{
|
{
|
||||||
{
|
{
|
||||||
LOCK(cs_db);
|
LOCK(cs_db);
|
||||||
if (mapDb[strFile] != NULL)
|
if (mapDb[strFile] != NULL) {
|
||||||
{
|
|
||||||
// Close the database handle
|
// Close the database handle
|
||||||
Db* pdb = mapDb[strFile];
|
Db* pdb = mapDb[strFile];
|
||||||
pdb->close(0);
|
pdb->close(0);
|
||||||
|
@ -343,12 +331,10 @@ bool CDBEnv::RemoveDb(const string& strFile)
|
||||||
|
|
||||||
bool CDB::Rewrite(const string& strFile, const char* pszSkip)
|
bool CDB::Rewrite(const string& strFile, const char* pszSkip)
|
||||||
{
|
{
|
||||||
while (true)
|
while (true) {
|
||||||
{
|
|
||||||
{
|
{
|
||||||
LOCK(bitdb.cs_db);
|
LOCK(bitdb.cs_db);
|
||||||
if (!bitdb.mapFileUseCount.count(strFile) || bitdb.mapFileUseCount[strFile] == 0)
|
if (!bitdb.mapFileUseCount.count(strFile) || bitdb.mapFileUseCount[strFile] == 0) {
|
||||||
{
|
|
||||||
// Flush log data to the dat file
|
// Flush log data to the dat file
|
||||||
bitdb.CloseDb(strFile);
|
bitdb.CloseDb(strFile);
|
||||||
bitdb.CheckpointLSN(strFile);
|
bitdb.CheckpointLSN(strFile);
|
||||||
|
@ -361,32 +347,27 @@ bool CDB::Rewrite(const string& strFile, const char* pszSkip)
|
||||||
CDB db(strFile.c_str(), "r");
|
CDB db(strFile.c_str(), "r");
|
||||||
Db* pdbCopy = new Db(&bitdb.dbenv, 0);
|
Db* pdbCopy = new Db(&bitdb.dbenv, 0);
|
||||||
|
|
||||||
int ret = pdbCopy->open(NULL, // Txn pointer
|
int ret = pdbCopy->open(NULL, // Txn pointer
|
||||||
strFileRes.c_str(), // Filename
|
strFileRes.c_str(), // Filename
|
||||||
"main", // Logical db name
|
"main", // Logical db name
|
||||||
DB_BTREE, // Database type
|
DB_BTREE, // Database type
|
||||||
DB_CREATE, // Flags
|
DB_CREATE, // Flags
|
||||||
0);
|
0);
|
||||||
if (ret > 0)
|
if (ret > 0) {
|
||||||
{
|
|
||||||
LogPrintf("CDB::Rewrite : Can't create database file %s\n", strFileRes);
|
LogPrintf("CDB::Rewrite : Can't create database file %s\n", strFileRes);
|
||||||
fSuccess = false;
|
fSuccess = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Dbc* pcursor = db.GetCursor();
|
Dbc* pcursor = db.GetCursor();
|
||||||
if (pcursor)
|
if (pcursor)
|
||||||
while (fSuccess)
|
while (fSuccess) {
|
||||||
{
|
|
||||||
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
||||||
CDataStream ssValue(SER_DISK, CLIENT_VERSION);
|
CDataStream ssValue(SER_DISK, CLIENT_VERSION);
|
||||||
int ret = db.ReadAtCursor(pcursor, ssKey, ssValue, DB_NEXT);
|
int ret = db.ReadAtCursor(pcursor, ssKey, ssValue, DB_NEXT);
|
||||||
if (ret == DB_NOTFOUND)
|
if (ret == DB_NOTFOUND) {
|
||||||
{
|
|
||||||
pcursor->close();
|
pcursor->close();
|
||||||
break;
|
break;
|
||||||
}
|
} else if (ret != 0) {
|
||||||
else if (ret != 0)
|
|
||||||
{
|
|
||||||
pcursor->close();
|
pcursor->close();
|
||||||
fSuccess = false;
|
fSuccess = false;
|
||||||
break;
|
break;
|
||||||
|
@ -394,8 +375,7 @@ bool CDB::Rewrite(const string& strFile, const char* pszSkip)
|
||||||
if (pszSkip &&
|
if (pszSkip &&
|
||||||
strncmp(&ssKey[0], pszSkip, std::min(ssKey.size(), strlen(pszSkip))) == 0)
|
strncmp(&ssKey[0], pszSkip, std::min(ssKey.size(), strlen(pszSkip))) == 0)
|
||||||
continue;
|
continue;
|
||||||
if (strncmp(&ssKey[0], "\x07version", 8) == 0)
|
if (strncmp(&ssKey[0], "\x07version", 8) == 0) {
|
||||||
{
|
|
||||||
// Update version:
|
// Update version:
|
||||||
ssValue.clear();
|
ssValue.clear();
|
||||||
ssValue << CLIENT_VERSION;
|
ssValue << CLIENT_VERSION;
|
||||||
|
@ -406,8 +386,7 @@ bool CDB::Rewrite(const string& strFile, const char* pszSkip)
|
||||||
if (ret2 > 0)
|
if (ret2 > 0)
|
||||||
fSuccess = false;
|
fSuccess = false;
|
||||||
}
|
}
|
||||||
if (fSuccess)
|
if (fSuccess) {
|
||||||
{
|
|
||||||
db.Close();
|
db.Close();
|
||||||
bitdb.CloseDb(strFile);
|
bitdb.CloseDb(strFile);
|
||||||
if (pdbCopy->close(0))
|
if (pdbCopy->close(0))
|
||||||
|
@ -415,8 +394,7 @@ bool CDB::Rewrite(const string& strFile, const char* pszSkip)
|
||||||
delete pdbCopy;
|
delete pdbCopy;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (fSuccess)
|
if (fSuccess) {
|
||||||
{
|
|
||||||
Db dbA(&bitdb.dbenv, 0);
|
Db dbA(&bitdb.dbenv, 0);
|
||||||
if (dbA.remove(strFile.c_str(), NULL, 0))
|
if (dbA.remove(strFile.c_str(), NULL, 0))
|
||||||
fSuccess = false;
|
fSuccess = false;
|
||||||
|
@ -445,13 +423,11 @@ void CDBEnv::Flush(bool fShutdown)
|
||||||
{
|
{
|
||||||
LOCK(cs_db);
|
LOCK(cs_db);
|
||||||
map<string, int>::iterator mi = mapFileUseCount.begin();
|
map<string, int>::iterator mi = mapFileUseCount.begin();
|
||||||
while (mi != mapFileUseCount.end())
|
while (mi != mapFileUseCount.end()) {
|
||||||
{
|
|
||||||
string strFile = (*mi).first;
|
string strFile = (*mi).first;
|
||||||
int nRefCount = (*mi).second;
|
int nRefCount = (*mi).second;
|
||||||
LogPrint("db", "CDBEnv::Flush : Flushing %s (refcount = %d)...\n", strFile, nRefCount);
|
LogPrint("db", "CDBEnv::Flush : Flushing %s (refcount = %d)...\n", strFile, nRefCount);
|
||||||
if (nRefCount == 0)
|
if (nRefCount == 0) {
|
||||||
{
|
|
||||||
// Move log data to the dat file
|
// Move log data to the dat file
|
||||||
CloseDb(strFile);
|
CloseDb(strFile);
|
||||||
LogPrint("db", "CDBEnv::Flush : %s checkpoint\n", strFile);
|
LogPrint("db", "CDBEnv::Flush : %s checkpoint\n", strFile);
|
||||||
|
@ -461,16 +437,13 @@ void CDBEnv::Flush(bool fShutdown)
|
||||||
dbenv.lsn_reset(strFile.c_str(), 0);
|
dbenv.lsn_reset(strFile.c_str(), 0);
|
||||||
LogPrint("db", "CDBEnv::Flush : %s closed\n", strFile);
|
LogPrint("db", "CDBEnv::Flush : %s closed\n", strFile);
|
||||||
mapFileUseCount.erase(mi++);
|
mapFileUseCount.erase(mi++);
|
||||||
}
|
} else
|
||||||
else
|
|
||||||
mi++;
|
mi++;
|
||||||
}
|
}
|
||||||
LogPrint("db", "CDBEnv::Flush : Flush(%s)%s took %15dms\n", fShutdown ? "true" : "false", fDbEnvInit ? "" : " database not started", GetTimeMillis() - nStart);
|
LogPrint("db", "CDBEnv::Flush : Flush(%s)%s took %15dms\n", fShutdown ? "true" : "false", fDbEnvInit ? "" : " database not started", GetTimeMillis() - nStart);
|
||||||
if (fShutdown)
|
if (fShutdown) {
|
||||||
{
|
|
||||||
char** listp;
|
char** listp;
|
||||||
if (mapFileUseCount.empty())
|
if (mapFileUseCount.empty()) {
|
||||||
{
|
|
||||||
dbenv.log_archive(&listp, DB_ARCH_REMOVE);
|
dbenv.log_archive(&listp, DB_ARCH_REMOVE);
|
||||||
Close();
|
Close();
|
||||||
if (!fMockDb)
|
if (!fMockDb)
|
||||||
|
@ -479,4 +452,3 @@ void CDBEnv::Flush(bool fShutdown)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
33
src/db.h
33
src/db.h
|
@ -54,7 +54,9 @@ public:
|
||||||
* This must be called BEFORE strFile is opened.
|
* This must be called BEFORE strFile is opened.
|
||||||
* Returns true if strFile is OK.
|
* Returns true if strFile is OK.
|
||||||
*/
|
*/
|
||||||
enum VerifyResult { VERIFY_OK, RECOVER_OK, RECOVER_FAIL };
|
enum VerifyResult { VERIFY_OK,
|
||||||
|
RECOVER_OK,
|
||||||
|
RECOVER_FAIL };
|
||||||
VerifyResult Verify(std::string strFile, bool (*recoverFunc)(CDBEnv& dbenv, std::string strFile));
|
VerifyResult Verify(std::string strFile, bool (*recoverFunc)(CDBEnv& dbenv, std::string strFile));
|
||||||
/*
|
/*
|
||||||
* Salvage data from a file that Verify says is bad.
|
* Salvage data from a file that Verify says is bad.
|
||||||
|
@ -66,7 +68,7 @@ public:
|
||||||
typedef std::pair<std::vector<unsigned char>, std::vector<unsigned char> > KeyValPair;
|
typedef std::pair<std::vector<unsigned char>, std::vector<unsigned char> > KeyValPair;
|
||||||
bool Salvage(std::string strFile, bool fAggressive, std::vector<KeyValPair>& vResult);
|
bool Salvage(std::string strFile, bool fAggressive, std::vector<KeyValPair>& vResult);
|
||||||
|
|
||||||
bool Open(const boost::filesystem::path &path);
|
bool Open(const boost::filesystem::path& path);
|
||||||
void Close();
|
void Close();
|
||||||
void Flush(bool fShutdown);
|
void Flush(bool fShutdown);
|
||||||
void CheckpointLSN(const std::string& strFile);
|
void CheckpointLSN(const std::string& strFile);
|
||||||
|
@ -74,7 +76,7 @@ public:
|
||||||
void CloseDb(const std::string& strFile);
|
void CloseDb(const std::string& strFile);
|
||||||
bool RemoveDb(const std::string& strFile);
|
bool RemoveDb(const std::string& strFile);
|
||||||
|
|
||||||
DbTxn *TxnBegin(int flags=DB_TXN_WRITE_NOSYNC)
|
DbTxn* TxnBegin(int flags = DB_TXN_WRITE_NOSYNC)
|
||||||
{
|
{
|
||||||
DbTxn* ptxn = NULL;
|
DbTxn* ptxn = NULL;
|
||||||
int ret = dbenv.txn_begin(NULL, &ptxn, flags);
|
int ret = dbenv.txn_begin(NULL, &ptxn, flags);
|
||||||
|
@ -93,10 +95,10 @@ class CDB
|
||||||
protected:
|
protected:
|
||||||
Db* pdb;
|
Db* pdb;
|
||||||
std::string strFile;
|
std::string strFile;
|
||||||
DbTxn *activeTxn;
|
DbTxn* activeTxn;
|
||||||
bool fReadOnly;
|
bool fReadOnly;
|
||||||
|
|
||||||
explicit CDB(const std::string& strFilename, const char* pszMode="r+");
|
explicit CDB(const std::string& strFilename, const char* pszMode = "r+");
|
||||||
~CDB() { Close(); }
|
~CDB() { Close(); }
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
@ -108,7 +110,7 @@ private:
|
||||||
void operator=(const CDB&);
|
void operator=(const CDB&);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
template<typename K, typename T>
|
template <typename K, typename T>
|
||||||
bool Read(const K& key, T& value)
|
bool Read(const K& key, T& value)
|
||||||
{
|
{
|
||||||
if (!pdb)
|
if (!pdb)
|
||||||
|
@ -132,8 +134,7 @@ protected:
|
||||||
try {
|
try {
|
||||||
CDataStream ssValue((char*)datValue.get_data(), (char*)datValue.get_data() + datValue.get_size(), SER_DISK, CLIENT_VERSION);
|
CDataStream ssValue((char*)datValue.get_data(), (char*)datValue.get_data() + datValue.get_size(), SER_DISK, CLIENT_VERSION);
|
||||||
ssValue >> value;
|
ssValue >> value;
|
||||||
}
|
} catch (const std::exception&) {
|
||||||
catch (const std::exception &) {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -143,8 +144,8 @@ protected:
|
||||||
return (ret == 0);
|
return (ret == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K, typename T>
|
template <typename K, typename T>
|
||||||
bool Write(const K& key, const T& value, bool fOverwrite=true)
|
bool Write(const K& key, const T& value, bool fOverwrite = true)
|
||||||
{
|
{
|
||||||
if (!pdb)
|
if (!pdb)
|
||||||
return false;
|
return false;
|
||||||
|
@ -172,7 +173,7 @@ protected:
|
||||||
return (ret == 0);
|
return (ret == 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K>
|
template <typename K>
|
||||||
bool Erase(const K& key)
|
bool Erase(const K& key)
|
||||||
{
|
{
|
||||||
if (!pdb)
|
if (!pdb)
|
||||||
|
@ -194,7 +195,7 @@ protected:
|
||||||
return (ret == 0 || ret == DB_NOTFOUND);
|
return (ret == 0 || ret == DB_NOTFOUND);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K>
|
template <typename K>
|
||||||
bool Exists(const K& key)
|
bool Exists(const K& key)
|
||||||
{
|
{
|
||||||
if (!pdb)
|
if (!pdb)
|
||||||
|
@ -225,18 +226,16 @@ protected:
|
||||||
return pcursor;
|
return pcursor;
|
||||||
}
|
}
|
||||||
|
|
||||||
int ReadAtCursor(Dbc* pcursor, CDataStream& ssKey, CDataStream& ssValue, unsigned int fFlags=DB_NEXT)
|
int ReadAtCursor(Dbc* pcursor, CDataStream& ssKey, CDataStream& ssValue, unsigned int fFlags = DB_NEXT)
|
||||||
{
|
{
|
||||||
// Read at cursor
|
// Read at cursor
|
||||||
Dbt datKey;
|
Dbt datKey;
|
||||||
if (fFlags == DB_SET || fFlags == DB_SET_RANGE || fFlags == DB_GET_BOTH || fFlags == DB_GET_BOTH_RANGE)
|
if (fFlags == DB_SET || fFlags == DB_SET_RANGE || fFlags == DB_GET_BOTH || fFlags == DB_GET_BOTH_RANGE) {
|
||||||
{
|
|
||||||
datKey.set_data(&ssKey[0]);
|
datKey.set_data(&ssKey[0]);
|
||||||
datKey.set_size(ssKey.size());
|
datKey.set_size(ssKey.size());
|
||||||
}
|
}
|
||||||
Dbt datValue;
|
Dbt datValue;
|
||||||
if (fFlags == DB_GET_BOTH || fFlags == DB_GET_BOTH_RANGE)
|
if (fFlags == DB_GET_BOTH || fFlags == DB_GET_BOTH_RANGE) {
|
||||||
{
|
|
||||||
datValue.set_data(&ssValue[0]);
|
datValue.set_data(&ssValue[0]);
|
||||||
datValue.set_size(ssValue.size());
|
datValue.set_size(ssValue.size());
|
||||||
}
|
}
|
||||||
|
|
32
src/hash.cpp
32
src/hash.cpp
|
@ -1,6 +1,6 @@
|
||||||
#include "hash.h"
|
#include "hash.h"
|
||||||
|
|
||||||
inline uint32_t ROTL32 ( uint32_t x, int8_t r )
|
inline uint32_t ROTL32(uint32_t x, int8_t r)
|
||||||
{
|
{
|
||||||
return (x << r) | (x >> (32 - r));
|
return (x << r) | (x >> (32 - r));
|
||||||
}
|
}
|
||||||
|
@ -16,33 +16,37 @@ unsigned int MurmurHash3(unsigned int nHashSeed, const std::vector<unsigned char
|
||||||
|
|
||||||
//----------
|
//----------
|
||||||
// body
|
// body
|
||||||
const uint32_t * blocks = (const uint32_t *)(&vDataToHash[0] + nblocks*4);
|
const uint32_t* blocks = (const uint32_t*)(&vDataToHash[0] + nblocks * 4);
|
||||||
|
|
||||||
for(int i = -nblocks; i; i++)
|
for (int i = -nblocks; i; i++) {
|
||||||
{
|
|
||||||
uint32_t k1 = blocks[i];
|
uint32_t k1 = blocks[i];
|
||||||
|
|
||||||
k1 *= c1;
|
k1 *= c1;
|
||||||
k1 = ROTL32(k1,15);
|
k1 = ROTL32(k1, 15);
|
||||||
k1 *= c2;
|
k1 *= c2;
|
||||||
|
|
||||||
h1 ^= k1;
|
h1 ^= k1;
|
||||||
h1 = ROTL32(h1,13);
|
h1 = ROTL32(h1, 13);
|
||||||
h1 = h1*5+0xe6546b64;
|
h1 = h1 * 5 + 0xe6546b64;
|
||||||
}
|
}
|
||||||
|
|
||||||
//----------
|
//----------
|
||||||
// tail
|
// tail
|
||||||
const uint8_t * tail = (const uint8_t*)(&vDataToHash[0] + nblocks*4);
|
const uint8_t* tail = (const uint8_t*)(&vDataToHash[0] + nblocks * 4);
|
||||||
|
|
||||||
uint32_t k1 = 0;
|
uint32_t k1 = 0;
|
||||||
|
|
||||||
switch(vDataToHash.size() & 3)
|
switch (vDataToHash.size() & 3) {
|
||||||
{
|
case 3:
|
||||||
case 3: k1 ^= tail[2] << 16;
|
k1 ^= tail[2] << 16;
|
||||||
case 2: k1 ^= tail[1] << 8;
|
case 2:
|
||||||
case 1: k1 ^= tail[0];
|
k1 ^= tail[1] << 8;
|
||||||
k1 *= c1; k1 = ROTL32(k1,15); k1 *= c2; h1 ^= k1;
|
case 1:
|
||||||
|
k1 ^= tail[0];
|
||||||
|
k1 *= c1;
|
||||||
|
k1 = ROTL32(k1, 15);
|
||||||
|
k1 *= c2;
|
||||||
|
h1 ^= k1;
|
||||||
};
|
};
|
||||||
|
|
||||||
//----------
|
//----------
|
||||||
|
|
|
@ -10,8 +10,9 @@
|
||||||
|
|
||||||
class CWallet;
|
class CWallet;
|
||||||
|
|
||||||
namespace boost {
|
namespace boost
|
||||||
class thread_group;
|
{
|
||||||
|
class thread_group;
|
||||||
} // namespace boost
|
} // namespace boost
|
||||||
|
|
||||||
extern CWallet* pwalletMain;
|
extern CWallet* pwalletMain;
|
||||||
|
@ -22,8 +23,7 @@ void Shutdown();
|
||||||
bool AppInit2(boost::thread_group& threadGroup);
|
bool AppInit2(boost::thread_group& threadGroup);
|
||||||
|
|
||||||
/* The help message mode determines what help message to show */
|
/* The help message mode determines what help message to show */
|
||||||
enum HelpMessageMode
|
enum HelpMessageMode {
|
||||||
{
|
|
||||||
HMM_BITCOIND,
|
HMM_BITCOIND,
|
||||||
HMM_BITCOIN_QT
|
HMM_BITCOIN_QT
|
||||||
};
|
};
|
||||||
|
|
131
src/key.h
131
src/key.h
|
@ -26,27 +26,29 @@
|
||||||
class CKeyID : public uint160
|
class CKeyID : public uint160
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CKeyID() : uint160(0) { }
|
CKeyID() : uint160(0) {}
|
||||||
CKeyID(const uint160 &in) : uint160(in) { }
|
CKeyID(const uint160& in) : uint160(in) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
/** A reference to a CScript: the Hash160 of its serialization (see script.h) */
|
/** A reference to a CScript: the Hash160 of its serialization (see script.h) */
|
||||||
class CScriptID : public uint160
|
class CScriptID : public uint160
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CScriptID() : uint160(0) { }
|
CScriptID() : uint160(0) {}
|
||||||
CScriptID(const uint160 &in) : uint160(in) { }
|
CScriptID(const uint160& in) : uint160(in) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
/** An encapsulated public key. */
|
/** An encapsulated public key. */
|
||||||
class CPubKey {
|
class CPubKey
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
// Just store the serialized data.
|
// Just store the serialized data.
|
||||||
// Its length can very cheaply be computed from the first byte.
|
// Its length can very cheaply be computed from the first byte.
|
||||||
unsigned char vch[65];
|
unsigned char vch[65];
|
||||||
|
|
||||||
// Compute the length of a pubkey with a given first byte.
|
// Compute the length of a pubkey with a given first byte.
|
||||||
unsigned int static GetLen(unsigned char chHeader) {
|
unsigned int static GetLen(unsigned char chHeader)
|
||||||
|
{
|
||||||
if (chHeader == 2 || chHeader == 3)
|
if (chHeader == 2 || chHeader == 3)
|
||||||
return 33;
|
return 33;
|
||||||
if (chHeader == 4 || chHeader == 6 || chHeader == 7)
|
if (chHeader == 4 || chHeader == 6 || chHeader == 7)
|
||||||
|
@ -55,66 +57,79 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set this key data to be invalid
|
// Set this key data to be invalid
|
||||||
void Invalidate() {
|
void Invalidate()
|
||||||
|
{
|
||||||
vch[0] = 0xFF;
|
vch[0] = 0xFF;
|
||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Construct an invalid public key.
|
// Construct an invalid public key.
|
||||||
CPubKey() {
|
CPubKey()
|
||||||
|
{
|
||||||
Invalidate();
|
Invalidate();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize a public key using begin/end iterators to byte data.
|
// Initialize a public key using begin/end iterators to byte data.
|
||||||
template<typename T>
|
template <typename T>
|
||||||
void Set(const T pbegin, const T pend) {
|
void Set(const T pbegin, const T pend)
|
||||||
|
{
|
||||||
int len = pend == pbegin ? 0 : GetLen(pbegin[0]);
|
int len = pend == pbegin ? 0 : GetLen(pbegin[0]);
|
||||||
if (len && len == (pend-pbegin))
|
if (len && len == (pend - pbegin))
|
||||||
memcpy(vch, (unsigned char*)&pbegin[0], len);
|
memcpy(vch, (unsigned char*)&pbegin[0], len);
|
||||||
else
|
else
|
||||||
Invalidate();
|
Invalidate();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Construct a public key using begin/end iterators to byte data.
|
// Construct a public key using begin/end iterators to byte data.
|
||||||
template<typename T>
|
template <typename T>
|
||||||
CPubKey(const T pbegin, const T pend) {
|
CPubKey(const T pbegin, const T pend)
|
||||||
|
{
|
||||||
Set(pbegin, pend);
|
Set(pbegin, pend);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Construct a public key from a byte vector.
|
// Construct a public key from a byte vector.
|
||||||
CPubKey(const std::vector<unsigned char> &vch) {
|
CPubKey(const std::vector<unsigned char>& vch)
|
||||||
|
{
|
||||||
Set(vch.begin(), vch.end());
|
Set(vch.begin(), vch.end());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Simple read-only vector-like interface to the pubkey data.
|
// Simple read-only vector-like interface to the pubkey data.
|
||||||
unsigned int size() const { return GetLen(vch[0]); }
|
unsigned int size() const { return GetLen(vch[0]); }
|
||||||
const unsigned char *begin() const { return vch; }
|
const unsigned char* begin() const { return vch; }
|
||||||
const unsigned char *end() const { return vch+size(); }
|
const unsigned char* end() const { return vch + size(); }
|
||||||
const unsigned char &operator[](unsigned int pos) const { return vch[pos]; }
|
const unsigned char& operator[](unsigned int pos) const { return vch[pos]; }
|
||||||
|
|
||||||
// Comparator implementation.
|
// Comparator implementation.
|
||||||
friend bool operator==(const CPubKey &a, const CPubKey &b) {
|
friend bool operator==(const CPubKey& a, const CPubKey& b)
|
||||||
|
{
|
||||||
return a.vch[0] == b.vch[0] &&
|
return a.vch[0] == b.vch[0] &&
|
||||||
memcmp(a.vch, b.vch, a.size()) == 0;
|
memcmp(a.vch, b.vch, a.size()) == 0;
|
||||||
}
|
}
|
||||||
friend bool operator!=(const CPubKey &a, const CPubKey &b) {
|
friend bool operator!=(const CPubKey& a, const CPubKey& b)
|
||||||
|
{
|
||||||
return !(a == b);
|
return !(a == b);
|
||||||
}
|
}
|
||||||
friend bool operator<(const CPubKey &a, const CPubKey &b) {
|
friend bool operator<(const CPubKey& a, const CPubKey& b)
|
||||||
|
{
|
||||||
return a.vch[0] < b.vch[0] ||
|
return a.vch[0] < b.vch[0] ||
|
||||||
(a.vch[0] == b.vch[0] && memcmp(a.vch, b.vch, a.size()) < 0);
|
(a.vch[0] == b.vch[0] && memcmp(a.vch, b.vch, a.size()) < 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Implement serialization, as if this was a byte vector.
|
// Implement serialization, as if this was a byte vector.
|
||||||
unsigned int GetSerializeSize(int nType, int nVersion) const {
|
unsigned int GetSerializeSize(int nType, int nVersion) const
|
||||||
|
{
|
||||||
return size() + 1;
|
return size() + 1;
|
||||||
}
|
}
|
||||||
template<typename Stream> void Serialize(Stream &s, int nType, int nVersion) const {
|
template <typename Stream>
|
||||||
|
void Serialize(Stream& s, int nType, int nVersion) const
|
||||||
|
{
|
||||||
unsigned int len = size();
|
unsigned int len = size();
|
||||||
::WriteCompactSize(s, len);
|
::WriteCompactSize(s, len);
|
||||||
s.write((char*)vch, len);
|
s.write((char*)vch, len);
|
||||||
}
|
}
|
||||||
template<typename Stream> void Unserialize(Stream &s, int nType, int nVersion) {
|
template <typename Stream>
|
||||||
|
void Unserialize(Stream& s, int nType, int nVersion)
|
||||||
|
{
|
||||||
unsigned int len = ::ReadCompactSize(s);
|
unsigned int len = ::ReadCompactSize(s);
|
||||||
if (len <= 65) {
|
if (len <= 65) {
|
||||||
s.read((char*)vch, len);
|
s.read((char*)vch, len);
|
||||||
|
@ -128,19 +143,22 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the KeyID of this public key (hash of its serialization)
|
// Get the KeyID of this public key (hash of its serialization)
|
||||||
CKeyID GetID() const {
|
CKeyID GetID() const
|
||||||
return CKeyID(Hash160(vch, vch+size()));
|
{
|
||||||
|
return CKeyID(Hash160(vch, vch + size()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the 256-bit hash of this public key.
|
// Get the 256-bit hash of this public key.
|
||||||
uint256 GetHash() const {
|
uint256 GetHash() const
|
||||||
return Hash(vch, vch+size());
|
{
|
||||||
|
return Hash(vch, vch + size());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check syntactic correctness.
|
// Check syntactic correctness.
|
||||||
//
|
//
|
||||||
// Note that this is consensus critical as CheckSig() calls it!
|
// Note that this is consensus critical as CheckSig() calls it!
|
||||||
bool IsValid() const {
|
bool IsValid() const
|
||||||
|
{
|
||||||
return size() > 0;
|
return size() > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -148,16 +166,17 @@ public:
|
||||||
bool IsFullyValid() const;
|
bool IsFullyValid() const;
|
||||||
|
|
||||||
// Check whether this is a compressed public key.
|
// Check whether this is a compressed public key.
|
||||||
bool IsCompressed() const {
|
bool IsCompressed() const
|
||||||
|
{
|
||||||
return size() == 33;
|
return size() == 33;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verify a DER signature (~72 bytes).
|
// Verify a DER signature (~72 bytes).
|
||||||
// If this public key is not fully valid, the return value will be false.
|
// If this public key is not fully valid, the return value will be false.
|
||||||
bool Verify(const uint256 &hash, const std::vector<unsigned char>& vchSig) const;
|
bool Verify(const uint256& hash, const std::vector<unsigned char>& vchSig) const;
|
||||||
|
|
||||||
// Recover a public key from a compact signature.
|
// Recover a public key from a compact signature.
|
||||||
bool RecoverCompact(const uint256 &hash, const std::vector<unsigned char>& vchSig);
|
bool RecoverCompact(const uint256& hash, const std::vector<unsigned char>& vchSig);
|
||||||
|
|
||||||
// Turn this public key into an uncompressed public key.
|
// Turn this public key into an uncompressed public key.
|
||||||
bool Decompress();
|
bool Decompress();
|
||||||
|
@ -172,7 +191,8 @@ public:
|
||||||
typedef std::vector<unsigned char, secure_allocator<unsigned char> > CPrivKey;
|
typedef std::vector<unsigned char, secure_allocator<unsigned char> > CPrivKey;
|
||||||
|
|
||||||
/** An encapsulated private key. */
|
/** An encapsulated private key. */
|
||||||
class CKey {
|
class CKey
|
||||||
|
{
|
||||||
private:
|
private:
|
||||||
// Whether this private key is valid. We check for correctness when modifying the key
|
// Whether this private key is valid. We check for correctness when modifying the key
|
||||||
// data, so fValid should always correspond to the actual state.
|
// data, so fValid should always correspond to the actual state.
|
||||||
|
@ -185,33 +205,38 @@ private:
|
||||||
unsigned char vch[32];
|
unsigned char vch[32];
|
||||||
|
|
||||||
// Check whether the 32-byte array pointed to be vch is valid keydata.
|
// Check whether the 32-byte array pointed to be vch is valid keydata.
|
||||||
bool static Check(const unsigned char *vch);
|
bool static Check(const unsigned char* vch);
|
||||||
public:
|
|
||||||
|
|
||||||
|
public:
|
||||||
// Construct an invalid private key.
|
// Construct an invalid private key.
|
||||||
CKey() : fValid(false), fCompressed(false) {
|
CKey() : fValid(false), fCompressed(false)
|
||||||
|
{
|
||||||
LockObject(vch);
|
LockObject(vch);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy constructor. This is necessary because of memlocking.
|
// Copy constructor. This is necessary because of memlocking.
|
||||||
CKey(const CKey &secret) : fValid(secret.fValid), fCompressed(secret.fCompressed) {
|
CKey(const CKey& secret) : fValid(secret.fValid), fCompressed(secret.fCompressed)
|
||||||
|
{
|
||||||
LockObject(vch);
|
LockObject(vch);
|
||||||
memcpy(vch, secret.vch, sizeof(vch));
|
memcpy(vch, secret.vch, sizeof(vch));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Destructor (again necessary because of memlocking).
|
// Destructor (again necessary because of memlocking).
|
||||||
~CKey() {
|
~CKey()
|
||||||
|
{
|
||||||
UnlockObject(vch);
|
UnlockObject(vch);
|
||||||
}
|
}
|
||||||
|
|
||||||
friend bool operator==(const CKey &a, const CKey &b) {
|
friend bool operator==(const CKey& a, const CKey& b)
|
||||||
|
{
|
||||||
return a.fCompressed == b.fCompressed && a.size() == b.size() &&
|
return a.fCompressed == b.fCompressed && a.size() == b.size() &&
|
||||||
memcmp(&a.vch[0], &b.vch[0], a.size()) == 0;
|
memcmp(&a.vch[0], &b.vch[0], a.size()) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize using begin and end iterators to byte data.
|
// Initialize using begin and end iterators to byte data.
|
||||||
template<typename T>
|
template <typename T>
|
||||||
void Set(const T pbegin, const T pend, bool fCompressedIn) {
|
void Set(const T pbegin, const T pend, bool fCompressedIn)
|
||||||
|
{
|
||||||
if (pend - pbegin != 32) {
|
if (pend - pbegin != 32) {
|
||||||
fValid = false;
|
fValid = false;
|
||||||
return;
|
return;
|
||||||
|
@ -227,8 +252,8 @@ public:
|
||||||
|
|
||||||
// Simple read-only vector-like interface.
|
// Simple read-only vector-like interface.
|
||||||
unsigned int size() const { return (fValid ? 32 : 0); }
|
unsigned int size() const { return (fValid ? 32 : 0); }
|
||||||
const unsigned char *begin() const { return vch; }
|
const unsigned char* begin() const { return vch; }
|
||||||
const unsigned char *end() const { return vch + size(); }
|
const unsigned char* end() const { return vch + size(); }
|
||||||
|
|
||||||
// Check whether this private key is valid.
|
// Check whether this private key is valid.
|
||||||
bool IsValid() const { return fValid; }
|
bool IsValid() const { return fValid; }
|
||||||
|
@ -237,7 +262,7 @@ public:
|
||||||
bool IsCompressed() const { return fCompressed; }
|
bool IsCompressed() const { return fCompressed; }
|
||||||
|
|
||||||
// Initialize from a CPrivKey (serialized OpenSSL private key data).
|
// Initialize from a CPrivKey (serialized OpenSSL private key data).
|
||||||
bool SetPrivKey(const CPrivKey &vchPrivKey, bool fCompressed);
|
bool SetPrivKey(const CPrivKey& vchPrivKey, bool fCompressed);
|
||||||
|
|
||||||
// Generate a new private key using a cryptographic PRNG.
|
// Generate a new private key using a cryptographic PRNG.
|
||||||
void MakeNewKey(bool fCompressed);
|
void MakeNewKey(bool fCompressed);
|
||||||
|
@ -251,23 +276,23 @@ public:
|
||||||
CPubKey GetPubKey() const;
|
CPubKey GetPubKey() const;
|
||||||
|
|
||||||
// Create a DER-serialized signature.
|
// Create a DER-serialized signature.
|
||||||
bool Sign(const uint256 &hash, std::vector<unsigned char>& vchSig) const;
|
bool Sign(const uint256& hash, std::vector<unsigned char>& vchSig) const;
|
||||||
|
|
||||||
// Create a compact signature (65 bytes), which allows reconstructing the used public key.
|
// Create a compact signature (65 bytes), which allows reconstructing the used public key.
|
||||||
// The format is one header byte, followed by two times 32 bytes for the serialized r and s values.
|
// The format is one header byte, followed by two times 32 bytes for the serialized r and s values.
|
||||||
// The header byte: 0x1B = first key with even y, 0x1C = first key with odd y,
|
// The header byte: 0x1B = first key with even y, 0x1C = first key with odd y,
|
||||||
// 0x1D = second key with even y, 0x1E = second key with odd y,
|
// 0x1D = second key with even y, 0x1E = second key with odd y,
|
||||||
// add 0x04 for compressed keys.
|
// add 0x04 for compressed keys.
|
||||||
bool SignCompact(const uint256 &hash, std::vector<unsigned char>& vchSig) const;
|
bool SignCompact(const uint256& hash, std::vector<unsigned char>& vchSig) const;
|
||||||
|
|
||||||
// Derive BIP32 child key.
|
// Derive BIP32 child key.
|
||||||
bool Derive(CKey& keyChild, unsigned char ccChild[32], unsigned int nChild, const unsigned char cc[32]) const;
|
bool Derive(CKey& keyChild, unsigned char ccChild[32], unsigned int nChild, const unsigned char cc[32]) const;
|
||||||
|
|
||||||
// Load private key and check that public key matches.
|
// Load private key and check that public key matches.
|
||||||
bool Load(CPrivKey &privkey, CPubKey &vchPubKey, bool fSkipCheck);
|
bool Load(CPrivKey& privkey, CPubKey& vchPubKey, bool fSkipCheck);
|
||||||
|
|
||||||
// Check whether an element of a signature (r or s) is valid.
|
// Check whether an element of a signature (r or s) is valid.
|
||||||
static bool CheckSignatureElement(const unsigned char *vch, int len, bool half);
|
static bool CheckSignatureElement(const unsigned char* vch, int len, bool half);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct CExtPubKey {
|
struct CExtPubKey {
|
||||||
|
@ -277,14 +302,15 @@ struct CExtPubKey {
|
||||||
unsigned char vchChainCode[32];
|
unsigned char vchChainCode[32];
|
||||||
CPubKey pubkey;
|
CPubKey pubkey;
|
||||||
|
|
||||||
friend bool operator==(const CExtPubKey &a, const CExtPubKey &b) {
|
friend bool operator==(const CExtPubKey& a, const CExtPubKey& b)
|
||||||
|
{
|
||||||
return a.nDepth == b.nDepth && memcmp(&a.vchFingerprint[0], &b.vchFingerprint[0], 4) == 0 && a.nChild == b.nChild &&
|
return a.nDepth == b.nDepth && memcmp(&a.vchFingerprint[0], &b.vchFingerprint[0], 4) == 0 && a.nChild == b.nChild &&
|
||||||
memcmp(&a.vchChainCode[0], &b.vchChainCode[0], 32) == 0 && a.pubkey == b.pubkey;
|
memcmp(&a.vchChainCode[0], &b.vchChainCode[0], 32) == 0 && a.pubkey == b.pubkey;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Encode(unsigned char code[74]) const;
|
void Encode(unsigned char code[74]) const;
|
||||||
void Decode(const unsigned char code[74]);
|
void Decode(const unsigned char code[74]);
|
||||||
bool Derive(CExtPubKey &out, unsigned int nChild) const;
|
bool Derive(CExtPubKey& out, unsigned int nChild) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct CExtKey {
|
struct CExtKey {
|
||||||
|
@ -294,16 +320,17 @@ struct CExtKey {
|
||||||
unsigned char vchChainCode[32];
|
unsigned char vchChainCode[32];
|
||||||
CKey key;
|
CKey key;
|
||||||
|
|
||||||
friend bool operator==(const CExtKey &a, const CExtKey &b) {
|
friend bool operator==(const CExtKey& a, const CExtKey& b)
|
||||||
|
{
|
||||||
return a.nDepth == b.nDepth && memcmp(&a.vchFingerprint[0], &b.vchFingerprint[0], 4) == 0 && a.nChild == b.nChild &&
|
return a.nDepth == b.nDepth && memcmp(&a.vchFingerprint[0], &b.vchFingerprint[0], 4) == 0 && a.nChild == b.nChild &&
|
||||||
memcmp(&a.vchChainCode[0], &b.vchChainCode[0], 32) == 0 && a.key == b.key;
|
memcmp(&a.vchChainCode[0], &b.vchChainCode[0], 32) == 0 && a.key == b.key;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Encode(unsigned char code[74]) const;
|
void Encode(unsigned char code[74]) const;
|
||||||
void Decode(const unsigned char code[74]);
|
void Decode(const unsigned char code[74]);
|
||||||
bool Derive(CExtKey &out, unsigned int nChild) const;
|
bool Derive(CExtKey& out, unsigned int nChild) const;
|
||||||
CExtPubKey Neuter() const;
|
CExtPubKey Neuter() const;
|
||||||
void SetMaster(const unsigned char *seed, unsigned int nSeedLen);
|
void SetMaster(const unsigned char* seed, unsigned int nSeedLen);
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Check that required EC support is available at runtime */
|
/** Check that required EC support is available at runtime */
|
||||||
|
|
|
@ -12,7 +12,8 @@
|
||||||
#include <leveldb/filter_policy.h>
|
#include <leveldb/filter_policy.h>
|
||||||
#include <memenv.h>
|
#include <memenv.h>
|
||||||
|
|
||||||
void HandleError(const leveldb::Status &status) throw(leveldb_error) {
|
void HandleError(const leveldb::Status& status) throw(leveldb_error)
|
||||||
|
{
|
||||||
if (status.ok())
|
if (status.ok())
|
||||||
return;
|
return;
|
||||||
LogPrintf("%s\n", status.ToString());
|
LogPrintf("%s\n", status.ToString());
|
||||||
|
@ -25,7 +26,8 @@ void HandleError(const leveldb::Status &status) throw(leveldb_error) {
|
||||||
throw leveldb_error("Unknown database error");
|
throw leveldb_error("Unknown database error");
|
||||||
}
|
}
|
||||||
|
|
||||||
static leveldb::Options GetOptions(size_t nCacheSize) {
|
static leveldb::Options GetOptions(size_t nCacheSize)
|
||||||
|
{
|
||||||
leveldb::Options options;
|
leveldb::Options options;
|
||||||
options.block_cache = leveldb::NewLRUCache(nCacheSize / 2);
|
options.block_cache = leveldb::NewLRUCache(nCacheSize / 2);
|
||||||
options.write_buffer_size = nCacheSize / 4; // up to two write buffers may be held in memory simultaneously
|
options.write_buffer_size = nCacheSize / 4; // up to two write buffers may be held in memory simultaneously
|
||||||
|
@ -40,7 +42,8 @@ static leveldb::Options GetOptions(size_t nCacheSize) {
|
||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
|
|
||||||
CLevelDBWrapper::CLevelDBWrapper(const boost::filesystem::path &path, size_t nCacheSize, bool fMemory, bool fWipe) {
|
CLevelDBWrapper::CLevelDBWrapper(const boost::filesystem::path& path, size_t nCacheSize, bool fMemory, bool fWipe)
|
||||||
|
{
|
||||||
penv = NULL;
|
penv = NULL;
|
||||||
readoptions.verify_checksums = true;
|
readoptions.verify_checksums = true;
|
||||||
iteroptions.verify_checksums = true;
|
iteroptions.verify_checksums = true;
|
||||||
|
@ -64,7 +67,8 @@ CLevelDBWrapper::CLevelDBWrapper(const boost::filesystem::path &path, size_t nCa
|
||||||
LogPrintf("Opened LevelDB successfully\n");
|
LogPrintf("Opened LevelDB successfully\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
CLevelDBWrapper::~CLevelDBWrapper() {
|
CLevelDBWrapper::~CLevelDBWrapper()
|
||||||
|
{
|
||||||
delete pdb;
|
delete pdb;
|
||||||
pdb = NULL;
|
pdb = NULL;
|
||||||
delete options.filter_policy;
|
delete options.filter_policy;
|
||||||
|
@ -75,7 +79,8 @@ CLevelDBWrapper::~CLevelDBWrapper() {
|
||||||
options.env = NULL;
|
options.env = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CLevelDBWrapper::WriteBatch(CLevelDBBatch &batch, bool fSync) throw(leveldb_error) {
|
bool CLevelDBWrapper::WriteBatch(CLevelDBBatch& batch, bool fSync) throw(leveldb_error)
|
||||||
|
{
|
||||||
leveldb::Status status = pdb->Write(fSync ? syncoptions : writeoptions, &batch.batch);
|
leveldb::Status status = pdb->Write(fSync ? syncoptions : writeoptions, &batch.batch);
|
||||||
HandleError(status);
|
HandleError(status);
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -17,10 +17,10 @@
|
||||||
class leveldb_error : public std::runtime_error
|
class leveldb_error : public std::runtime_error
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
leveldb_error(const std::string &msg) : std::runtime_error(msg) {}
|
leveldb_error(const std::string& msg) : std::runtime_error(msg) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
void HandleError(const leveldb::Status &status) throw(leveldb_error);
|
void HandleError(const leveldb::Status& status) throw(leveldb_error);
|
||||||
|
|
||||||
// Batch of changes queued to be written to a CLevelDBWrapper
|
// Batch of changes queued to be written to a CLevelDBWrapper
|
||||||
class CLevelDBBatch
|
class CLevelDBBatch
|
||||||
|
@ -31,7 +31,9 @@ private:
|
||||||
leveldb::WriteBatch batch;
|
leveldb::WriteBatch batch;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
template<typename K, typename V> void Write(const K& key, const V& value) {
|
template <typename K, typename V>
|
||||||
|
void Write(const K& key, const V& value)
|
||||||
|
{
|
||||||
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
||||||
ssKey.reserve(ssKey.GetSerializeSize(key));
|
ssKey.reserve(ssKey.GetSerializeSize(key));
|
||||||
ssKey << key;
|
ssKey << key;
|
||||||
|
@ -45,7 +47,9 @@ public:
|
||||||
batch.Put(slKey, slValue);
|
batch.Put(slKey, slValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K> void Erase(const K& key) {
|
template <typename K>
|
||||||
|
void Erase(const K& key)
|
||||||
|
{
|
||||||
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
||||||
ssKey.reserve(ssKey.GetSerializeSize(key));
|
ssKey.reserve(ssKey.GetSerializeSize(key));
|
||||||
ssKey << key;
|
ssKey << key;
|
||||||
|
@ -59,7 +63,7 @@ class CLevelDBWrapper
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
// custom environment this database is using (may be NULL in case of default environment)
|
// custom environment this database is using (may be NULL in case of default environment)
|
||||||
leveldb::Env *penv;
|
leveldb::Env* penv;
|
||||||
|
|
||||||
// database options used
|
// database options used
|
||||||
leveldb::Options options;
|
leveldb::Options options;
|
||||||
|
@ -77,13 +81,15 @@ private:
|
||||||
leveldb::WriteOptions syncoptions;
|
leveldb::WriteOptions syncoptions;
|
||||||
|
|
||||||
// the database itself
|
// the database itself
|
||||||
leveldb::DB *pdb;
|
leveldb::DB* pdb;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CLevelDBWrapper(const boost::filesystem::path &path, size_t nCacheSize, bool fMemory = false, bool fWipe = false);
|
CLevelDBWrapper(const boost::filesystem::path& path, size_t nCacheSize, bool fMemory = false, bool fWipe = false);
|
||||||
~CLevelDBWrapper();
|
~CLevelDBWrapper();
|
||||||
|
|
||||||
template<typename K, typename V> bool Read(const K& key, V& value) const throw(leveldb_error) {
|
template <typename K, typename V>
|
||||||
|
bool Read(const K& key, V& value) const throw(leveldb_error)
|
||||||
|
{
|
||||||
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
||||||
ssKey.reserve(ssKey.GetSerializeSize(key));
|
ssKey.reserve(ssKey.GetSerializeSize(key));
|
||||||
ssKey << key;
|
ssKey << key;
|
||||||
|
@ -100,19 +106,23 @@ public:
|
||||||
try {
|
try {
|
||||||
CDataStream ssValue(strValue.data(), strValue.data() + strValue.size(), SER_DISK, CLIENT_VERSION);
|
CDataStream ssValue(strValue.data(), strValue.data() + strValue.size(), SER_DISK, CLIENT_VERSION);
|
||||||
ssValue >> value;
|
ssValue >> value;
|
||||||
} catch(const std::exception &) {
|
} catch (const std::exception&) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K, typename V> bool Write(const K& key, const V& value, bool fSync = false) throw(leveldb_error) {
|
template <typename K, typename V>
|
||||||
|
bool Write(const K& key, const V& value, bool fSync = false) throw(leveldb_error)
|
||||||
|
{
|
||||||
CLevelDBBatch batch;
|
CLevelDBBatch batch;
|
||||||
batch.Write(key, value);
|
batch.Write(key, value);
|
||||||
return WriteBatch(batch, fSync);
|
return WriteBatch(batch, fSync);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K> bool Exists(const K& key) const throw(leveldb_error) {
|
template <typename K>
|
||||||
|
bool Exists(const K& key) const throw(leveldb_error)
|
||||||
|
{
|
||||||
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
|
||||||
ssKey.reserve(ssKey.GetSerializeSize(key));
|
ssKey.reserve(ssKey.GetSerializeSize(key));
|
||||||
ssKey << key;
|
ssKey << key;
|
||||||
|
@ -129,26 +139,31 @@ public:
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename K> bool Erase(const K& key, bool fSync = false) throw(leveldb_error) {
|
template <typename K>
|
||||||
|
bool Erase(const K& key, bool fSync = false) throw(leveldb_error)
|
||||||
|
{
|
||||||
CLevelDBBatch batch;
|
CLevelDBBatch batch;
|
||||||
batch.Erase(key);
|
batch.Erase(key);
|
||||||
return WriteBatch(batch, fSync);
|
return WriteBatch(batch, fSync);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool WriteBatch(CLevelDBBatch &batch, bool fSync = false) throw(leveldb_error);
|
bool WriteBatch(CLevelDBBatch& batch, bool fSync = false) throw(leveldb_error);
|
||||||
|
|
||||||
// not available for LevelDB; provide for compatibility with BDB
|
// not available for LevelDB; provide for compatibility with BDB
|
||||||
bool Flush() {
|
bool Flush()
|
||||||
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Sync() throw(leveldb_error) {
|
bool Sync() throw(leveldb_error)
|
||||||
|
{
|
||||||
CLevelDBBatch batch;
|
CLevelDBBatch batch;
|
||||||
return WriteBatch(batch, true);
|
return WriteBatch(batch, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// not exactly clean encapsulation, but it's easiest for now
|
// not exactly clean encapsulation, but it's easiest for now
|
||||||
leveldb::Iterator *NewIterator() {
|
leveldb::Iterator* NewIterator()
|
||||||
|
{
|
||||||
return pdb->NewIterator(iteroptions);
|
return pdb->NewIterator(iteroptions);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -9,7 +9,8 @@
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
/** STL-like map container that only keeps the N elements with the highest value. */
|
/** STL-like map container that only keeps the N elements with the highest value. */
|
||||||
template <typename K, typename V> class limitedmap
|
template <typename K, typename V>
|
||||||
|
class limitedmap
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
typedef K key_type;
|
typedef K key_type;
|
||||||
|
@ -36,10 +37,8 @@ public:
|
||||||
void insert(const value_type& x)
|
void insert(const value_type& x)
|
||||||
{
|
{
|
||||||
std::pair<iterator, bool> ret = map.insert(x);
|
std::pair<iterator, bool> ret = map.insert(x);
|
||||||
if (ret.second)
|
if (ret.second) {
|
||||||
{
|
if (nMaxSize && map.size() == nMaxSize) {
|
||||||
if (nMaxSize && map.size() == nMaxSize)
|
|
||||||
{
|
|
||||||
map.erase(rmap.begin()->second);
|
map.erase(rmap.begin()->second);
|
||||||
rmap.erase(rmap.begin());
|
rmap.erase(rmap.begin());
|
||||||
}
|
}
|
||||||
|
@ -54,8 +53,7 @@ public:
|
||||||
return;
|
return;
|
||||||
std::pair<rmap_iterator, rmap_iterator> itPair = rmap.equal_range(itTarget->second);
|
std::pair<rmap_iterator, rmap_iterator> itPair = rmap.equal_range(itTarget->second);
|
||||||
for (rmap_iterator it = itPair.first; it != itPair.second; ++it)
|
for (rmap_iterator it = itPair.first; it != itPair.second; ++it)
|
||||||
if (it->second == itTarget)
|
if (it->second == itTarget) {
|
||||||
{
|
|
||||||
rmap.erase(it);
|
rmap.erase(it);
|
||||||
map.erase(itTarget);
|
map.erase(itTarget);
|
||||||
return;
|
return;
|
||||||
|
@ -72,8 +70,7 @@ public:
|
||||||
return;
|
return;
|
||||||
std::pair<rmap_iterator, rmap_iterator> itPair = rmap.equal_range(itTarget->second);
|
std::pair<rmap_iterator, rmap_iterator> itPair = rmap.equal_range(itTarget->second);
|
||||||
for (rmap_iterator it = itPair.first; it != itPair.second; ++it)
|
for (rmap_iterator it = itPair.first; it != itPair.second; ++it)
|
||||||
if (it->second == itTarget)
|
if (it->second == itTarget) {
|
||||||
{
|
|
||||||
rmap.erase(it);
|
rmap.erase(it);
|
||||||
itTarget->second = v;
|
itTarget->second = v;
|
||||||
rmap.insert(make_pair(v, itTarget));
|
rmap.insert(make_pair(v, itTarget));
|
||||||
|
@ -88,8 +85,7 @@ public:
|
||||||
size_type max_size(size_type s)
|
size_type max_size(size_type s)
|
||||||
{
|
{
|
||||||
if (s)
|
if (s)
|
||||||
while (map.size() > s)
|
while (map.size() > s) {
|
||||||
{
|
|
||||||
map.erase(rmap.begin()->second);
|
map.erase(rmap.begin()->second);
|
||||||
rmap.erase(rmap.begin());
|
rmap.erase(rmap.begin());
|
||||||
}
|
}
|
||||||
|
|
18
src/mruset.h
18
src/mruset.h
|
@ -10,7 +10,8 @@
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
/** STL-like set container that only keeps the most recent N elements. */
|
/** STL-like set container that only keeps the most recent N elements. */
|
||||||
template <typename T> class mruset
|
template <typename T>
|
||||||
|
class mruset
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
typedef T key_type;
|
typedef T key_type;
|
||||||
|
@ -32,17 +33,19 @@ public:
|
||||||
bool empty() const { return set.empty(); }
|
bool empty() const { return set.empty(); }
|
||||||
iterator find(const key_type& k) const { return set.find(k); }
|
iterator find(const key_type& k) const { return set.find(k); }
|
||||||
size_type count(const key_type& k) const { return set.count(k); }
|
size_type count(const key_type& k) const { return set.count(k); }
|
||||||
void clear() { set.clear(); queue.clear(); }
|
void clear()
|
||||||
|
{
|
||||||
|
set.clear();
|
||||||
|
queue.clear();
|
||||||
|
}
|
||||||
bool inline friend operator==(const mruset<T>& a, const mruset<T>& b) { return a.set == b.set; }
|
bool inline friend operator==(const mruset<T>& a, const mruset<T>& b) { return a.set == b.set; }
|
||||||
bool inline friend operator==(const mruset<T>& a, const std::set<T>& b) { return a.set == b; }
|
bool inline friend operator==(const mruset<T>& a, const std::set<T>& b) { return a.set == b; }
|
||||||
bool inline friend operator<(const mruset<T>& a, const mruset<T>& b) { return a.set < b.set; }
|
bool inline friend operator<(const mruset<T>& a, const mruset<T>& b) { return a.set < b.set; }
|
||||||
std::pair<iterator, bool> insert(const key_type& x)
|
std::pair<iterator, bool> insert(const key_type& x)
|
||||||
{
|
{
|
||||||
std::pair<iterator, bool> ret = set.insert(x);
|
std::pair<iterator, bool> ret = set.insert(x);
|
||||||
if (ret.second)
|
if (ret.second) {
|
||||||
{
|
if (nMaxSize && queue.size() == nMaxSize) {
|
||||||
if (nMaxSize && queue.size() == nMaxSize)
|
|
||||||
{
|
|
||||||
set.erase(queue.front());
|
set.erase(queue.front());
|
||||||
queue.pop_front();
|
queue.pop_front();
|
||||||
}
|
}
|
||||||
|
@ -54,8 +57,7 @@ public:
|
||||||
size_type max_size(size_type s)
|
size_type max_size(size_type s)
|
||||||
{
|
{
|
||||||
if (s)
|
if (s)
|
||||||
while (queue.size() > s)
|
while (queue.size() > s) {
|
||||||
{
|
|
||||||
set.erase(queue.front());
|
set.erase(queue.front());
|
||||||
queue.pop_front();
|
queue.pop_front();
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,7 +35,7 @@ static bool noui_ThreadSafeMessageBox(const std::string& message, const std::str
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void noui_InitMessage(const std::string &message)
|
static void noui_InitMessage(const std::string& message)
|
||||||
{
|
{
|
||||||
LogPrintf("init message: %s\n", message);
|
LogPrintf("init message: %s\n", message);
|
||||||
}
|
}
|
||||||
|
|
143
src/protocol.h
143
src/protocol.h
|
@ -4,7 +4,7 @@
|
||||||
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||||
|
|
||||||
#ifndef __cplusplus
|
#ifndef __cplusplus
|
||||||
# error This header can only be compiled as C++.
|
#error This header can only be compiled as C++.
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef __INCLUDED_PROTOCOL_H__
|
#ifndef __INCLUDED_PROTOCOL_H__
|
||||||
|
@ -28,43 +28,43 @@
|
||||||
*/
|
*/
|
||||||
class CMessageHeader
|
class CMessageHeader
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CMessageHeader();
|
CMessageHeader();
|
||||||
CMessageHeader(const char* pszCommand, unsigned int nMessageSizeIn);
|
CMessageHeader(const char* pszCommand, unsigned int nMessageSizeIn);
|
||||||
|
|
||||||
std::string GetCommand() const;
|
std::string GetCommand() const;
|
||||||
bool IsValid() const;
|
bool IsValid() const;
|
||||||
|
|
||||||
ADD_SERIALIZE_METHODS;
|
ADD_SERIALIZE_METHODS;
|
||||||
|
|
||||||
template <typename Stream, typename Operation>
|
template <typename Stream, typename Operation>
|
||||||
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion)
|
||||||
READWRITE(FLATDATA(pchMessageStart));
|
{
|
||||||
READWRITE(FLATDATA(pchCommand));
|
READWRITE(FLATDATA(pchMessageStart));
|
||||||
READWRITE(nMessageSize);
|
READWRITE(FLATDATA(pchCommand));
|
||||||
READWRITE(nChecksum);
|
READWRITE(nMessageSize);
|
||||||
}
|
READWRITE(nChecksum);
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: make private (improves encapsulation)
|
// TODO: make private (improves encapsulation)
|
||||||
public:
|
public:
|
||||||
enum {
|
enum {
|
||||||
COMMAND_SIZE=12,
|
COMMAND_SIZE = 12,
|
||||||
MESSAGE_SIZE_SIZE=sizeof(int),
|
MESSAGE_SIZE_SIZE = sizeof(int),
|
||||||
CHECKSUM_SIZE=sizeof(int),
|
CHECKSUM_SIZE = sizeof(int),
|
||||||
|
|
||||||
MESSAGE_SIZE_OFFSET=MESSAGE_START_SIZE+COMMAND_SIZE,
|
MESSAGE_SIZE_OFFSET = MESSAGE_START_SIZE + COMMAND_SIZE,
|
||||||
CHECKSUM_OFFSET=MESSAGE_SIZE_OFFSET+MESSAGE_SIZE_SIZE,
|
CHECKSUM_OFFSET = MESSAGE_SIZE_OFFSET + MESSAGE_SIZE_SIZE,
|
||||||
HEADER_SIZE=MESSAGE_START_SIZE+COMMAND_SIZE+MESSAGE_SIZE_SIZE+CHECKSUM_SIZE
|
HEADER_SIZE = MESSAGE_START_SIZE + COMMAND_SIZE + MESSAGE_SIZE_SIZE + CHECKSUM_SIZE
|
||||||
};
|
};
|
||||||
char pchMessageStart[MESSAGE_START_SIZE];
|
char pchMessageStart[MESSAGE_START_SIZE];
|
||||||
char pchCommand[COMMAND_SIZE];
|
char pchCommand[COMMAND_SIZE];
|
||||||
unsigned int nMessageSize;
|
unsigned int nMessageSize;
|
||||||
unsigned int nChecksum;
|
unsigned int nChecksum;
|
||||||
};
|
};
|
||||||
|
|
||||||
/** nServices flags */
|
/** nServices flags */
|
||||||
enum
|
enum {
|
||||||
{
|
|
||||||
NODE_NETWORK = (1 << 0),
|
NODE_NETWORK = (1 << 0),
|
||||||
|
|
||||||
// Bits 24-31 are reserved for temporary experiments. Just pick a bit that
|
// Bits 24-31 are reserved for temporary experiments. Just pick a bit that
|
||||||
|
@ -79,68 +79,69 @@ enum
|
||||||
/** A CService with information about it as peer */
|
/** A CService with information about it as peer */
|
||||||
class CAddress : public CService
|
class CAddress : public CService
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CAddress();
|
CAddress();
|
||||||
explicit CAddress(CService ipIn, uint64_t nServicesIn=NODE_NETWORK);
|
explicit CAddress(CService ipIn, uint64_t nServicesIn = NODE_NETWORK);
|
||||||
|
|
||||||
void Init();
|
void Init();
|
||||||
|
|
||||||
ADD_SERIALIZE_METHODS;
|
ADD_SERIALIZE_METHODS;
|
||||||
|
|
||||||
template <typename Stream, typename Operation>
|
template <typename Stream, typename Operation>
|
||||||
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion)
|
||||||
if (ser_action.ForRead())
|
{
|
||||||
Init();
|
if (ser_action.ForRead())
|
||||||
if (nType & SER_DISK)
|
Init();
|
||||||
READWRITE(nVersion);
|
if (nType & SER_DISK)
|
||||||
if ((nType & SER_DISK) ||
|
READWRITE(nVersion);
|
||||||
(nVersion >= CADDR_TIME_VERSION && !(nType & SER_GETHASH)))
|
if ((nType & SER_DISK) ||
|
||||||
READWRITE(nTime);
|
(nVersion >= CADDR_TIME_VERSION && !(nType & SER_GETHASH)))
|
||||||
READWRITE(nServices);
|
READWRITE(nTime);
|
||||||
READWRITE(*(CService*)this);
|
READWRITE(nServices);
|
||||||
}
|
READWRITE(*(CService*)this);
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: make private (improves encapsulation)
|
// TODO: make private (improves encapsulation)
|
||||||
public:
|
public:
|
||||||
uint64_t nServices;
|
uint64_t nServices;
|
||||||
|
|
||||||
// disk and network only
|
// disk and network only
|
||||||
unsigned int nTime;
|
unsigned int nTime;
|
||||||
|
|
||||||
// memory only
|
// memory only
|
||||||
int64_t nLastTry;
|
int64_t nLastTry;
|
||||||
};
|
};
|
||||||
|
|
||||||
/** inv message data */
|
/** inv message data */
|
||||||
class CInv
|
class CInv
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
CInv();
|
CInv();
|
||||||
CInv(int typeIn, const uint256& hashIn);
|
CInv(int typeIn, const uint256& hashIn);
|
||||||
CInv(const std::string& strType, const uint256& hashIn);
|
CInv(const std::string& strType, const uint256& hashIn);
|
||||||
|
|
||||||
ADD_SERIALIZE_METHODS;
|
ADD_SERIALIZE_METHODS;
|
||||||
|
|
||||||
template <typename Stream, typename Operation>
|
template <typename Stream, typename Operation>
|
||||||
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion)
|
||||||
READWRITE(type);
|
{
|
||||||
READWRITE(hash);
|
READWRITE(type);
|
||||||
}
|
READWRITE(hash);
|
||||||
|
}
|
||||||
|
|
||||||
friend bool operator<(const CInv& a, const CInv& b);
|
friend bool operator<(const CInv& a, const CInv& b);
|
||||||
|
|
||||||
bool IsKnownType() const;
|
bool IsKnownType() const;
|
||||||
const char* GetCommand() const;
|
const char* GetCommand() const;
|
||||||
std::string ToString() const;
|
std::string ToString() const;
|
||||||
|
|
||||||
// TODO: make private (improves encapsulation)
|
// TODO: make private (improves encapsulation)
|
||||||
public:
|
public:
|
||||||
int type;
|
int type;
|
||||||
uint256 hash;
|
uint256 hash;
|
||||||
};
|
};
|
||||||
|
|
||||||
enum
|
enum {
|
||||||
{
|
|
||||||
MSG_TX = 1,
|
MSG_TX = 1,
|
||||||
MSG_BLOCK,
|
MSG_BLOCK,
|
||||||
// Nodes may always request a MSG_FILTERED_BLOCK in a getdata, however,
|
// Nodes may always request a MSG_FILTERED_BLOCK in a getdata, however,
|
||||||
|
|
|
@ -8,8 +8,8 @@
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
#include "compat.h" // for Windows API
|
#include "compat.h" // for Windows API
|
||||||
#endif
|
#endif
|
||||||
#include "serialize.h" // for begin_ptr(vec)
|
#include "serialize.h" // for begin_ptr(vec)
|
||||||
#include "util.h" // for LogPrint()
|
#include "util.h" // for LogPrint()
|
||||||
#include "utilstrencodings.h" // for GetTime()
|
#include "utilstrencodings.h" // for GetTime()
|
||||||
|
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
@ -56,28 +56,25 @@ void RandAddSeedPerfmon()
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
// Don't need this on Linux, OpenSSL automatically uses /dev/urandom
|
// Don't need this on Linux, OpenSSL automatically uses /dev/urandom
|
||||||
// Seed with the entire set of perfmon data
|
// Seed with the entire set of perfmon data
|
||||||
std::vector <unsigned char> vData(250000,0);
|
std::vector<unsigned char> vData(250000, 0);
|
||||||
long ret = 0;
|
long ret = 0;
|
||||||
unsigned long nSize = 0;
|
unsigned long nSize = 0;
|
||||||
const size_t nMaxSize = 10000000; // Bail out at more than 10MB of performance data
|
const size_t nMaxSize = 10000000; // Bail out at more than 10MB of performance data
|
||||||
while (true)
|
while (true) {
|
||||||
{
|
|
||||||
nSize = vData.size();
|
nSize = vData.size();
|
||||||
ret = RegQueryValueExA(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, begin_ptr(vData), &nSize);
|
ret = RegQueryValueExA(HKEY_PERFORMANCE_DATA, "Global", NULL, NULL, begin_ptr(vData), &nSize);
|
||||||
if (ret != ERROR_MORE_DATA || vData.size() >= nMaxSize)
|
if (ret != ERROR_MORE_DATA || vData.size() >= nMaxSize)
|
||||||
break;
|
break;
|
||||||
vData.resize(std::max((vData.size()*3)/2, nMaxSize)); // Grow size of buffer exponentially
|
vData.resize(std::max((vData.size() * 3) / 2, nMaxSize)); // Grow size of buffer exponentially
|
||||||
}
|
}
|
||||||
RegCloseKey(HKEY_PERFORMANCE_DATA);
|
RegCloseKey(HKEY_PERFORMANCE_DATA);
|
||||||
if (ret == ERROR_SUCCESS)
|
if (ret == ERROR_SUCCESS) {
|
||||||
{
|
RAND_add(begin_ptr(vData), nSize, nSize / 100.0);
|
||||||
RAND_add(begin_ptr(vData), nSize, nSize/100.0);
|
|
||||||
OPENSSL_cleanse(begin_ptr(vData), nSize);
|
OPENSSL_cleanse(begin_ptr(vData), nSize);
|
||||||
LogPrint("rand", "%s: %lu bytes\n", __func__, nSize);
|
LogPrint("rand", "%s: %lu bytes\n", __func__, nSize);
|
||||||
} else {
|
} else {
|
||||||
static bool warned = false; // Warn only once
|
static bool warned = false; // Warn only once
|
||||||
if (!warned)
|
if (!warned) {
|
||||||
{
|
|
||||||
LogPrintf("%s: Warning: RegQueryValueExA(HKEY_PERFORMANCE_DATA) failed with code %i\n", __func__, ret);
|
LogPrintf("%s: Warning: RegQueryValueExA(HKEY_PERFORMANCE_DATA) failed with code %i\n", __func__, ret);
|
||||||
warned = true;
|
warned = true;
|
||||||
}
|
}
|
||||||
|
@ -85,7 +82,7 @@ void RandAddSeedPerfmon()
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool GetRandBytes(unsigned char *buf, int num)
|
bool GetRandBytes(unsigned char* buf, int num)
|
||||||
{
|
{
|
||||||
if (RAND_bytes(buf, num) != 1) {
|
if (RAND_bytes(buf, num) != 1) {
|
||||||
LogPrintf("%s: OpenSSL RAND_bytes() failed with error: %s\n", __func__, ERR_error_string(ERR_get_error(), NULL));
|
LogPrintf("%s: OpenSSL RAND_bytes() failed with error: %s\n", __func__, ERR_error_string(ERR_get_error(), NULL));
|
||||||
|
@ -126,18 +123,17 @@ uint32_t insecure_rand_Rw = 11;
|
||||||
void seed_insecure_rand(bool fDeterministic)
|
void seed_insecure_rand(bool fDeterministic)
|
||||||
{
|
{
|
||||||
// The seed values have some unlikely fixed points which we avoid.
|
// The seed values have some unlikely fixed points which we avoid.
|
||||||
if(fDeterministic)
|
if (fDeterministic) {
|
||||||
{
|
|
||||||
insecure_rand_Rz = insecure_rand_Rw = 11;
|
insecure_rand_Rz = insecure_rand_Rw = 11;
|
||||||
} else {
|
} else {
|
||||||
uint32_t tmp;
|
uint32_t tmp;
|
||||||
do {
|
do {
|
||||||
GetRandBytes((unsigned char*)&tmp, 4);
|
GetRandBytes((unsigned char*)&tmp, 4);
|
||||||
} while(tmp == 0 || tmp == 0x9068ffffU);
|
} while (tmp == 0 || tmp == 0x9068ffffU);
|
||||||
insecure_rand_Rz = tmp;
|
insecure_rand_Rz = tmp;
|
||||||
do {
|
do {
|
||||||
GetRandBytes((unsigned char*)&tmp, 4);
|
GetRandBytes((unsigned char*)&tmp, 4);
|
||||||
} while(tmp == 0 || tmp == 0x464fffffU);
|
} while (tmp == 0 || tmp == 0x464fffffU);
|
||||||
insecure_rand_Rw = tmp;
|
insecure_rand_Rw = tmp;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,7 +19,7 @@ void RandAddSeedPerfmon();
|
||||||
/**
|
/**
|
||||||
* Functions to gather random data via the OpenSSL PRNG
|
* Functions to gather random data via the OpenSSL PRNG
|
||||||
*/
|
*/
|
||||||
bool GetRandBytes(unsigned char *buf, int num);
|
bool GetRandBytes(unsigned char* buf, int num);
|
||||||
uint64_t GetRand(uint64_t nMax);
|
uint64_t GetRand(uint64_t nMax);
|
||||||
int GetRandInt(int nMax);
|
int GetRandInt(int nMax);
|
||||||
uint256 GetRandHash();
|
uint256 GetRandHash();
|
||||||
|
|
|
@ -10,6 +10,6 @@
|
||||||
#include "json/json_spirit_utils.h"
|
#include "json/json_spirit_utils.h"
|
||||||
#include "json/json_spirit_writer_template.h"
|
#include "json/json_spirit_writer_template.h"
|
||||||
|
|
||||||
json_spirit::Array RPCConvertValues(const std::string &strMethod, const std::vector<std::string> &strParams);
|
json_spirit::Array RPCConvertValues(const std::string& strMethod, const std::vector<std::string>& strParams);
|
||||||
|
|
||||||
#endif // _BITCOINRPC_CLIENT_H_
|
#endif // _BITCOINRPC_CLIENT_H_
|
||||||
|
|
48
src/sync.cpp
48
src/sync.cpp
|
@ -32,8 +32,7 @@ void PrintLockContention(const char* pszName, const char* pszFile, int nLine)
|
||||||
// Complain if any thread tries to lock in a different order.
|
// Complain if any thread tries to lock in a different order.
|
||||||
//
|
//
|
||||||
|
|
||||||
struct CLockLocation
|
struct CLockLocation {
|
||||||
{
|
|
||||||
CLockLocation(const char* pszName, const char* pszFile, int nLine)
|
CLockLocation(const char* pszName, const char* pszFile, int nLine)
|
||||||
{
|
{
|
||||||
mutexName = pszName;
|
mutexName = pszName;
|
||||||
|
@ -43,7 +42,7 @@ struct CLockLocation
|
||||||
|
|
||||||
std::string ToString() const
|
std::string ToString() const
|
||||||
{
|
{
|
||||||
return mutexName+" "+sourceFile+":"+itostr(sourceLine);
|
return mutexName + " " + sourceFile + ":" + itostr(sourceLine);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string MutexName() const { return mutexName; }
|
std::string MutexName() const { return mutexName; }
|
||||||
|
@ -54,7 +53,7 @@ private:
|
||||||
int sourceLine;
|
int sourceLine;
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef std::vector< std::pair<void*, CLockLocation> > LockStack;
|
typedef std::vector<std::pair<void*, CLockLocation> > LockStack;
|
||||||
|
|
||||||
static boost::mutex dd_mutex;
|
static boost::mutex dd_mutex;
|
||||||
static std::map<std::pair<void*, void*>, LockStack> lockorders;
|
static std::map<std::pair<void*, void*>, LockStack> lockorders;
|
||||||
|
@ -65,17 +64,19 @@ static void potential_deadlock_detected(const std::pair<void*, void*>& mismatch,
|
||||||
{
|
{
|
||||||
LogPrintf("POTENTIAL DEADLOCK DETECTED\n");
|
LogPrintf("POTENTIAL DEADLOCK DETECTED\n");
|
||||||
LogPrintf("Previous lock order was:\n");
|
LogPrintf("Previous lock order was:\n");
|
||||||
BOOST_FOREACH(const PAIRTYPE(void*, CLockLocation)& i, s2)
|
BOOST_FOREACH (const PAIRTYPE(void*, CLockLocation) & i, s2) {
|
||||||
{
|
if (i.first == mismatch.first)
|
||||||
if (i.first == mismatch.first) LogPrintf(" (1)");
|
LogPrintf(" (1)");
|
||||||
if (i.first == mismatch.second) LogPrintf(" (2)");
|
if (i.first == mismatch.second)
|
||||||
|
LogPrintf(" (2)");
|
||||||
LogPrintf(" %s\n", i.second.ToString());
|
LogPrintf(" %s\n", i.second.ToString());
|
||||||
}
|
}
|
||||||
LogPrintf("Current lock order is:\n");
|
LogPrintf("Current lock order is:\n");
|
||||||
BOOST_FOREACH(const PAIRTYPE(void*, CLockLocation)& i, s1)
|
BOOST_FOREACH (const PAIRTYPE(void*, CLockLocation) & i, s1) {
|
||||||
{
|
if (i.first == mismatch.first)
|
||||||
if (i.first == mismatch.first) LogPrintf(" (1)");
|
LogPrintf(" (1)");
|
||||||
if (i.first == mismatch.second) LogPrintf(" (2)");
|
if (i.first == mismatch.second)
|
||||||
|
LogPrintf(" (2)");
|
||||||
LogPrintf(" %s\n", i.second.ToString());
|
LogPrintf(" %s\n", i.second.ToString());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -91,8 +92,9 @@ static void push_lock(void* c, const CLockLocation& locklocation, bool fTry)
|
||||||
(*lockstack).push_back(std::make_pair(c, locklocation));
|
(*lockstack).push_back(std::make_pair(c, locklocation));
|
||||||
|
|
||||||
if (!fTry) {
|
if (!fTry) {
|
||||||
BOOST_FOREACH(const PAIRTYPE(void*, CLockLocation)& i, (*lockstack)) {
|
BOOST_FOREACH (const PAIRTYPE(void*, CLockLocation) & i, (*lockstack)) {
|
||||||
if (i.first == c) break;
|
if (i.first == c)
|
||||||
|
break;
|
||||||
|
|
||||||
std::pair<void*, void*> p1 = std::make_pair(i.first, c);
|
std::pair<void*, void*> p1 = std::make_pair(i.first, c);
|
||||||
if (lockorders.count(p1))
|
if (lockorders.count(p1))
|
||||||
|
@ -100,8 +102,7 @@ static void push_lock(void* c, const CLockLocation& locklocation, bool fTry)
|
||||||
lockorders[p1] = (*lockstack);
|
lockorders[p1] = (*lockstack);
|
||||||
|
|
||||||
std::pair<void*, void*> p2 = std::make_pair(c, i.first);
|
std::pair<void*, void*> p2 = std::make_pair(c, i.first);
|
||||||
if (lockorders.count(p2))
|
if (lockorders.count(p2)) {
|
||||||
{
|
|
||||||
potential_deadlock_detected(p1, lockorders[p2], lockorders[p1]);
|
potential_deadlock_detected(p1, lockorders[p2], lockorders[p1]);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -112,8 +113,7 @@ static void push_lock(void* c, const CLockLocation& locklocation, bool fTry)
|
||||||
|
|
||||||
static void pop_lock()
|
static void pop_lock()
|
||||||
{
|
{
|
||||||
if (fDebug)
|
if (fDebug) {
|
||||||
{
|
|
||||||
const CLockLocation& locklocation = (*lockstack).rbegin()->second;
|
const CLockLocation& locklocation = (*lockstack).rbegin()->second;
|
||||||
LogPrint("lock", "Unlocked: %s\n", locklocation.ToString());
|
LogPrint("lock", "Unlocked: %s\n", locklocation.ToString());
|
||||||
}
|
}
|
||||||
|
@ -135,17 +135,17 @@ void LeaveCritical()
|
||||||
std::string LocksHeld()
|
std::string LocksHeld()
|
||||||
{
|
{
|
||||||
std::string result;
|
std::string result;
|
||||||
BOOST_FOREACH(const PAIRTYPE(void*, CLockLocation)&i, *lockstack)
|
BOOST_FOREACH (const PAIRTYPE(void*, CLockLocation) & i, *lockstack)
|
||||||
result += i.second.ToString() + std::string("\n");
|
result += i.second.ToString() + std::string("\n");
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AssertLockHeldInternal(const char *pszName, const char* pszFile, int nLine, void *cs)
|
void AssertLockHeldInternal(const char* pszName, const char* pszFile, int nLine, void* cs)
|
||||||
{
|
{
|
||||||
BOOST_FOREACH(const PAIRTYPE(void*, CLockLocation)&i, *lockstack)
|
BOOST_FOREACH (const PAIRTYPE(void*, CLockLocation) & i, *lockstack)
|
||||||
if (i.first == cs) return;
|
if (i.first == cs)
|
||||||
fprintf(stderr, "Assertion failed: lock %s not held in %s:%i; locks held:\n%s",
|
return;
|
||||||
pszName, pszFile, nLine, LocksHeld().c_str());
|
fprintf(stderr, "Assertion failed: lock %s not held in %s:%i; locks held:\n%s", pszName, pszFile, nLine, LocksHeld().c_str());
|
||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
70
src/sync.h
70
src/sync.h
|
@ -48,7 +48,6 @@ LEAVE_CRITICAL_SECTION(mutex); // no RAII
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
///////////////////////////////
|
///////////////////////////////
|
||||||
// //
|
// //
|
||||||
// THE ACTUAL IMPLEMENTATION //
|
// THE ACTUAL IMPLEMENTATION //
|
||||||
|
@ -63,17 +62,17 @@ class LOCKABLE AnnotatedMixin : public PARENT
|
||||||
public:
|
public:
|
||||||
void lock() EXCLUSIVE_LOCK_FUNCTION()
|
void lock() EXCLUSIVE_LOCK_FUNCTION()
|
||||||
{
|
{
|
||||||
PARENT::lock();
|
PARENT::lock();
|
||||||
}
|
}
|
||||||
|
|
||||||
void unlock() UNLOCK_FUNCTION()
|
void unlock() UNLOCK_FUNCTION()
|
||||||
{
|
{
|
||||||
PARENT::unlock();
|
PARENT::unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool try_lock() EXCLUSIVE_TRYLOCK_FUNCTION(true)
|
bool try_lock() EXCLUSIVE_TRYLOCK_FUNCTION(true)
|
||||||
{
|
{
|
||||||
return PARENT::try_lock();
|
return PARENT::try_lock();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -91,11 +90,13 @@ typedef boost::condition_variable CConditionVariable;
|
||||||
void EnterCritical(const char* pszName, const char* pszFile, int nLine, void* cs, bool fTry = false);
|
void EnterCritical(const char* pszName, const char* pszFile, int nLine, void* cs, bool fTry = false);
|
||||||
void LeaveCritical();
|
void LeaveCritical();
|
||||||
std::string LocksHeld();
|
std::string LocksHeld();
|
||||||
void AssertLockHeldInternal(const char* pszName, const char* pszFile, int nLine, void *cs);
|
void AssertLockHeldInternal(const char* pszName, const char* pszFile, int nLine, void* cs);
|
||||||
#else
|
#else
|
||||||
void static inline EnterCritical(const char* pszName, const char* pszFile, int nLine, void* cs, bool fTry = false) {}
|
void static inline EnterCritical(const char* pszName, const char* pszFile, int nLine, void* cs, bool fTry = false)
|
||||||
|
{
|
||||||
|
}
|
||||||
void static inline LeaveCritical() {}
|
void static inline LeaveCritical() {}
|
||||||
void static inline AssertLockHeldInternal(const char* pszName, const char* pszFile, int nLine, void *cs) {}
|
void static inline AssertLockHeldInternal(const char* pszName, const char* pszFile, int nLine, void* cs) {}
|
||||||
#endif
|
#endif
|
||||||
#define AssertLockHeld(cs) AssertLockHeldInternal(#cs, __FILE__, __LINE__, &cs)
|
#define AssertLockHeld(cs) AssertLockHeldInternal(#cs, __FILE__, __LINE__, &cs)
|
||||||
|
|
||||||
|
@ -104,7 +105,7 @@ void PrintLockContention(const char* pszName, const char* pszFile, int nLine);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/** Wrapper around boost::unique_lock<Mutex> */
|
/** Wrapper around boost::unique_lock<Mutex> */
|
||||||
template<typename Mutex>
|
template <typename Mutex>
|
||||||
class CMutexLock
|
class CMutexLock
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
|
@ -114,11 +115,10 @@ private:
|
||||||
{
|
{
|
||||||
EnterCritical(pszName, pszFile, nLine, (void*)(lock.mutex()));
|
EnterCritical(pszName, pszFile, nLine, (void*)(lock.mutex()));
|
||||||
#ifdef DEBUG_LOCKCONTENTION
|
#ifdef DEBUG_LOCKCONTENTION
|
||||||
if (!lock.try_lock())
|
if (!lock.try_lock()) {
|
||||||
{
|
|
||||||
PrintLockContention(pszName, pszFile, nLine);
|
PrintLockContention(pszName, pszFile, nLine);
|
||||||
#endif
|
#endif
|
||||||
lock.lock();
|
lock.lock();
|
||||||
#ifdef DEBUG_LOCKCONTENTION
|
#ifdef DEBUG_LOCKCONTENTION
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -157,19 +157,19 @@ public:
|
||||||
typedef CMutexLock<CCriticalSection> CCriticalBlock;
|
typedef CMutexLock<CCriticalSection> CCriticalBlock;
|
||||||
|
|
||||||
#define LOCK(cs) CCriticalBlock criticalblock(cs, #cs, __FILE__, __LINE__)
|
#define LOCK(cs) CCriticalBlock criticalblock(cs, #cs, __FILE__, __LINE__)
|
||||||
#define LOCK2(cs1,cs2) CCriticalBlock criticalblock1(cs1, #cs1, __FILE__, __LINE__),criticalblock2(cs2, #cs2, __FILE__, __LINE__)
|
#define LOCK2(cs1, cs2) CCriticalBlock criticalblock1(cs1, #cs1, __FILE__, __LINE__), criticalblock2(cs2, #cs2, __FILE__, __LINE__)
|
||||||
#define TRY_LOCK(cs,name) CCriticalBlock name(cs, #cs, __FILE__, __LINE__, true)
|
#define TRY_LOCK(cs, name) CCriticalBlock name(cs, #cs, __FILE__, __LINE__, true)
|
||||||
|
|
||||||
#define ENTER_CRITICAL_SECTION(cs) \
|
#define ENTER_CRITICAL_SECTION(cs) \
|
||||||
{ \
|
{ \
|
||||||
EnterCritical(#cs, __FILE__, __LINE__, (void*)(&cs)); \
|
EnterCritical(#cs, __FILE__, __LINE__, (void*)(&cs)); \
|
||||||
(cs).lock(); \
|
(cs).lock(); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define LEAVE_CRITICAL_SECTION(cs) \
|
#define LEAVE_CRITICAL_SECTION(cs) \
|
||||||
{ \
|
{ \
|
||||||
(cs).unlock(); \
|
(cs).unlock(); \
|
||||||
LeaveCritical(); \
|
LeaveCritical(); \
|
||||||
}
|
}
|
||||||
|
|
||||||
class CSemaphore
|
class CSemaphore
|
||||||
|
@ -182,7 +182,8 @@ private:
|
||||||
public:
|
public:
|
||||||
CSemaphore(int init) : value(init) {}
|
CSemaphore(int init) : value(init) {}
|
||||||
|
|
||||||
void wait() {
|
void wait()
|
||||||
|
{
|
||||||
boost::unique_lock<boost::mutex> lock(mutex);
|
boost::unique_lock<boost::mutex> lock(mutex);
|
||||||
while (value < 1) {
|
while (value < 1) {
|
||||||
condition.wait(lock);
|
condition.wait(lock);
|
||||||
|
@ -190,7 +191,8 @@ public:
|
||||||
value--;
|
value--;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool try_wait() {
|
bool try_wait()
|
||||||
|
{
|
||||||
boost::unique_lock<boost::mutex> lock(mutex);
|
boost::unique_lock<boost::mutex> lock(mutex);
|
||||||
if (value < 1)
|
if (value < 1)
|
||||||
return false;
|
return false;
|
||||||
|
@ -198,7 +200,8 @@ public:
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void post() {
|
void post()
|
||||||
|
{
|
||||||
{
|
{
|
||||||
boost::unique_lock<boost::mutex> lock(mutex);
|
boost::unique_lock<boost::mutex> lock(mutex);
|
||||||
value++;
|
value++;
|
||||||
|
@ -211,31 +214,35 @@ public:
|
||||||
class CSemaphoreGrant
|
class CSemaphoreGrant
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
CSemaphore *sem;
|
CSemaphore* sem;
|
||||||
bool fHaveGrant;
|
bool fHaveGrant;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
void Acquire() {
|
void Acquire()
|
||||||
|
{
|
||||||
if (fHaveGrant)
|
if (fHaveGrant)
|
||||||
return;
|
return;
|
||||||
sem->wait();
|
sem->wait();
|
||||||
fHaveGrant = true;
|
fHaveGrant = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Release() {
|
void Release()
|
||||||
|
{
|
||||||
if (!fHaveGrant)
|
if (!fHaveGrant)
|
||||||
return;
|
return;
|
||||||
sem->post();
|
sem->post();
|
||||||
fHaveGrant = false;
|
fHaveGrant = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool TryAcquire() {
|
bool TryAcquire()
|
||||||
|
{
|
||||||
if (!fHaveGrant && sem->try_wait())
|
if (!fHaveGrant && sem->try_wait())
|
||||||
fHaveGrant = true;
|
fHaveGrant = true;
|
||||||
return fHaveGrant;
|
return fHaveGrant;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MoveTo(CSemaphoreGrant &grant) {
|
void MoveTo(CSemaphoreGrant& grant)
|
||||||
|
{
|
||||||
grant.Release();
|
grant.Release();
|
||||||
grant.sem = sem;
|
grant.sem = sem;
|
||||||
grant.fHaveGrant = fHaveGrant;
|
grant.fHaveGrant = fHaveGrant;
|
||||||
|
@ -245,18 +252,21 @@ public:
|
||||||
|
|
||||||
CSemaphoreGrant() : sem(NULL), fHaveGrant(false) {}
|
CSemaphoreGrant() : sem(NULL), fHaveGrant(false) {}
|
||||||
|
|
||||||
CSemaphoreGrant(CSemaphore &sema, bool fTry = false) : sem(&sema), fHaveGrant(false) {
|
CSemaphoreGrant(CSemaphore& sema, bool fTry = false) : sem(&sema), fHaveGrant(false)
|
||||||
|
{
|
||||||
if (fTry)
|
if (fTry)
|
||||||
TryAcquire();
|
TryAcquire();
|
||||||
else
|
else
|
||||||
Acquire();
|
Acquire();
|
||||||
}
|
}
|
||||||
|
|
||||||
~CSemaphoreGrant() {
|
~CSemaphoreGrant()
|
||||||
|
{
|
||||||
Release();
|
Release();
|
||||||
}
|
}
|
||||||
|
|
||||||
operator bool() {
|
operator bool()
|
||||||
|
{
|
||||||
return fHaveGrant;
|
return fHaveGrant;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -13,24 +13,24 @@
|
||||||
// See http://clang.llvm.org/docs/LanguageExtensions.html#threadsafety
|
// See http://clang.llvm.org/docs/LanguageExtensions.html#threadsafety
|
||||||
// for documentation. The clang compiler can do advanced static analysis
|
// for documentation. The clang compiler can do advanced static analysis
|
||||||
// of locking when given the -Wthread-safety option.
|
// of locking when given the -Wthread-safety option.
|
||||||
#define LOCKABLE __attribute__ ((lockable))
|
#define LOCKABLE __attribute__((lockable))
|
||||||
#define SCOPED_LOCKABLE __attribute__ ((scoped_lockable))
|
#define SCOPED_LOCKABLE __attribute__((scoped_lockable))
|
||||||
#define GUARDED_BY(x) __attribute__ ((guarded_by(x)))
|
#define GUARDED_BY(x) __attribute__((guarded_by(x)))
|
||||||
#define GUARDED_VAR __attribute__ ((guarded_var))
|
#define GUARDED_VAR __attribute__((guarded_var))
|
||||||
#define PT_GUARDED_BY(x) __attribute__ ((pt_guarded_by(x)))
|
#define PT_GUARDED_BY(x) __attribute__((pt_guarded_by(x)))
|
||||||
#define PT_GUARDED_VAR __attribute__ ((pt_guarded_var))
|
#define PT_GUARDED_VAR __attribute__((pt_guarded_var))
|
||||||
#define ACQUIRED_AFTER(...) __attribute__ ((acquired_after(__VA_ARGS__)))
|
#define ACQUIRED_AFTER(...) __attribute__((acquired_after(__VA_ARGS__)))
|
||||||
#define ACQUIRED_BEFORE(...) __attribute__ ((acquired_before(__VA_ARGS__)))
|
#define ACQUIRED_BEFORE(...) __attribute__((acquired_before(__VA_ARGS__)))
|
||||||
#define EXCLUSIVE_LOCK_FUNCTION(...) __attribute__ ((exclusive_lock_function(__VA_ARGS__)))
|
#define EXCLUSIVE_LOCK_FUNCTION(...) __attribute__((exclusive_lock_function(__VA_ARGS__)))
|
||||||
#define SHARED_LOCK_FUNCTION(...) __attribute__ ((shared_lock_function(__VA_ARGS__)))
|
#define SHARED_LOCK_FUNCTION(...) __attribute__((shared_lock_function(__VA_ARGS__)))
|
||||||
#define EXCLUSIVE_TRYLOCK_FUNCTION(...) __attribute__ ((exclusive_trylock_function(__VA_ARGS__)))
|
#define EXCLUSIVE_TRYLOCK_FUNCTION(...) __attribute__((exclusive_trylock_function(__VA_ARGS__)))
|
||||||
#define SHARED_TRYLOCK_FUNCTION(...) __attribute__ ((shared_trylock_function(__VA_ARGS__)))
|
#define SHARED_TRYLOCK_FUNCTION(...) __attribute__((shared_trylock_function(__VA_ARGS__)))
|
||||||
#define UNLOCK_FUNCTION(...) __attribute__ ((unlock_function(__VA_ARGS__)))
|
#define UNLOCK_FUNCTION(...) __attribute__((unlock_function(__VA_ARGS__)))
|
||||||
#define LOCK_RETURNED(x) __attribute__ ((lock_returned(x)))
|
#define LOCK_RETURNED(x) __attribute__((lock_returned(x)))
|
||||||
#define LOCKS_EXCLUDED(...) __attribute__ ((locks_excluded(__VA_ARGS__)))
|
#define LOCKS_EXCLUDED(...) __attribute__((locks_excluded(__VA_ARGS__)))
|
||||||
#define EXCLUSIVE_LOCKS_REQUIRED(...) __attribute__ ((exclusive_locks_required(__VA_ARGS__)))
|
#define EXCLUSIVE_LOCKS_REQUIRED(...) __attribute__((exclusive_locks_required(__VA_ARGS__)))
|
||||||
#define SHARED_LOCKS_REQUIRED(...) __attribute__ ((shared_locks_required(__VA_ARGS__)))
|
#define SHARED_LOCKS_REQUIRED(...) __attribute__((shared_locks_required(__VA_ARGS__)))
|
||||||
#define NO_THREAD_SAFETY_ANALYSIS __attribute__ ((no_thread_safety_analysis))
|
#define NO_THREAD_SAFETY_ANALYSIS __attribute__((no_thread_safety_analysis))
|
||||||
#else
|
#else
|
||||||
#define LOCKABLE
|
#define LOCKABLE
|
||||||
#define SCOPED_LOCKABLE
|
#define SCOPED_LOCKABLE
|
||||||
|
@ -50,6 +50,6 @@
|
||||||
#define EXCLUSIVE_LOCKS_REQUIRED(...)
|
#define EXCLUSIVE_LOCKS_REQUIRED(...)
|
||||||
#define SHARED_LOCKS_REQUIRED(...)
|
#define SHARED_LOCKS_REQUIRED(...)
|
||||||
#define NO_THREAD_SAFETY_ANALYSIS
|
#define NO_THREAD_SAFETY_ANALYSIS
|
||||||
#endif // __GNUC__
|
#endif // __GNUC__
|
||||||
|
|
||||||
#endif // BITCOIN_THREADSAFETY_H
|
#endif // BITCOIN_THREADSAFETY_H
|
||||||
|
|
|
@ -15,15 +15,16 @@ class CNetAddr;
|
||||||
/** Median filter over a stream of values.
|
/** Median filter over a stream of values.
|
||||||
* Returns the median of the last N numbers
|
* Returns the median of the last N numbers
|
||||||
*/
|
*/
|
||||||
template <typename T> class CMedianFilter
|
template <typename T>
|
||||||
|
class CMedianFilter
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
std::vector<T> vValues;
|
std::vector<T> vValues;
|
||||||
std::vector<T> vSorted;
|
std::vector<T> vSorted;
|
||||||
unsigned int nSize;
|
unsigned int nSize;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
CMedianFilter(unsigned int size, T initial_value):
|
CMedianFilter(unsigned int size, T initial_value) : nSize(size)
|
||||||
nSize(size)
|
|
||||||
{
|
{
|
||||||
vValues.reserve(size);
|
vValues.reserve(size);
|
||||||
vValues.push_back(initial_value);
|
vValues.push_back(initial_value);
|
||||||
|
@ -32,8 +33,7 @@ public:
|
||||||
|
|
||||||
void input(T value)
|
void input(T value)
|
||||||
{
|
{
|
||||||
if(vValues.size() == nSize)
|
if (vValues.size() == nSize) {
|
||||||
{
|
|
||||||
vValues.erase(vValues.begin());
|
vValues.erase(vValues.begin());
|
||||||
}
|
}
|
||||||
vValues.push_back(value);
|
vValues.push_back(value);
|
||||||
|
@ -46,14 +46,13 @@ public:
|
||||||
T median() const
|
T median() const
|
||||||
{
|
{
|
||||||
int size = vSorted.size();
|
int size = vSorted.size();
|
||||||
assert(size>0);
|
assert(size > 0);
|
||||||
if(size & 1) // Odd number of elements
|
if (size & 1) // Odd number of elements
|
||||||
{
|
{
|
||||||
return vSorted[size/2];
|
return vSorted[size / 2];
|
||||||
}
|
} else // Even number of elements
|
||||||
else // Even number of elements
|
|
||||||
{
|
{
|
||||||
return (vSorted[size/2-1] + vSorted[size/2]) / 2;
|
return (vSorted[size / 2 - 1] + vSorted[size / 2]) / 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,7 +61,7 @@ public:
|
||||||
return vValues.size();
|
return vValues.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<T> sorted () const
|
std::vector<T> sorted() const
|
||||||
{
|
{
|
||||||
return vSorted;
|
return vSorted;
|
||||||
}
|
}
|
||||||
|
|
133
src/uint256.cpp
133
src/uint256.cpp
|
@ -10,13 +10,13 @@
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>::base_uint(const std::string& str)
|
base_uint<BITS>::base_uint(const std::string& str)
|
||||||
{
|
{
|
||||||
SetHex(str);
|
SetHex(str);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>::base_uint(const std::vector<unsigned char>& vch)
|
base_uint<BITS>::base_uint(const std::vector<unsigned char>& vch)
|
||||||
{
|
{
|
||||||
if (vch.size() != sizeof(pn))
|
if (vch.size() != sizeof(pn))
|
||||||
|
@ -24,7 +24,7 @@ base_uint<BITS>::base_uint(const std::vector<unsigned char>& vch)
|
||||||
memcpy(pn, &vch[0], sizeof(pn));
|
memcpy(pn, &vch[0], sizeof(pn));
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>& base_uint<BITS>::operator<<=(unsigned int shift)
|
base_uint<BITS>& base_uint<BITS>::operator<<=(unsigned int shift)
|
||||||
{
|
{
|
||||||
base_uint<BITS> a(*this);
|
base_uint<BITS> a(*this);
|
||||||
|
@ -33,15 +33,15 @@ base_uint<BITS>& base_uint<BITS>::operator<<=(unsigned int shift)
|
||||||
int k = shift / 32;
|
int k = shift / 32;
|
||||||
shift = shift % 32;
|
shift = shift % 32;
|
||||||
for (int i = 0; i < WIDTH; i++) {
|
for (int i = 0; i < WIDTH; i++) {
|
||||||
if (i+k+1 < WIDTH && shift != 0)
|
if (i + k + 1 < WIDTH && shift != 0)
|
||||||
pn[i+k+1] |= (a.pn[i] >> (32-shift));
|
pn[i + k + 1] |= (a.pn[i] >> (32 - shift));
|
||||||
if (i+k < WIDTH)
|
if (i + k < WIDTH)
|
||||||
pn[i+k] |= (a.pn[i] << shift);
|
pn[i + k] |= (a.pn[i] << shift);
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>& base_uint<BITS>::operator>>=(unsigned int shift)
|
base_uint<BITS>& base_uint<BITS>::operator>>=(unsigned int shift)
|
||||||
{
|
{
|
||||||
base_uint<BITS> a(*this);
|
base_uint<BITS> a(*this);
|
||||||
|
@ -50,15 +50,15 @@ base_uint<BITS>& base_uint<BITS>::operator>>=(unsigned int shift)
|
||||||
int k = shift / 32;
|
int k = shift / 32;
|
||||||
shift = shift % 32;
|
shift = shift % 32;
|
||||||
for (int i = 0; i < WIDTH; i++) {
|
for (int i = 0; i < WIDTH; i++) {
|
||||||
if (i-k-1 >= 0 && shift != 0)
|
if (i - k - 1 >= 0 && shift != 0)
|
||||||
pn[i-k-1] |= (a.pn[i] << (32-shift));
|
pn[i - k - 1] |= (a.pn[i] << (32 - shift));
|
||||||
if (i-k >= 0)
|
if (i - k >= 0)
|
||||||
pn[i-k] |= (a.pn[i] >> shift);
|
pn[i - k] |= (a.pn[i] >> shift);
|
||||||
}
|
}
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>& base_uint<BITS>::operator*=(uint32_t b32)
|
base_uint<BITS>& base_uint<BITS>::operator*=(uint32_t b32)
|
||||||
{
|
{
|
||||||
uint64_t carry = 0;
|
uint64_t carry = 0;
|
||||||
|
@ -70,7 +70,7 @@ base_uint<BITS>& base_uint<BITS>::operator*=(uint32_t b32)
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>& base_uint<BITS>::operator*=(const base_uint& b)
|
base_uint<BITS>& base_uint<BITS>::operator*=(const base_uint& b)
|
||||||
{
|
{
|
||||||
base_uint<BITS> a = *this;
|
base_uint<BITS> a = *this;
|
||||||
|
@ -86,12 +86,12 @@ base_uint<BITS>& base_uint<BITS>::operator*=(const base_uint& b)
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
base_uint<BITS>& base_uint<BITS>::operator/=(const base_uint& b)
|
base_uint<BITS>& base_uint<BITS>::operator/=(const base_uint& b)
|
||||||
{
|
{
|
||||||
base_uint<BITS> div = b; // make a copy, so we can shift.
|
base_uint<BITS> div = b; // make a copy, so we can shift.
|
||||||
base_uint<BITS> num = *this; // make a copy, so we can subtract.
|
base_uint<BITS> num = *this; // make a copy, so we can subtract.
|
||||||
*this = 0; // the quotient.
|
*this = 0; // the quotient.
|
||||||
int num_bits = num.bits();
|
int num_bits = num.bits();
|
||||||
int div_bits = div.bits();
|
int div_bits = div.bits();
|
||||||
if (div_bits == 0)
|
if (div_bits == 0)
|
||||||
|
@ -112,9 +112,10 @@ base_uint<BITS>& base_uint<BITS>::operator/=(const base_uint& b)
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
int base_uint<BITS>::CompareTo(const base_uint<BITS>& b) const {
|
int base_uint<BITS>::CompareTo(const base_uint<BITS>& b) const
|
||||||
for (int i = WIDTH-1; i >= 0; i--) {
|
{
|
||||||
|
for (int i = WIDTH - 1; i >= 0; i--) {
|
||||||
if (pn[i] < b.pn[i])
|
if (pn[i] < b.pn[i])
|
||||||
return -1;
|
return -1;
|
||||||
if (pn[i] > b.pn[i])
|
if (pn[i] > b.pn[i])
|
||||||
|
@ -123,9 +124,10 @@ int base_uint<BITS>::CompareTo(const base_uint<BITS>& b) const {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
bool base_uint<BITS>::EqualTo(uint64_t b) const {
|
bool base_uint<BITS>::EqualTo(uint64_t b) const
|
||||||
for (int i = WIDTH-1; i >= 2; i--) {
|
{
|
||||||
|
for (int i = WIDTH - 1; i >= 2; i--) {
|
||||||
if (pn[i])
|
if (pn[i])
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -136,7 +138,7 @@ bool base_uint<BITS>::EqualTo(uint64_t b) const {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
double base_uint<BITS>::getdouble() const
|
double base_uint<BITS>::getdouble() const
|
||||||
{
|
{
|
||||||
double ret = 0.0;
|
double ret = 0.0;
|
||||||
|
@ -148,19 +150,19 @@ double base_uint<BITS>::getdouble() const
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
std::string base_uint<BITS>::GetHex() const
|
std::string base_uint<BITS>::GetHex() const
|
||||||
{
|
{
|
||||||
char psz[sizeof(pn)*2 + 1];
|
char psz[sizeof(pn) * 2 + 1];
|
||||||
for (unsigned int i = 0; i < sizeof(pn); i++)
|
for (unsigned int i = 0; i < sizeof(pn); i++)
|
||||||
sprintf(psz + i*2, "%02x", ((unsigned char*)pn)[sizeof(pn) - i - 1]);
|
sprintf(psz + i * 2, "%02x", ((unsigned char*)pn)[sizeof(pn) - i - 1]);
|
||||||
return std::string(psz, psz + sizeof(pn)*2);
|
return std::string(psz, psz + sizeof(pn) * 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
void base_uint<BITS>::SetHex(const char* psz)
|
void base_uint<BITS>::SetHex(const char* psz)
|
||||||
{
|
{
|
||||||
memset(pn,0,sizeof(pn));
|
memset(pn, 0, sizeof(pn));
|
||||||
|
|
||||||
// skip leading spaces
|
// skip leading spaces
|
||||||
while (isspace(*psz))
|
while (isspace(*psz))
|
||||||
|
@ -186,28 +188,28 @@ void base_uint<BITS>::SetHex(const char* psz)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
void base_uint<BITS>::SetHex(const std::string& str)
|
void base_uint<BITS>::SetHex(const std::string& str)
|
||||||
{
|
{
|
||||||
SetHex(str.c_str());
|
SetHex(str.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
std::string base_uint<BITS>::ToString() const
|
std::string base_uint<BITS>::ToString() const
|
||||||
{
|
{
|
||||||
return (GetHex());
|
return (GetHex());
|
||||||
}
|
}
|
||||||
|
|
||||||
template<unsigned int BITS>
|
template <unsigned int BITS>
|
||||||
unsigned int base_uint<BITS>::bits() const
|
unsigned int base_uint<BITS>::bits() const
|
||||||
{
|
{
|
||||||
for (int pos = WIDTH-1; pos >= 0; pos--) {
|
for (int pos = WIDTH - 1; pos >= 0; pos--) {
|
||||||
if (pn[pos]) {
|
if (pn[pos]) {
|
||||||
for (int bits = 31; bits > 0; bits--) {
|
for (int bits = 31; bits > 0; bits--) {
|
||||||
if (pn[pos] & 1<<bits)
|
if (pn[pos] & 1 << bits)
|
||||||
return 32*pos + bits + 1;
|
return 32 * pos + bits + 1;
|
||||||
}
|
}
|
||||||
return 32*pos + 1;
|
return 32 * pos + 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -249,16 +251,16 @@ template unsigned int base_uint<256>::bits() const;
|
||||||
|
|
||||||
// This implementation directly uses shifts instead of going
|
// This implementation directly uses shifts instead of going
|
||||||
// through an intermediate MPI representation.
|
// through an intermediate MPI representation.
|
||||||
uint256& uint256::SetCompact(uint32_t nCompact, bool *pfNegative, bool *pfOverflow)
|
uint256& uint256::SetCompact(uint32_t nCompact, bool* pfNegative, bool* pfOverflow)
|
||||||
{
|
{
|
||||||
int nSize = nCompact >> 24;
|
int nSize = nCompact >> 24;
|
||||||
uint32_t nWord = nCompact & 0x007fffff;
|
uint32_t nWord = nCompact & 0x007fffff;
|
||||||
if (nSize <= 3) {
|
if (nSize <= 3) {
|
||||||
nWord >>= 8*(3-nSize);
|
nWord >>= 8 * (3 - nSize);
|
||||||
*this = nWord;
|
*this = nWord;
|
||||||
} else {
|
} else {
|
||||||
*this = nWord;
|
*this = nWord;
|
||||||
*this <<= 8*(nSize-3);
|
*this <<= 8 * (nSize - 3);
|
||||||
}
|
}
|
||||||
if (pfNegative)
|
if (pfNegative)
|
||||||
*pfNegative = nWord != 0 && (nCompact & 0x00800000) != 0;
|
*pfNegative = nWord != 0 && (nCompact & 0x00800000) != 0;
|
||||||
|
@ -274,9 +276,9 @@ uint32_t uint256::GetCompact(bool fNegative) const
|
||||||
int nSize = (bits() + 7) / 8;
|
int nSize = (bits() + 7) / 8;
|
||||||
uint32_t nCompact = 0;
|
uint32_t nCompact = 0;
|
||||||
if (nSize <= 3) {
|
if (nSize <= 3) {
|
||||||
nCompact = GetLow64() << 8*(3-nSize);
|
nCompact = GetLow64() << 8 * (3 - nSize);
|
||||||
} else {
|
} else {
|
||||||
uint256 bn = *this >> 8*(nSize-3);
|
uint256 bn = *this >> 8 * (nSize - 3);
|
||||||
nCompact = bn.GetLow64();
|
nCompact = bn.GetLow64();
|
||||||
}
|
}
|
||||||
// The 0x00800000 bit denotes the sign.
|
// The 0x00800000 bit denotes the sign.
|
||||||
|
@ -295,27 +297,46 @@ uint32_t uint256::GetCompact(bool fNegative) const
|
||||||
static void inline HashMix(uint32_t& a, uint32_t& b, uint32_t& c)
|
static void inline HashMix(uint32_t& a, uint32_t& b, uint32_t& c)
|
||||||
{
|
{
|
||||||
// Taken from lookup3, by Bob Jenkins.
|
// Taken from lookup3, by Bob Jenkins.
|
||||||
a -= c; a ^= ((c << 4) | (c >> 28)); c += b;
|
a -= c;
|
||||||
b -= a; b ^= ((a << 6) | (a >> 26)); a += c;
|
a ^= ((c << 4) | (c >> 28));
|
||||||
c -= b; c ^= ((b << 8) | (b >> 24)); b += a;
|
c += b;
|
||||||
a -= c; a ^= ((c << 16) | (c >> 16)); c += b;
|
b -= a;
|
||||||
b -= a; b ^= ((a << 19) | (a >> 13)); a += c;
|
b ^= ((a << 6) | (a >> 26));
|
||||||
c -= b; c ^= ((b << 4) | (b >> 28)); b += a;
|
a += c;
|
||||||
|
c -= b;
|
||||||
|
c ^= ((b << 8) | (b >> 24));
|
||||||
|
b += a;
|
||||||
|
a -= c;
|
||||||
|
a ^= ((c << 16) | (c >> 16));
|
||||||
|
c += b;
|
||||||
|
b -= a;
|
||||||
|
b ^= ((a << 19) | (a >> 13));
|
||||||
|
a += c;
|
||||||
|
c -= b;
|
||||||
|
c ^= ((b << 4) | (b >> 28));
|
||||||
|
b += a;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void inline HashFinal(uint32_t& a, uint32_t& b, uint32_t& c)
|
static void inline HashFinal(uint32_t& a, uint32_t& b, uint32_t& c)
|
||||||
{
|
{
|
||||||
// Taken from lookup3, by Bob Jenkins.
|
// Taken from lookup3, by Bob Jenkins.
|
||||||
c ^= b; c -= ((b << 14) | (b >> 18));
|
c ^= b;
|
||||||
a ^= c; a -= ((c << 11) | (c >> 21));
|
c -= ((b << 14) | (b >> 18));
|
||||||
b ^= a; b -= ((a << 25) | (a >> 7));
|
a ^= c;
|
||||||
c ^= b; c -= ((b << 16) | (b >> 16));
|
a -= ((c << 11) | (c >> 21));
|
||||||
a ^= c; a -= ((c << 4) | (c >> 28));
|
b ^= a;
|
||||||
b ^= a; b -= ((a << 14) | (a >> 18));
|
b -= ((a << 25) | (a >> 7));
|
||||||
c ^= b; c -= ((b << 24) | (b >> 8));
|
c ^= b;
|
||||||
|
c -= ((b << 16) | (b >> 16));
|
||||||
|
a ^= c;
|
||||||
|
a -= ((c << 4) | (c >> 28));
|
||||||
|
b ^= a;
|
||||||
|
b -= ((a << 14) | (a >> 18));
|
||||||
|
c ^= b;
|
||||||
|
c -= ((b << 24) | (b >> 8));
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t uint256::GetHash(const uint256 &salt) const
|
uint64_t uint256::GetHash(const uint256& salt) const
|
||||||
{
|
{
|
||||||
uint32_t a, b, c;
|
uint32_t a, b, c;
|
||||||
a = b = c = 0xdeadbeef + (WIDTH << 2);
|
a = b = c = 0xdeadbeef + (WIDTH << 2);
|
||||||
|
|
|
@ -16,7 +16,7 @@
|
||||||
const std::string CLIENT_NAME("Satoshi");
|
const std::string CLIENT_NAME("Satoshi");
|
||||||
|
|
||||||
// Client version number
|
// Client version number
|
||||||
#define CLIENT_VERSION_SUFFIX ""
|
#define CLIENT_VERSION_SUFFIX ""
|
||||||
|
|
||||||
|
|
||||||
// The following part of the code determines the CLIENT_BUILD variable.
|
// The following part of the code determines the CLIENT_BUILD variable.
|
||||||
|
@ -35,40 +35,40 @@ const std::string CLIENT_NAME("Satoshi");
|
||||||
|
|
||||||
// First, include build.h if requested
|
// First, include build.h if requested
|
||||||
#ifdef HAVE_BUILD_INFO
|
#ifdef HAVE_BUILD_INFO
|
||||||
# include "build.h"
|
#include "build.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// git will put "#define GIT_ARCHIVE 1" on the next line inside archives. $Format:%n#define GIT_ARCHIVE 1$
|
// git will put "#define GIT_ARCHIVE 1" on the next line inside archives. $Format:%n#define GIT_ARCHIVE 1$
|
||||||
#ifdef GIT_ARCHIVE
|
#ifdef GIT_ARCHIVE
|
||||||
# define GIT_COMMIT_ID "$Format:%h$"
|
#define GIT_COMMIT_ID "$Format:%h$"
|
||||||
# define GIT_COMMIT_DATE "$Format:%cD$"
|
#define GIT_COMMIT_DATE "$Format:%cD$"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define BUILD_DESC_WITH_SUFFIX(maj,min,rev,build,suffix) \
|
#define BUILD_DESC_WITH_SUFFIX(maj, min, rev, build, suffix) \
|
||||||
"v" DO_STRINGIZE(maj) "." DO_STRINGIZE(min) "." DO_STRINGIZE(rev) "." DO_STRINGIZE(build) "-" DO_STRINGIZE(suffix)
|
"v" DO_STRINGIZE(maj) "." DO_STRINGIZE(min) "." DO_STRINGIZE(rev) "." DO_STRINGIZE(build) "-" DO_STRINGIZE(suffix)
|
||||||
|
|
||||||
#define BUILD_DESC_FROM_COMMIT(maj,min,rev,build,commit) \
|
#define BUILD_DESC_FROM_COMMIT(maj, min, rev, build, commit) \
|
||||||
"v" DO_STRINGIZE(maj) "." DO_STRINGIZE(min) "." DO_STRINGIZE(rev) "." DO_STRINGIZE(build) "-g" commit
|
"v" DO_STRINGIZE(maj) "." DO_STRINGIZE(min) "." DO_STRINGIZE(rev) "." DO_STRINGIZE(build) "-g" commit
|
||||||
|
|
||||||
#define BUILD_DESC_FROM_UNKNOWN(maj,min,rev,build) \
|
#define BUILD_DESC_FROM_UNKNOWN(maj, min, rev, build) \
|
||||||
"v" DO_STRINGIZE(maj) "." DO_STRINGIZE(min) "." DO_STRINGIZE(rev) "." DO_STRINGIZE(build) "-unk"
|
"v" DO_STRINGIZE(maj) "." DO_STRINGIZE(min) "." DO_STRINGIZE(rev) "." DO_STRINGIZE(build) "-unk"
|
||||||
|
|
||||||
#ifndef BUILD_DESC
|
#ifndef BUILD_DESC
|
||||||
# ifdef BUILD_SUFFIX
|
#ifdef BUILD_SUFFIX
|
||||||
# define BUILD_DESC BUILD_DESC_WITH_SUFFIX(CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR, CLIENT_VERSION_REVISION, CLIENT_VERSION_BUILD, BUILD_SUFFIX)
|
#define BUILD_DESC BUILD_DESC_WITH_SUFFIX(CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR, CLIENT_VERSION_REVISION, CLIENT_VERSION_BUILD, BUILD_SUFFIX)
|
||||||
# elif defined(GIT_COMMIT_ID)
|
#elif defined(GIT_COMMIT_ID)
|
||||||
# define BUILD_DESC BUILD_DESC_FROM_COMMIT(CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR, CLIENT_VERSION_REVISION, CLIENT_VERSION_BUILD, GIT_COMMIT_ID)
|
#define BUILD_DESC BUILD_DESC_FROM_COMMIT(CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR, CLIENT_VERSION_REVISION, CLIENT_VERSION_BUILD, GIT_COMMIT_ID)
|
||||||
# else
|
#else
|
||||||
# define BUILD_DESC BUILD_DESC_FROM_UNKNOWN(CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR, CLIENT_VERSION_REVISION, CLIENT_VERSION_BUILD)
|
#define BUILD_DESC BUILD_DESC_FROM_UNKNOWN(CLIENT_VERSION_MAJOR, CLIENT_VERSION_MINOR, CLIENT_VERSION_REVISION, CLIENT_VERSION_BUILD)
|
||||||
# endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef BUILD_DATE
|
#ifndef BUILD_DATE
|
||||||
# ifdef GIT_COMMIT_DATE
|
#ifdef GIT_COMMIT_DATE
|
||||||
# define BUILD_DATE GIT_COMMIT_DATE
|
#define BUILD_DATE GIT_COMMIT_DATE
|
||||||
# else
|
#else
|
||||||
# define BUILD_DATE __DATE__ ", " __TIME__
|
#define BUILD_DATE __DATE__ ", " __TIME__
|
||||||
# endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
const std::string CLIENT_BUILD(BUILD_DESC CLIENT_VERSION_SUFFIX);
|
const std::string CLIENT_BUILD(BUILD_DESC CLIENT_VERSION_SUFFIX);
|
||||||
|
@ -76,10 +76,10 @@ const std::string CLIENT_DATE(BUILD_DATE);
|
||||||
|
|
||||||
static std::string FormatVersion(int nVersion)
|
static std::string FormatVersion(int nVersion)
|
||||||
{
|
{
|
||||||
if (nVersion%100 == 0)
|
if (nVersion % 100 == 0)
|
||||||
return strprintf("%d.%d.%d", nVersion/1000000, (nVersion/10000)%100, (nVersion/100)%100);
|
return strprintf("%d.%d.%d", nVersion / 1000000, (nVersion / 10000) % 100, (nVersion / 100) % 100);
|
||||||
else
|
else
|
||||||
return strprintf("%d.%d.%d.%d", nVersion/1000000, (nVersion/10000)%100, (nVersion/100)%100, nVersion%100);
|
return strprintf("%d.%d.%d.%d", nVersion / 1000000, (nVersion / 10000) % 100, (nVersion / 100) % 100, nVersion % 100);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string FormatFullVersion()
|
std::string FormatFullVersion()
|
||||||
|
|
Loading…
Add table
Reference in a new issue