/root/bitcoin/src/index/base.h
Line | Count | Source |
1 | | // Copyright (c) 2017-present The Bitcoin Core 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_INDEX_BASE_H |
6 | | #define BITCOIN_INDEX_BASE_H |
7 | | |
8 | | #include <attributes.h> |
9 | | #include <dbwrapper.h> |
10 | | #include <interfaces/chain.h> |
11 | | #include <kernel/cs_main.h> |
12 | | #include <threadsafety.h> |
13 | | #include <uint256.h> |
14 | | #include <util/fs.h> |
15 | | #include <util/threadinterrupt.h> |
16 | | #include <validationinterface.h> |
17 | | |
18 | | #include <atomic> |
19 | | #include <cstddef> |
20 | | #include <memory> |
21 | | #include <optional> |
22 | | #include <string> |
23 | | #include <thread> |
24 | | |
25 | | class CBlock; |
26 | | class CBlockIndex; |
27 | | class Chainstate; |
28 | | |
29 | | struct CBlockLocator; |
30 | | struct IndexSummary { |
31 | | std::string name; |
32 | | bool synced{false}; |
33 | | int best_block_height{0}; |
34 | | uint256 best_block_hash; |
35 | | }; |
36 | | namespace interfaces { |
37 | | struct BlockRef; |
38 | | } |
39 | | namespace util { |
40 | | template <unsigned int num_params> |
41 | | struct ConstevalFormatString; |
42 | | } |
43 | | |
44 | | /** |
45 | | * Base class for indices of blockchain data. This implements |
46 | | * CValidationInterface and ensures blocks are indexed sequentially according |
47 | | * to their position in the active chain. |
48 | | * |
49 | | * In the presence of multiple chainstates (i.e. if a UTXO snapshot is loaded), |
50 | | * only the background "IBD" chainstate will be indexed to avoid building the |
51 | | * index out of order. When the background chainstate completes validation, the |
52 | | * index will be reinitialized and indexing will continue. |
53 | | */ |
54 | | class BaseIndex : public CValidationInterface |
55 | | { |
56 | | protected: |
57 | | /** |
58 | | * The database stores a block locator of the chain the database is synced to |
59 | | * so that the index can efficiently determine the point it last stopped at. |
60 | | * A locator is used instead of a simple hash of the chain tip because blocks |
61 | | * and block index entries may not be flushed to disk until after this database |
62 | | * is updated. |
63 | | */ |
64 | | class DB : public CDBWrapper |
65 | | { |
66 | | public: |
67 | | DB(const fs::path& path, size_t n_cache_size, |
68 | | bool f_memory = false, bool f_wipe = false, bool f_obfuscate = false); |
69 | | |
70 | | /// Read block locator of the chain that the index is in sync with. |
71 | | /// Note, the returned locator will be empty if no record exists. |
72 | | CBlockLocator ReadBestBlock() const; |
73 | | |
74 | | /// Write block locator of the chain that the index is in sync with. |
75 | | void WriteBestBlock(CDBBatch& batch, const CBlockLocator& locator); |
76 | | }; |
77 | | |
78 | | private: |
79 | | /// Whether the index has been initialized or not. |
80 | | std::atomic<bool> m_init{false}; |
81 | | /// Whether the index is in sync with the main chain. The flag is flipped |
82 | | /// from false to true once, after which point this starts processing |
83 | | /// ValidationInterface notifications to stay in sync. |
84 | | /// |
85 | | /// Note that this will latch to true *immediately* upon startup if |
86 | | /// `m_chainstate->m_chain` is empty, which will be the case upon startup |
87 | | /// with an empty datadir if, e.g., `-txindex=1` is specified. |
88 | | std::atomic<bool> m_synced{false}; |
89 | | |
90 | | /// The last block in the chain that the index is in sync with. |
91 | | std::atomic<const CBlockIndex*> m_best_block_index{nullptr}; |
92 | | |
93 | | std::thread m_thread_sync; |
94 | | CThreadInterrupt m_interrupt; |
95 | | |
96 | | /// Write the current index state (eg. chain block locator and subclass-specific items) to disk. |
97 | | /// |
98 | | /// Recommendations for error handling: |
99 | | /// If called on a successor of the previous committed best block in the index, the index can |
100 | | /// continue processing without risk of corruption, though the index state will need to catch up |
101 | | /// from further behind on reboot. If the new state is not a successor of the previous state (due |
102 | | /// to a chain reorganization), the index must halt until Commit succeeds or else it could end up |
103 | | /// getting corrupted. |
104 | | bool Commit(); |
105 | | |
106 | | /// Loop over disconnected blocks and call CustomRemove. |
107 | | bool Rewind(const CBlockIndex* current_tip, const CBlockIndex* new_tip); |
108 | | |
109 | | bool ProcessBlock(const CBlockIndex* pindex, const CBlock* block_data = nullptr); |
110 | | |
111 | | virtual bool AllowPrune() const = 0; |
112 | | |
113 | | template <typename... Args> |
114 | | void FatalErrorf(util::ConstevalFormatString<sizeof...(Args)> fmt, const Args&... args); |
115 | | |
116 | | protected: |
117 | | std::unique_ptr<interfaces::Chain> m_chain; |
118 | | Chainstate* m_chainstate{nullptr}; |
119 | | const std::string m_name; |
120 | | |
121 | | void BlockConnected(const kernel::ChainstateRole& role, const std::shared_ptr<const CBlock>& block, const CBlockIndex* pindex) override; |
122 | | |
123 | | void ChainStateFlushed(const kernel::ChainstateRole& role, const CBlockLocator& locator) override; |
124 | | |
125 | | /// Initialize internal state from the database and block index. |
126 | 0 | [[nodiscard]] virtual bool CustomInit(const std::optional<interfaces::BlockRef>& block) { return true; } |
127 | | |
128 | | /// Write update index entries for a newly connected block. |
129 | 0 | [[nodiscard]] virtual bool CustomAppend(const interfaces::BlockInfo& block) { return true; } |
130 | | |
131 | | /// Virtual method called internally by Commit that can be overridden to atomically |
132 | | /// commit more index state. |
133 | 0 | virtual bool CustomCommit(CDBBatch& batch) { return true; } |
134 | | |
135 | | /// Rewind index by one block during a chain reorg. |
136 | 0 | [[nodiscard]] virtual bool CustomRemove(const interfaces::BlockInfo& block) { return true; } |
137 | | |
138 | | virtual DB& GetDB() const = 0; |
139 | | |
140 | | /// Update the internal best block index as well as the prune lock. |
141 | | void SetBestBlockIndex(const CBlockIndex* block); |
142 | | |
143 | | public: |
144 | | BaseIndex(std::unique_ptr<interfaces::Chain> chain, std::string name); |
145 | | /// Destructor interrupts sync thread if running and blocks until it exits. |
146 | | virtual ~BaseIndex(); |
147 | | |
148 | | /// Get the name of the index for display in logs. |
149 | 0 | const std::string& GetName() const LIFETIMEBOUND { return m_name; } |
150 | | |
151 | | /// Return custom notification options for index. |
152 | 0 | [[nodiscard]] virtual interfaces::Chain::NotifyOptions CustomOptions() { return {}; } |
153 | | |
154 | | /// Blocks the current thread until the index is caught up to the current |
155 | | /// state of the block chain. This only blocks if the index has gotten in |
156 | | /// sync once and only needs to process blocks in the ValidationInterface |
157 | | /// queue. If the index is catching up from far behind, this method does |
158 | | /// not block and immediately returns false. |
159 | | bool BlockUntilSyncedToCurrentChain() const LOCKS_EXCLUDED(::cs_main); |
160 | | |
161 | | void Interrupt(); |
162 | | |
163 | | /// Initializes the sync state and registers the instance to the |
164 | | /// validation interface so that it stays in sync with blockchain updates. |
165 | | [[nodiscard]] bool Init(); |
166 | | |
167 | | /// Starts the initial sync process on a background thread. |
168 | | [[nodiscard]] bool StartBackgroundSync(); |
169 | | |
170 | | /// Sync the index with the block index starting from the current best block. |
171 | | /// Intended to be run in its own thread, m_thread_sync, and can be |
172 | | /// interrupted with m_interrupt. Once the index gets in sync, the m_synced |
173 | | /// flag is set and the BlockConnected ValidationInterface callback takes |
174 | | /// over and the sync thread exits. |
175 | | void Sync(); |
176 | | |
177 | | /// Stops the instance from staying in sync with blockchain updates. |
178 | | void Stop(); |
179 | | |
180 | | /// Get a summary of the index and its state. |
181 | | IndexSummary GetSummary() const; |
182 | | }; |
183 | | |
184 | | #endif // BITCOIN_INDEX_BASE_H |