Browse Source

CAddrDB: Replace BDB-managed addr.dat with internally managed peers.dat

0.8
Jeff Garzik 13 years ago committed by Jeff Garzik
parent
commit
928d3a011c
  1. 125
      src/db.cpp
  2. 17
      src/db.h
  3. 12
      src/init.cpp
  4. 7
      src/net.cpp
  5. 1
      src/net.h

125
src/db.cpp

@ -164,8 +164,6 @@ void CDB::Close()
unsigned int nMinutes = 0; unsigned int nMinutes = 0;
if (fReadOnly) if (fReadOnly)
nMinutes = 1; nMinutes = 1;
if (strFile == "addr.dat")
nMinutes = 2;
if (strFile == "blkindex.dat") if (strFile == "blkindex.dat")
nMinutes = 2; nMinutes = 2;
if (strFile == "blkindex.dat" && IsInitialBlockDownload()) if (strFile == "blkindex.dat" && IsInitialBlockDownload())
@ -310,7 +308,7 @@ void DBFlush(bool fShutdown)
CloseDb(strFile); CloseDb(strFile);
printf("%s checkpoint\n", strFile.c_str()); printf("%s checkpoint\n", strFile.c_str());
dbenv.txn_checkpoint(0, 0, 0); dbenv.txn_checkpoint(0, 0, 0);
if ((strFile != "blkindex.dat" && strFile != "addr.dat") || fDetachDB) { if (strFile != "blkindex.dat" || fDetachDB) {
printf("%s detach\n", strFile.c_str()); printf("%s detach\n", strFile.c_str());
dbenv.lsn_reset(strFile.c_str(), 0); dbenv.lsn_reset(strFile.c_str(), 0);
} }
@ -737,65 +735,96 @@ bool CTxDB::LoadBlockIndex()
// CAddrDB // CAddrDB
// //
bool CAddrDB::WriteAddrman(const CAddrMan& addrman)
CAddrDB::CAddrDB()
{ {
return Write(string("addrman"), addrman); pathAddr = GetDataDir() / "peers.dat";
} }
bool CAddrDB::LoadAddresses() bool CAddrDB::Write(const CAddrMan& addr)
{ {
if (Read(string("addrman"), addrman)) // Generate random temporary filename
{ unsigned short randv = 0;
printf("Loaded %i addresses\n", addrman.size()); RAND_bytes((unsigned char *)&randv, sizeof(randv));
std::string tmpfn = strprintf("peers.dat.%04x", randv);
// serialize addresses, checksum data up to that point, then append csum
CDataStream ssPeers(SER_DISK, CLIENT_VERSION);
ssPeers << FLATDATA(pchMessageStart);
ssPeers << addr;
uint256 hash = Hash(ssPeers.begin(), ssPeers.end());
ssPeers << hash;
// open temp output file, and associate with CAutoFile
boost::filesystem::path pathTmp = GetDataDir() / tmpfn;
FILE *file = fopen(pathTmp.string().c_str(), "wb");
CAutoFile fileout = CAutoFile(file, SER_DISK, CLIENT_VERSION);
if (!fileout)
return error("CAddrman::Write() : open failed");
// Write and commit header, data
try {
fileout << ssPeers;
}
catch (std::exception &e) {
return error("CAddrman::Write() : I/O error");
}
FileCommit(fileout);
fileout.fclose();
// replace existing peers.dat, if any, with new peers.dat.XXXX
if (!RenameOver(pathTmp, pathAddr))
return error("CAddrman::Write() : Rename-into-place failed");
return true; return true;
} }
// Read pre-0.6 addr records bool CAddrDB::Read(CAddrMan& addr)
{
// open input file, and associate with CAutoFile
FILE *file = fopen(pathAddr.string().c_str(), "rb");
CAutoFile filein = CAutoFile(file, SER_DISK, CLIENT_VERSION);
if (!filein)
return error("CAddrman::Read() : open failed");
vector<CAddress> vAddr; // use file size to size memory buffer
vector<vector<unsigned char> > vDelete; int fileSize = GetFilesize(filein);
int dataSize = fileSize - sizeof(uint256);
vector<unsigned char> vchData;
vchData.resize(dataSize);
uint256 hashIn;
// Get cursor // read data and checksum from file
Dbc* pcursor = GetCursor(); try {
if (!pcursor) filein.read((char *)&vchData[0], dataSize);
return false; filein >> hashIn;
}
catch (std::exception &e) {
return error("CAddrman::Read() 2 : I/O error or stream data corrupted");
}
filein.fclose();
loop CDataStream ssPeers(vchData, SER_DISK, CLIENT_VERSION);
{
// Read next record
CDataStream ssKey(SER_DISK, CLIENT_VERSION);
CDataStream ssValue(SER_DISK, CLIENT_VERSION);
int ret = ReadAtCursor(pcursor, ssKey, ssValue);
if (ret == DB_NOTFOUND)
break;
else if (ret != 0)
return false;
// Unserialize // verify stored checksum matches input data
string strType; uint256 hashTmp = Hash(ssPeers.begin(), ssPeers.end());
ssKey >> strType; if (hashIn != hashTmp)
if (strType == "addr") return error("CAddrman::Read() : checksum mismatch; data corrupted");
{
CAddress addr; // de-serialize address data
ssValue >> addr; unsigned char pchMsgTmp[4];
vAddr.push_back(addr); try {
ssPeers >> FLATDATA(pchMsgTmp);
ssPeers >> addr;
} }
catch (std::exception &e) {
return error("CAddrman::Read() : I/O error or stream data corrupted");
} }
pcursor->close();
addrman.Add(vAddr, CNetAddr("0.0.0.0"));
printf("Loaded %i addresses\n", addrman.size());
// Note: old records left; we ran into hangs-on-startup // finally, verify the network matches ours
// bugs for some users who (we think) were running after if (memcmp(pchMsgTmp, pchMessageStart, sizeof(pchMsgTmp)))
// an unclean shutdown. return error("CAddrman::Read() : invalid network magic number");
return true; return true;
} }
bool LoadAddresses()
{
return CAddrDB("cr+").LoadAddresses();
}

