Bitcoin ABC 0.30.9
P2P Digital Currency
blockstorage.h
Go to the documentation of this file.
1// Copyright (c) 2011-2021 The Bitcoin developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5#ifndef BITCOIN_NODE_BLOCKSTORAGE_H
6#define BITCOIN_NODE_BLOCKSTORAGE_H
7
8#include <cstdint>
9#include <unordered_map>
10#include <vector>
11
12#include <chain.h>
13#include <chainparams.h>
15#include <kernel/cs_main.h>
16#include <protocol.h> // For CMessageHeader::MessageStartChars
17#include <sync.h>
18#include <txdb.h>
19#include <util/fs.h>
20
22class CBlock;
23class CBlockFileInfo;
24class CBlockHeader;
25class CBlockUndo;
26class CChainParams;
27class CTxUndo;
28class Chainstate;
30struct CCheckpointData;
31class Config;
32struct FlatFilePos;
33namespace Consensus {
34struct Params;
35}
36namespace avalanche {
37class Processor;
38}
39
40namespace node {
41
43static constexpr unsigned int BLOCKFILE_CHUNK_SIZE = 0x1000000; // 16 MiB
45static const unsigned int UNDOFILE_CHUNK_SIZE = 0x100000; // 1 MiB
47static const unsigned int MAX_BLOCKFILE_SIZE = 0x8000000; // 128 MiB
48
50static constexpr size_t BLOCK_SERIALIZATION_HEADER_SIZE =
51 CMessageHeader::MESSAGE_START_SIZE + sizeof(unsigned int);
52
53extern std::atomic_bool fReindex;
54
55// Because validation code takes pointers to the map's CBlockIndex objects, if
56// we ever switch to another associative container, we need to either use a
57// container that has stable addressing (true of all std associative
58// containers), or make the key a `std::unique_ptr<CBlockIndex>`
59using BlockMap = std::unordered_map<BlockHash, CBlockIndex, BlockHasher>;
60
63 int height_first{std::numeric_limits<int>::max()};
64};
65
74 friend Chainstate;
76
77private:
78 const CChainParams &GetParams() const { return m_opts.chainparams; }
81 }
88 void FlushBlockFile(bool fFinalize = false, bool finalize_undo = false);
89 void FlushUndoFile(int block_file, bool finalize = false);
90 bool FindBlockPos(FlatFilePos &pos, unsigned int nAddSize,
91 unsigned int nHeight, uint64_t nTime, bool fKnown);
92 bool FindUndoPos(BlockValidationState &state, int nFile, FlatFilePos &pos,
93 unsigned int nAddSize);
94
97
98 FILE *OpenUndoFile(const FlatFilePos &pos, bool fReadOnly = false) const;
99
100 bool
101 WriteBlockToDisk(const CBlock &block, FlatFilePos &pos,
102 const CMessageHeader::MessageMagic &messageStart) const;
103 bool
104 UndoWriteToDisk(const CBlockUndo &blockundo, FlatFilePos &pos,
105 const BlockHash &hashBlock,
106 const CMessageHeader::MessageMagic &messageStart) const;
107
112 void FindFilesToPruneManual(std::set<int> &setFilesToPrune,
113 int nManualPruneHeight, int chain_tip_height);
114
136 void FindFilesToPrune(std::set<int> &setFilesToPrune,
137 uint64_t nPruneAfterHeight, int chain_tip_height,
138 int prune_height, bool is_ibd);
139
143
144 // Track the height of the highest block in m_last_blockfile whose undo
145 // data has been written. Block data is written to block files in download
146 // order, but is written to undo files in validation order, which is
147 // usually in order by height. To avoid wasting disk space, undo files will
148 // be trimmed whenever the corresponding block file is finalized and
149 // the height of the highest block written to the block file equals the
150 // height of the highest block written to the undo file. This is a
151 // heuristic and can sometimes preemptively trim undo files that will write
152 // more data later, and sometimes fail to trim undo files that can't have
153 // more data written later.
155
162
163 const bool m_prune_mode;
164
167
170
178 std::unordered_map<std::string, PruneLockInfo>
179 m_prune_locks GUARDED_BY(::cs_main);
180
181 const kernel::BlockManagerOpts m_opts;
182
183public:
184 using Options = kernel::BlockManagerOpts;
185
186 explicit BlockManager(Options opts)
187 : m_prune_mode{opts.prune_target > 0}, m_opts{std::move(opts)} {};
188
189 std::atomic<bool> m_importing{false};
190
192
193 std::vector<CBlockIndex *> GetAllBlockIndices()
195
202
203 std::unique_ptr<CBlockTreeDB> m_block_tree_db GUARDED_BY(::cs_main);
204
205 bool WriteBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(::cs_main);
206 bool LoadBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(::cs_main);
207
213 void ScanAndUnlinkAlreadyPrunedFiles() EXCLUSIVE_LOCKS_REQUIRED(::cs_main);
214
216 CBlockIndex *&best_header)
221
223 void PruneOneBlockFile(const int fileNumber)
225
228 const CBlockIndex *LookupBlockIndex(const BlockHash &hash) const
230
233
234 bool WriteUndoDataForBlock(const CBlockUndo &blockundo,
235 BlockValidationState &state, CBlockIndex &block)
237
243 const FlatFilePos *dbp);
244
246 [[nodiscard]] bool IsPruneMode() const { return m_prune_mode; }
247
249 [[nodiscard]] uint64_t GetPruneTarget() const {
250 return m_opts.prune_target;
251 }
252 static constexpr auto PRUNE_TARGET_MANUAL{
253 std::numeric_limits<uint64_t>::max()};
254
255 [[nodiscard]] bool LoadingBlocks() const { return m_importing || fReindex; }
256
257 [[nodiscard]] bool StopAfterBlockImport() const {
259 }
260
264 uint64_t CalculateCurrentUsage();
265
269
271 const CBlockIndex *GetFirstStoredBlock(const CBlockIndex &start_block)
273
275 bool m_have_pruned = false;
276
279 bool IsBlockPruned(const CBlockIndex *pblockindex)
281
283 void UpdatePruneLock(const std::string &name,
284 const PruneLockInfo &lock_info)
286
288 FILE *OpenBlockFile(const FlatFilePos &pos, bool fReadOnly = false) const;
289
291 fs::path GetBlockPosFilename(const FlatFilePos &pos) const;
292
296 void UnlinkPrunedFiles(const std::set<int> &setFilesToPrune) const;
297
299 bool ReadBlockFromDisk(CBlock &block, const FlatFilePos &pos) const;
300 bool ReadBlockFromDisk(CBlock &block, const CBlockIndex &index) const;
301 bool UndoReadFromDisk(CBlockUndo &blockundo,
302 const CBlockIndex &index) const;
303
305 bool ReadTxFromDisk(CMutableTransaction &tx, const FlatFilePos &pos) const;
306 bool ReadTxUndoFromDisk(CTxUndo &tx, const FlatFilePos &pos) const;
307
308 void CleanupBlockRevFiles() const;
309};
310
311void ThreadImport(ChainstateManager &chainman,
313 std::vector<fs::path> vImportFiles,
314 const fs::path &mempool_path);
315} // namespace node
316
317#endif // BITCOIN_NODE_BLOCKSTORAGE_H
const CChainParams & Params()
Return the currently selected parameters.
Definition: chainparams.cpp:19
Nodes collect new transactions into a block, hash them into a hash tree, and scan through nonce value...
Definition: block.h:23
Definition: block.h:60
The block chain is a tree shaped structure starting with the genesis block at the root,...
Definition: blockindex.h:25
Access to the block database (blocks/index/)
Definition: txdb.h:117
Undo information for a CBlock.
Definition: undo.h:73
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
Definition: chainparams.h:80
const Consensus::Params & GetConsensus() const
Definition: chainparams.h:92
Message header.
Definition: protocol.h:34
static constexpr size_t MESSAGE_START_SIZE
Definition: protocol.h:36
A mutable version of CTransaction.
Definition: transaction.h:274
Restore the UTXO in a Coin at a given COutPoint.
Definition: undo.h:62
Chainstate stores and provides an API to update our local knowledge of the current best chain.
Definition: validation.h:699
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
Definition: validation.h:1154
Definition: config.h:19
FlatFileSeq represents a sequence of numbered files storing raw data.
Definition: flatfile.h:49
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
Definition: fs.h:30
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
Definition: blockstorage.h:73
const kernel::BlockManagerOpts m_opts
Definition: blockstorage.h:181
std::set< int > m_dirty_fileinfo
Dirty block file entries.
Definition: blockstorage.h:169
bool ReadBlockFromDisk(CBlock &block, const FlatFilePos &pos) const
Functions for disk access for blocks.
bool WriteBlockToDisk(const CBlock &block, FlatFilePos &pos, const CMessageHeader::MessageMagic &messageStart) const
FlatFileSeq UndoFileSeq() const
RecursiveMutex cs_LastBlockFile
Definition: blockstorage.h:140
const CChainParams & GetParams() const
Definition: blockstorage.h:78
void FindFilesToPrune(std::set< int > &setFilesToPrune, uint64_t nPruneAfterHeight, int chain_tip_height, int prune_height, bool is_ibd)
Prune block and undo files (blk???.dat and undo???.dat) so that the disk space used is less than a us...
bool WriteUndoDataForBlock(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos SaveBlockToDisk(const CBlock &block, int nHeight, const FlatFilePos *dbp)
Store block on disk.
Definition: blockstorage.h:242
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, int chain_tip_height)
Calculate the block/rev files to delete based on height specified by user with RPC command pruneblock...
FlatFileSeq BlockFileSeq() const
void FlushUndoFile(int block_file, bool finalize=false)
const CBlockIndex *GetFirstStoredBlock(const CBlockIndex &start_block) EXCLUSIVE_LOCKS_REQUIRED(bool m_have_pruned
Find the first block that is not pruned.
Definition: blockstorage.h:275
FILE * OpenUndoFile(const FlatFilePos &pos, bool fReadOnly=false) const
Open an undo file (rev?????.dat)
static constexpr auto PRUNE_TARGET_MANUAL
Definition: blockstorage.h:252
bool StopAfterBlockImport() const
Definition: blockstorage.h:257
bool LoadBlockIndex() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Load the blocktree off disk and into memory.
void PruneOneBlockFile(const int fileNumber) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark one block file as pruned (modify associated database entries)
CBlockIndex * LookupBlockIndex(const BlockHash &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ReadTxFromDisk(CMutableTransaction &tx, const FlatFilePos &pos) const
Functions for disk access for txs.
const Consensus::Params & GetConsensus() const
Definition: blockstorage.h:79
std::unordered_map< std::string, PruneLockInfo > m_prune_locks GUARDED_BY(::cs_main)
Map from external index name to oldest block that must not be pruned.
CBlockIndex * InsertBlockIndex(const BlockHash &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Create a new block index entry for a given block hash.
bool ReadTxUndoFromDisk(CTxUndo &tx, const FlatFilePos &pos) const
bool LoadingBlocks() const
Definition: blockstorage.h:255
bool UndoReadFromDisk(CBlockUndo &blockundo, const CBlockIndex &index) const
fs::path GetBlockPosFilename(const FlatFilePos &pos) const
Translation to a filesystem path.
bool FindBlockPos(FlatFilePos &pos, unsigned int nAddSize, unsigned int nHeight, uint64_t nTime, bool fKnown)
uint64_t GetPruneTarget() const
Attempt to stay below this number of bytes of block files.
Definition: blockstorage.h:249
void UnlinkPrunedFiles(const std::set< int > &setFilesToPrune) const
Actually unlink the specified files.
bool WriteBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(bool LoadBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(void ScanAndUnlinkAlreadyPrunedFiles() EXCLUSIVE_LOCKS_REQUIRED(CBlockIndex * AddToBlockIndex(const CBlockHeader &block, CBlockIndex *&best_header) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove any pruned block & undo files that are still on disk.
const bool m_prune_mode
Definition: blockstorage.h:163
uint64_t CalculateCurrentUsage()
Calculate the amount of disk space the block & undo files currently use.
bool UndoWriteToDisk(const CBlockUndo &blockundo, FlatFilePos &pos, const BlockHash &hashBlock, const CMessageHeader::MessageMagic &messageStart) const
const CBlockIndex * GetLastCheckpoint(const CCheckpointData &data) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Returns last CBlockIndex* that is a checkpoint.
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
Definition: blockstorage.h:166
bool m_check_for_pruning
Global flag to indicate we should check to see if there are block/undo files that should be deleted.
Definition: blockstorage.h:161
bool FindUndoPos(BlockValidationState &state, int nFile, FlatFilePos &pos, unsigned int nAddSize)
friend ChainstateManager
Definition: blockstorage.h:75
bool IsPruneMode() const
Whether running in -prune mode.
Definition: blockstorage.h:246
void CleanupBlockRevFiles() const
std::atomic< bool > m_importing
Definition: blockstorage.h:189
std::vector< CBlockFileInfo > m_blockfile_info
Definition: blockstorage.h:141
CBlockFileInfo * GetBlockFileInfo(size_t n)
Get block file info entry for one block file.
bool IsBlockPruned(const CBlockIndex *pblockindex) EXCLUSIVE_LOCKS_REQUIRED(void UpdatePruneLock(const std::string &name, const PruneLockInfo &lock_info) EXCLUSIVE_LOCKS_REQUIRED(FILE OpenBlockFile)(const FlatFilePos &pos, bool fReadOnly=false) const
Check whether the block associated with this index entry is pruned or not.
Definition: blockstorage.h:288
unsigned int m_undo_height_in_last_blockfile
Definition: blockstorage.h:154
std::vector< CBlockIndex * > GetAllBlockIndices() EXCLUSIVE_LOCKS_REQUIRED(std::multimap< CBlockIndex *, CBlockIndex * > m_blocks_unlinked
All pairs A->B, where A (or one of its ancestors) misses transactions, but B has transactions.
Definition: blockstorage.h:193
BlockMap m_block_index GUARDED_BY(cs_main)
void FlushBlockFile(bool fFinalize=false, bool finalize_undo=false)
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
Definition: cs_main.cpp:7
unsigned int nHeight
Definition: init.h:28
static const unsigned int UNDOFILE_CHUNK_SIZE
The pre-allocation chunk size for rev?????.dat files (since 0.8)
Definition: blockstorage.h:45
std::unordered_map< BlockHash, CBlockIndex, BlockHasher > BlockMap
Definition: blockstorage.h:59
void ThreadImport(ChainstateManager &chainman, avalanche::Processor *const avalanche, std::vector< fs::path > vImportFiles, const fs::path &mempool_path)
static constexpr unsigned int BLOCKFILE_CHUNK_SIZE
The pre-allocation chunk size for blk?????.dat files (since 0.8)
Definition: blockstorage.h:43
static constexpr size_t BLOCK_SERIALIZATION_HEADER_SIZE
Size of header written by WriteBlockToDisk before a serialized CBlock.
Definition: blockstorage.h:50
static const unsigned int MAX_BLOCKFILE_SIZE
The maximum size of a blk?????.dat file (since 0.8)
Definition: blockstorage.h:47
std::atomic_bool fReindex
Implement std::hash so RCUPtr can be used as a key for maps or sets.
Definition: rcu.h:259
const char * name
Definition: rest.cpp:47
A BlockHash is a unqiue identifier for a block.
Definition: blockhash.h:13
Parameters that influence chain consensus.
Definition: params.h:34
An options struct for BlockManager, more ergonomically referred to as BlockManager::Options due to th...
const CChainParams & chainparams
int height_first
Height of earliest block that should be kept and not pruned.
Definition: blockstorage.h:63
#define EXCLUSIVE_LOCKS_REQUIRED(...)
Definition: threadsafety.h:56