1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2016 The Bitcoin Core developers
3 // Distributed under the MIT software license, see the accompanying
4 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
8 #include "chainparams.h"
14 #include "ui_interface.h"
19 #include <boost/thread.hpp>
21 static const char DB_COIN
= 'C';
22 static const char DB_COINS
= 'c';
23 static const char DB_BLOCK_FILES
= 'f';
24 static const char DB_TXINDEX
= 't';
25 static const char DB_BLOCK_INDEX
= 'b';
27 static const char DB_BEST_BLOCK
= 'B';
28 static const char DB_HEAD_BLOCKS
= 'H';
29 static const char DB_FLAG
= 'F';
30 static const char DB_REINDEX_FLAG
= 'R';
31 static const char DB_LAST_BLOCK
= 'l';
38 CoinEntry(const COutPoint
* ptr
) : outpoint(const_cast<COutPoint
*>(ptr
)), key(DB_COIN
) {}
40 template<typename Stream
>
41 void Serialize(Stream
&s
) const {
44 s
<< VARINT(outpoint
->n
);
47 template<typename Stream
>
48 void Unserialize(Stream
& s
) {
51 s
>> VARINT(outpoint
->n
);
57 CCoinsViewDB::CCoinsViewDB(size_t nCacheSize
, bool fMemory
, bool fWipe
) : db(GetDataDir() / "chainstate", nCacheSize
, fMemory
, fWipe
, true)
61 bool CCoinsViewDB::GetCoin(const COutPoint
&outpoint
, Coin
&coin
) const {
62 return db
.Read(CoinEntry(&outpoint
), coin
);
65 bool CCoinsViewDB::HaveCoin(const COutPoint
&outpoint
) const {
66 return db
.Exists(CoinEntry(&outpoint
));
69 uint256
CCoinsViewDB::GetBestBlock() const {
70 uint256 hashBestChain
;
71 if (!db
.Read(DB_BEST_BLOCK
, hashBestChain
))
76 std::vector
<uint256
> CCoinsViewDB::GetHeadBlocks() const {
77 std::vector
<uint256
> vhashHeadBlocks
;
78 if (!db
.Read(DB_HEAD_BLOCKS
, vhashHeadBlocks
)) {
79 return std::vector
<uint256
>();
81 return vhashHeadBlocks
;
84 bool CCoinsViewDB::BatchWrite(CCoinsMap
&mapCoins
, const uint256
&hashBlock
) {
88 size_t batch_size
= (size_t)GetArg("-dbbatchsize", nDefaultDbBatchSize
);
89 int crash_simulate
= GetArg("-dbcrashratio", 0);
90 assert(!hashBlock
.IsNull());
92 uint256 old_tip
= GetBestBlock();
93 if (old_tip
.IsNull()) {
94 // We may be in the middle of replaying.
95 std::vector
<uint256
> old_heads
= GetHeadBlocks();
96 if (old_heads
.size() == 2) {
97 assert(old_heads
[0] == hashBlock
);
98 old_tip
= old_heads
[1];
102 // In the first batch, mark the database as being in the middle of a
103 // transition from old_tip to hashBlock.
104 // A vector is used for future extensibility, as we may want to support
105 // interrupting after partial writes from multiple independent reorgs.
106 batch
.Erase(DB_BEST_BLOCK
);
107 batch
.Write(DB_HEAD_BLOCKS
, std::vector
<uint256
>{hashBlock
, old_tip
});
109 for (CCoinsMap::iterator it
= mapCoins
.begin(); it
!= mapCoins
.end();) {
110 if (it
->second
.flags
& CCoinsCacheEntry::DIRTY
) {
111 CoinEntry
entry(&it
->first
);
112 if (it
->second
.coin
.IsSpent())
115 batch
.Write(entry
, it
->second
.coin
);
119 CCoinsMap::iterator itOld
= it
++;
120 mapCoins
.erase(itOld
);
121 if (batch
.SizeEstimate() > batch_size
) {
122 LogPrint(BCLog::COINDB
, "Writing partial batch of %.2f MiB\n", batch
.SizeEstimate() * (1.0 / 1048576.0));
123 db
.WriteBatch(batch
);
125 if (crash_simulate
) {
126 static FastRandomContext rng
;
127 if (rng
.randrange(crash_simulate
) == 0) {
128 LogPrintf("Simulating a crash. Goodbye.\n");
135 // In the last batch, mark the database as consistent with hashBlock again.
136 batch
.Erase(DB_HEAD_BLOCKS
);
137 batch
.Write(DB_BEST_BLOCK
, hashBlock
);
139 LogPrint(BCLog::COINDB
, "Writing final batch of %.2f MiB\n", batch
.SizeEstimate() * (1.0 / 1048576.0));
140 bool ret
= db
.WriteBatch(batch
);
141 LogPrint(BCLog::COINDB
, "Committed %u changed transaction outputs (out of %u) to coin database...\n", (unsigned int)changed
, (unsigned int)count
);
145 size_t CCoinsViewDB::EstimateSize() const
147 return db
.EstimateSize(DB_COIN
, (char)(DB_COIN
+1));
150 CBlockTreeDB::CBlockTreeDB(size_t nCacheSize
, bool fMemory
, bool fWipe
) : CDBWrapper(GetDataDir() / "blocks" / "index", nCacheSize
, fMemory
, fWipe
) {
153 bool CBlockTreeDB::ReadBlockFileInfo(int nFile
, CBlockFileInfo
&info
) {
154 return Read(std::make_pair(DB_BLOCK_FILES
, nFile
), info
);
157 bool CBlockTreeDB::WriteReindexing(bool fReindexing
) {
159 return Write(DB_REINDEX_FLAG
, '1');
161 return Erase(DB_REINDEX_FLAG
);
164 bool CBlockTreeDB::ReadReindexing(bool &fReindexing
) {
165 fReindexing
= Exists(DB_REINDEX_FLAG
);
169 bool CBlockTreeDB::ReadLastBlockFile(int &nFile
) {
170 return Read(DB_LAST_BLOCK
, nFile
);
173 CCoinsViewCursor
*CCoinsViewDB::Cursor() const
175 CCoinsViewDBCursor
*i
= new CCoinsViewDBCursor(const_cast<CDBWrapper
&>(db
).NewIterator(), GetBestBlock());
176 /* It seems that there are no "const iterators" for LevelDB. Since we
177 only need read operations on it, use a const-cast to get around
179 i
->pcursor
->Seek(DB_COIN
);
180 // Cache key of first record
181 if (i
->pcursor
->Valid()) {
182 CoinEntry
entry(&i
->keyTmp
.second
);
183 i
->pcursor
->GetKey(entry
);
184 i
->keyTmp
.first
= entry
.key
;
186 i
->keyTmp
.first
= 0; // Make sure Valid() and GetKey() return false
191 bool CCoinsViewDBCursor::GetKey(COutPoint
&key
) const
194 if (keyTmp
.first
== DB_COIN
) {
201 bool CCoinsViewDBCursor::GetValue(Coin
&coin
) const
203 return pcursor
->GetValue(coin
);
206 unsigned int CCoinsViewDBCursor::GetValueSize() const
208 return pcursor
->GetValueSize();
211 bool CCoinsViewDBCursor::Valid() const
213 return keyTmp
.first
== DB_COIN
;
216 void CCoinsViewDBCursor::Next()
219 CoinEntry
entry(&keyTmp
.second
);
220 if (!pcursor
->Valid() || !pcursor
->GetKey(entry
)) {
221 keyTmp
.first
= 0; // Invalidate cached key after last record so that Valid() and GetKey() return false
223 keyTmp
.first
= entry
.key
;
227 bool CBlockTreeDB::WriteBatchSync(const std::vector
<std::pair
<int, const CBlockFileInfo
*> >& fileInfo
, int nLastFile
, const std::vector
<const CBlockIndex
*>& blockinfo
) {
228 CDBBatch
batch(*this);
229 for (std::vector
<std::pair
<int, const CBlockFileInfo
*> >::const_iterator it
=fileInfo
.begin(); it
!= fileInfo
.end(); it
++) {
230 batch
.Write(std::make_pair(DB_BLOCK_FILES
, it
->first
), *it
->second
);
232 batch
.Write(DB_LAST_BLOCK
, nLastFile
);
233 for (std::vector
<const CBlockIndex
*>::const_iterator it
=blockinfo
.begin(); it
!= blockinfo
.end(); it
++) {
234 batch
.Write(std::make_pair(DB_BLOCK_INDEX
, (*it
)->GetBlockHash()), CDiskBlockIndex(*it
));
236 return WriteBatch(batch
, true);
239 bool CBlockTreeDB::ReadTxIndex(const uint256
&txid
, CDiskTxPos
&pos
) {
240 return Read(std::make_pair(DB_TXINDEX
, txid
), pos
);
243 bool CBlockTreeDB::WriteTxIndex(const std::vector
<std::pair
<uint256
, CDiskTxPos
> >&vect
) {
244 CDBBatch
batch(*this);
245 for (std::vector
<std::pair
<uint256
,CDiskTxPos
> >::const_iterator it
=vect
.begin(); it
!=vect
.end(); it
++)
246 batch
.Write(std::make_pair(DB_TXINDEX
, it
->first
), it
->second
);
247 return WriteBatch(batch
);
250 bool CBlockTreeDB::WriteFlag(const std::string
&name
, bool fValue
) {
251 return Write(std::make_pair(DB_FLAG
, name
), fValue
? '1' : '0');
254 bool CBlockTreeDB::ReadFlag(const std::string
&name
, bool &fValue
) {
256 if (!Read(std::make_pair(DB_FLAG
, name
), ch
))
262 bool CBlockTreeDB::LoadBlockIndexGuts(const Consensus::Params
& consensusParams
, std::function
<CBlockIndex
*(const uint256
&)> insertBlockIndex
)
264 std::unique_ptr
<CDBIterator
> pcursor(NewIterator());
266 pcursor
->Seek(std::make_pair(DB_BLOCK_INDEX
, uint256()));
268 // Load mapBlockIndex
269 while (pcursor
->Valid()) {
270 boost::this_thread::interruption_point();
271 std::pair
<char, uint256
> key
;
272 if (pcursor
->GetKey(key
) && key
.first
== DB_BLOCK_INDEX
) {
273 CDiskBlockIndex diskindex
;
274 if (pcursor
->GetValue(diskindex
)) {
275 // Construct block index object
276 CBlockIndex
* pindexNew
= insertBlockIndex(diskindex
.GetBlockHash());
277 pindexNew
->pprev
= insertBlockIndex(diskindex
.hashPrev
);
278 pindexNew
->nHeight
= diskindex
.nHeight
;
279 pindexNew
->nFile
= diskindex
.nFile
;
280 pindexNew
->nDataPos
= diskindex
.nDataPos
;
281 pindexNew
->nUndoPos
= diskindex
.nUndoPos
;
282 pindexNew
->nVersion
= diskindex
.nVersion
;
283 pindexNew
->hashMerkleRoot
= diskindex
.hashMerkleRoot
;
284 pindexNew
->nTime
= diskindex
.nTime
;
285 pindexNew
->nBits
= diskindex
.nBits
;
286 pindexNew
->nNonce
= diskindex
.nNonce
;
287 pindexNew
->nStatus
= diskindex
.nStatus
;
288 pindexNew
->nTx
= diskindex
.nTx
;
290 if (!CheckProofOfWork(pindexNew
->GetBlockHash(), pindexNew
->nBits
, consensusParams
))
291 return error("%s: CheckProofOfWork failed: %s", __func__
, pindexNew
->ToString());
295 return error("%s: failed to read value", __func__
);
307 //! Legacy class to deserialize pre-pertxout database entries without reindex.
311 //! whether transaction is a coinbase
314 //! unspent transaction outputs; spent outputs are .IsNull(); spent outputs at the end of the array are dropped
315 std::vector
<CTxOut
> vout
;
317 //! at which height this transaction was included in the active block chain
320 //! empty constructor
321 CCoins() : fCoinBase(false), vout(0), nHeight(0) { }
323 template<typename Stream
>
324 void Unserialize(Stream
&s
) {
325 unsigned int nCode
= 0;
328 ::Unserialize(s
, VARINT(nVersionDummy
));
330 ::Unserialize(s
, VARINT(nCode
));
331 fCoinBase
= nCode
& 1;
332 std::vector
<bool> vAvail(2, false);
333 vAvail
[0] = (nCode
& 2) != 0;
334 vAvail
[1] = (nCode
& 4) != 0;
335 unsigned int nMaskCode
= (nCode
/ 8) + ((nCode
& 6) != 0 ? 0 : 1);
337 while (nMaskCode
> 0) {
338 unsigned char chAvail
= 0;
339 ::Unserialize(s
, chAvail
);
340 for (unsigned int p
= 0; p
< 8; p
++) {
341 bool f
= (chAvail
& (1 << p
)) != 0;
348 vout
.assign(vAvail
.size(), CTxOut());
349 for (unsigned int i
= 0; i
< vAvail
.size(); i
++) {
351 ::Unserialize(s
, REF(CTxOutCompressor(vout
[i
])));
354 ::Unserialize(s
, VARINT(nHeight
));
360 /** Upgrade the database from older formats.
362 * Currently implemented: from the per-tx utxo model (0.8..0.14.x) to per-txout.
364 bool CCoinsViewDB::Upgrade() {
365 std::unique_ptr
<CDBIterator
> pcursor(db
.NewIterator());
366 pcursor
->Seek(std::make_pair(DB_COINS
, uint256()));
367 if (!pcursor
->Valid()) {
372 LogPrintf("Upgrading utxo-set database...\n");
373 LogPrintf("[0%%]...");
374 size_t batch_size
= 1 << 24;
376 uiInterface
.SetProgressBreakAction(StartShutdown
);
378 while (pcursor
->Valid()) {
379 boost::this_thread::interruption_point();
380 if (ShutdownRequested()) {
383 std::pair
<unsigned char, uint256
> key
;
384 if (pcursor
->GetKey(key
) && key
.first
== DB_COINS
) {
385 if (count
++ % 256 == 0) {
386 uint32_t high
= 0x100 * *key
.second
.begin() + *(key
.second
.begin() + 1);
387 int percentageDone
= (int)(high
* 100.0 / 65536.0 + 0.5);
388 uiInterface
.ShowProgress(_("Upgrading UTXO database") + "\n"+ _("(press q to shutdown and continue later)") + "\n", percentageDone
);
389 if (reportDone
< percentageDone
/10) {
390 // report max. every 10% step
391 LogPrintf("[%d%%]...", percentageDone
);
392 reportDone
= percentageDone
/10;
396 if (!pcursor
->GetValue(old_coins
)) {
397 return error("%s: cannot parse CCoins record", __func__
);
399 COutPoint
outpoint(key
.second
, 0);
400 for (size_t i
= 0; i
< old_coins
.vout
.size(); ++i
) {
401 if (!old_coins
.vout
[i
].IsNull() && !old_coins
.vout
[i
].scriptPubKey
.IsUnspendable()) {
402 Coin
newcoin(std::move(old_coins
.vout
[i
]), old_coins
.nHeight
, old_coins
.fCoinBase
);
404 CoinEntry
entry(&outpoint
);
405 batch
.Write(entry
, newcoin
);
409 if (batch
.SizeEstimate() > batch_size
) {
410 db
.WriteBatch(batch
);
418 db
.WriteBatch(batch
);
419 uiInterface
.SetProgressBreakAction(std::function
<void(void)>());
420 LogPrintf("[%s].\n", ShutdownRequested() ? "CANCELLED" : "DONE");