Branch data Line data Source code
1 : : // Copyright (c) 2018-2022 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_BLOCKFILTERINDEX_H
6 : : #define BITCOIN_INDEX_BLOCKFILTERINDEX_H
7 : :
8 : : #include <attributes.h>
9 : : #include <flatfile.h>
10 : : #include <index/base.h>
11 : : #include <interfaces/chain.h>
12 : : #include <sync.h>
13 : : #include <uint256.h>
14 : : #include <util/hasher.h>
15 : :
16 : : #include <cstddef>
17 : : #include <cstdint>
18 : : #include <functional>
19 : : #include <memory>
20 : : #include <optional>
21 : : #include <unordered_map>
22 : : #include <vector>
23 : :
24 : : class BlockFilter;
25 : : class CBlockIndex;
26 : : enum class BlockFilterType : uint8_t;
27 : :
28 : : static const char* const DEFAULT_BLOCKFILTERINDEX = "0";
29 : :
30 : : /** Interval between compact filter checkpoints. See BIP 157. */
31 : : static constexpr int CFCHECKPT_INTERVAL = 1000;
32 : :
33 : : /**
34 : : * BlockFilterIndex is used to store and retrieve block filters, hashes, and headers for a range of
35 : : * blocks by height. An index is constructed for each supported filter type with its own database
36 : : * (ie. filter data for different types are stored in separate databases).
37 : : *
38 : : * This index is used to serve BIP 157 net requests.
39 : : */
40 : : class BlockFilterIndex final : public BaseIndex
41 : : {
42 : : private:
43 : : BlockFilterType m_filter_type;
44 : : std::unique_ptr<BaseIndex::DB> m_db;
45 : :
46 : : FlatFilePos m_next_filter_pos;
47 : : std::unique_ptr<FlatFileSeq> m_filter_fileseq;
48 : :
49 : : bool ReadFilterFromDisk(const FlatFilePos& pos, const uint256& hash, BlockFilter& filter) const;
50 : : size_t WriteFilterToDisk(FlatFilePos& pos, const BlockFilter& filter);
51 : :
52 : : Mutex m_cs_headers_cache;
53 : : /** cache of block hash to filter header, to avoid disk access when responding to getcfcheckpt. */
54 : : std::unordered_map<uint256, uint256, FilterHeaderHasher> m_headers_cache GUARDED_BY(m_cs_headers_cache);
55 : :
56 : : // Last computed header to avoid disk reads on every new block.
57 : : uint256 m_last_header{};
58 : :
59 : 20787 : bool AllowPrune() const override { return true; }
60 : :
61 : : bool Write(const BlockFilter& filter, uint32_t block_height, const uint256& filter_header);
62 : :
63 : : std::optional<uint256> ReadFilterHeader(int height, const uint256& expected_block_hash);
64 : :
65 : : protected:
66 : : interfaces::Chain::NotifyOptions CustomOptions() override;
67 : :
68 : : bool CustomInit(const std::optional<interfaces::BlockRef>& block) override;
69 : :
70 : : bool CustomCommit(CDBBatch& batch) override;
71 : :
72 : : bool CustomAppend(const interfaces::BlockInfo& block) override;
73 : :
74 : : bool CustomRemove(const interfaces::BlockInfo& block) override;
75 : :
76 : 315 : BaseIndex::DB& GetDB() const LIFETIMEBOUND override { return *m_db; }
77 : :
78 : : public:
79 : : /** Constructs the index, which becomes available to be queried. */
80 : : explicit BlockFilterIndex(std::unique_ptr<interfaces::Chain> chain, BlockFilterType filter_type,
81 : : size_t n_cache_size, bool f_memory = false, bool f_wipe = false);
82 : :
83 [ + + + - ]: 18575 : BlockFilterType GetFilterType() const { return m_filter_type; }
84 : :
85 : : /** Get a single filter by block. */
86 : : bool LookupFilter(const CBlockIndex* block_index, BlockFilter& filter_out) const;
87 : :
88 : : /** Get a single filter header by block. */
89 : : bool LookupFilterHeader(const CBlockIndex* block_index, uint256& header_out) EXCLUSIVE_LOCKS_REQUIRED(!m_cs_headers_cache);
90 : :
91 : : /** Get a range of filters between two heights on a chain. */
92 : : bool LookupFilterRange(int start_height, const CBlockIndex* stop_index,
93 : : std::vector<BlockFilter>& filters_out) const;
94 : :
95 : : /** Get a range of filter hashes between two heights on a chain. */
96 : : bool LookupFilterHashRange(int start_height, const CBlockIndex* stop_index,
97 : : std::vector<uint256>& hashes_out) const;
98 : : };
99 : :
100 : : /**
101 : : * Get a block filter index by type. Returns nullptr if index has not been initialized or was
102 : : * already destroyed.
103 : : */
104 : : BlockFilterIndex* GetBlockFilterIndex(BlockFilterType filter_type);
105 : :
106 : : /** Iterate over all running block filter indexes, invoking fn on each. */
107 : : void ForEachBlockFilterIndex(std::function<void (BlockFilterIndex&)> fn);
108 : :
109 : : /**
110 : : * Initialize a block filter index for the given type if one does not already exist. Returns true if
111 : : * a new index is created and false if one has already been initialized.
112 : : */
113 : : bool InitBlockFilterIndex(std::function<std::unique_ptr<interfaces::Chain>()> make_chain, BlockFilterType filter_type,
114 : : size_t n_cache_size, bool f_memory = false, bool f_wipe = false);
115 : :
116 : : /**
117 : : * Destroy the block filter index with the given type. Returns false if no such index exists. This
118 : : * just releases the allocated memory and closes the database connection, it does not delete the
119 : : * index data.
120 : : */
121 : : bool DestroyBlockFilterIndex(BlockFilterType filter_type);
122 : :
123 : : /** Destroy all open block filter indexes. */
124 : : void DestroyAllBlockFilterIndexes();
125 : :
126 : : #endif // BITCOIN_INDEX_BLOCKFILTERINDEX_H
|