Browse Source

Merge pull request #4933

20e01b1 Apply clang-format on some infrequently-updated files (Pieter Wuille)
0.10
Pieter Wuille 10 years ago
parent
commit
934b153a2c
No known key found for this signature in database
GPG Key ID: 8F653255C87992E0
  1. 225
      src/addrman.cpp
  2. 7
      src/allocators.cpp
  3. 90
      src/allocators.h
  4. 129
      src/base58.cpp
  5. 56
      src/chainparamsbase.cpp
  6. 3
      src/chainparamsbase.h
  7. 20
      src/checkpoints.h
  8. 59
      src/checkqueue.h
  9. 12
      src/clientversion.h
  10. 1
      src/coincontrol.h
  11. 144
      src/db.cpp
  12. 33
      src/db.h
  13. 32
      src/hash.cpp
  14. 8
      src/init.h
  15. 131
      src/key.h
  16. 15
      src/leveldbwrapper.cpp
  17. 47
      src/leveldbwrapper.h
  18. 18
      src/limitedmap.h
  19. 18
      src/mruset.h
  20. 2
      src/noui.cpp
  21. 151
      src/protocol.h
  22. 28
      src/random.cpp
  23. 2
      src/random.h
  24. 2
      src/rpcclient.h
  25. 48
      src/sync.cpp
  26. 70
      src/sync.h
  27. 40
      src/threadsafety.h
  28. 23
      src/timedata.h
  29. 133
      src/uint256.cpp
  30. 44
      src/version.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)

7
src/allocators.cpp

@ -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())
{ {
} }

90
src/allocators.h

@ -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,10 +133,10 @@ 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()
{ {
boost::call_once(LockedPageManager::CreateInstance, LockedPageManager::init_flag); boost::call_once(LockedPageManager::CreateInstance, LockedPageManager::init_flag);
return *LockedPageManager::_instance; return *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() {} ~secure_allocator() throw() {}
template<typename _Other> struct rebind template <typename _Other>
{ typedef secure_allocator<_Other> other; }; struct rebind {
typedef secure_allocator<_Other> other;
};
T* allocate(std::size_t n, const void *hint = 0) T* allocate(std::size_t n, const void* hint = 0)
{ {
T *p; 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

@ -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 &params) 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());
} }

56
src/chainparamsbase.cpp

@ -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 = &regTestParams; pCurrentBaseParams = &regTestParams;
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;
} }

3
src/chainparamsbase.h

@ -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);

20
src/checkpoints.h

@ -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

59
src/checkqueue.h

@ -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();
} }

12
src/clientversion.h

@ -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

1
src/coincontrol.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

@ -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

@ -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

@ -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;
}; };
//---------- //----------

8
src/init.h

@ -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

@ -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 */

15
src/leveldbwrapper.cpp

@ -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;

47
src/leveldbwrapper.h

@ -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);
} }
}; };

18
src/limitedmap.h

@ -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

@ -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();
} }

2
src/noui.cpp

@ -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);
} }

151
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,

28
src/random.cpp

@ -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;
} }
} }

2
src/random.h

@ -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();

2
src/rpcclient.h

@ -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

@ -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

@ -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;
} }
}; };

40
src/threadsafety.h

@ -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

23
src/timedata.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

@ -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);

44
src/version.cpp

@ -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…
Cancel
Save