17
src/db.h

@ -296,20 +296,15 @@ public:
/** Access to the (IP) address database (addr.dat) */ /** Access to the (IP) address database (peers.dat) */
class CAddrDB : public CDB class CAddrDB
{ {
public:
CAddrDB(const char* pszMode="r+") : CDB("addr.dat", pszMode) { }
private: private:
CAddrDB(const CAddrDB&); boost::filesystem::path pathAddr;
void operator=(const CAddrDB&);
public: public:
bool WriteAddrman(const CAddrMan& addr); CAddrDB();
bool LoadAddresses(); bool Write(const CAddrMan& addr);
bool Read(CAddrMan& addr);
}; };
bool LoadAddresses();
#endif // BITCOIN_DB_H #endif // BITCOIN_DB_H

12
src/init.cpp

@ -371,9 +371,15 @@ bool AppInit2()
InitMessage(_("Loading addresses...")); InitMessage(_("Loading addresses..."));
printf("Loading addresses...\n"); printf("Loading addresses...\n");
nStart = GetTimeMillis(); nStart = GetTimeMillis();
if (!LoadAddresses())
strErrors << _("Error loading addr.dat") << "\n"; {
printf(" addresses %15"PRI64d"ms\n", GetTimeMillis() - nStart); CAddrDB adb;
if (!adb.Read(addrman))
printf("Invalid or missing peers.dat; recreating\n");
}
printf("Loaded %i addresses from peers.dat %"PRI64d"ms\n",
addrman.size(), GetTimeMillis() - nStart);
InitMessage(_("Loading block index...")); InitMessage(_("Loading block index..."));
printf("Loading block index...\n"); printf("Loading block index...\n");

7
src/net.cpp

@ -1278,8 +1278,13 @@ unsigned int pnSeed[] =
void DumpAddresses() void DumpAddresses()
{ {
int64 nStart = GetTimeMillis();
CAddrDB adb; CAddrDB adb;
adb.WriteAddrman(addrman); adb.Write(addrman);
printf("Flushed %d addresses to peers.dat %"PRI64d"ms\n",
addrman.size(), GetTimeMillis() - nStart);
} }
void ThreadDumpAddress2(void* parg) void ThreadDumpAddress2(void* parg)

1
src/net.h

@ -19,7 +19,6 @@
#include "protocol.h" #include "protocol.h"
#include "addrman.h" #include "addrman.h"
class CAddrDB;
class CRequestTracker; class CRequestTracker;
class CNode; class CNode;
class CBlockIndex; class CBlockIndex;

Loading…
Cancel
Save