Bitcoin Core Fuzz Coverage Report

Coverage Report

Created: 2026-03-24 13:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/root/bitcoin/src/dbwrapper.h
Line
Count
Source
1
// Copyright (c) 2012-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_DBWRAPPER_H
6
#define BITCOIN_DBWRAPPER_H
7
8
#include <attributes.h>
9
#include <serialize.h>
10
#include <span.h>
11
#include <streams.h>
12
#include <util/check.h>
13
#include <util/fs.h>
14
15
#include <cstddef>
16
#include <exception>
17
#include <memory>
18
#include <optional>
19
#include <stdexcept>
20
#include <string>
21
22
static const size_t DBWRAPPER_PREALLOC_KEY_SIZE = 64;
23
static const size_t DBWRAPPER_PREALLOC_VALUE_SIZE = 1024;
24
static const size_t DBWRAPPER_MAX_FILE_SIZE = 32 << 20; // 32 MiB
25
26
//! User-controlled performance and debug options.
27
struct DBOptions {
28
    //! Compact database on startup.
29
    bool force_compact = false;
30
};
31
32
//! Application-specific storage settings.
33
struct DBParams {
34
    //! Location in the filesystem where leveldb data will be stored.
35
    fs::path path;
36
    //! Configures various leveldb cache settings.
37
    size_t cache_bytes;
38
    //! If true, use leveldb's memory environment.
39
    bool memory_only = false;
40
    //! If true, remove all existing data.
41
    bool wipe_data = false;
42
    //! If true, store data obfuscated via simple XOR. If false, XOR with a
43
    //! zero'd byte array.
44
    bool obfuscate = false;
45
    //! Passed-through options.
46
    DBOptions options{};
47
};
48
49
class dbwrapper_error : public std::runtime_error
50
{
51
public:
52
0
    explicit dbwrapper_error(const std::string& msg) : std::runtime_error(msg) {}
53
};
54
55
class CDBWrapper;
56
57
/** These should be considered an implementation detail of the specific database.
58
 */
59
namespace dbwrapper_private {
60
61
/** Work around circular dependency, as well as for testing in dbwrapper_tests.
62
 * Database obfuscation should be considered an implementation detail of the
63
 * specific database.
64
 */
65
const Obfuscation& GetObfuscation(const CDBWrapper&);
66
}; // namespace dbwrapper_private
67
68
bool DestroyDB(const std::string& path_str);
69
70
/** Batch of changes queued to be written to a CDBWrapper */
71
class CDBBatch
72
{
73
    friend class CDBWrapper;
74
75
private:
76
    const CDBWrapper &parent;
77
78
    struct WriteBatchImpl;
79
    const std::unique_ptr<WriteBatchImpl> m_impl_batch;
80
81
    DataStream ssKey{};
82
    DataStream ssValue{};
83
84
    void WriteImpl(std::span<const std::byte> key, DataStream& ssValue);
85
    void EraseImpl(std::span<const std::byte> key);
86
87
public:
88
    /**
89
     * @param[in] _parent   CDBWrapper that this batch is to be submitted to
90
     */
91
    explicit CDBBatch(const CDBWrapper& _parent);
92
    ~CDBBatch();
93
    void Clear();
94
95
    template <typename K, typename V>
96
    void Write(const K& key, const V& value)
97
0
    {
98
0
        ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
99
0
        ssValue.reserve(DBWRAPPER_PREALLOC_VALUE_SIZE);
100
0
        ssKey << key;
101
0
        ssValue << value;
102
0
        WriteImpl(ssKey, ssValue);
103
0
        ssKey.clear();
104
0
        ssValue.clear();
105
0
    }
Unexecuted instantiation: void CDBBatch::Write<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, Obfuscation>(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Obfuscation const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, unsigned char>(unsigned char const&, unsigned char const&)
Unexecuted instantiation: void CDBBatch::Write<std::pair<unsigned char, int>, kernel::CBlockFileInfo>(std::pair<unsigned char, int> const&, kernel::CBlockFileInfo const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, int>(unsigned char const&, int const&)
Unexecuted instantiation: void CDBBatch::Write<std::pair<unsigned char, uint256>, CDiskBlockIndex>(std::pair<unsigned char, uint256> const&, CDiskBlockIndex const&)
Unexecuted instantiation: void CDBBatch::Write<std::pair<unsigned char, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, unsigned char>(std::pair<unsigned char, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > const&, unsigned char const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, std::vector<uint256, std::allocator<uint256> > >(unsigned char const&, std::vector<uint256, std::allocator<uint256> > const&)
Unexecuted instantiation: txdb.cpp:void CDBBatch::Write<(anonymous namespace)::CoinEntry, Coin>((anonymous namespace)::CoinEntry const&, Coin const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, uint256>(unsigned char const&, uint256 const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, CBlockLocator>(unsigned char const&, CBlockLocator const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, FlatFilePos>(unsigned char const&, FlatFilePos const&)
Unexecuted instantiation: blockfilterindex.cpp:void CDBBatch::Write<index_util::DBHeightKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHeightKey const&, std::pair<uint256, (anonymous namespace)::DBVal> const&)
Unexecuted instantiation: blockfilterindex.cpp:void CDBBatch::Write<index_util::DBHashKey, (anonymous namespace)::DBVal>(index_util::DBHashKey const&, (anonymous namespace)::DBVal const&)
Unexecuted instantiation: coinstatsindex.cpp:void CDBBatch::Write<index_util::DBHeightKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHeightKey const&, std::pair<uint256, (anonymous namespace)::DBVal> const&)
Unexecuted instantiation: coinstatsindex.cpp:void CDBBatch::Write<index_util::DBHashKey, (anonymous namespace)::DBVal>(index_util::DBHashKey const&, (anonymous namespace)::DBVal const&)
Unexecuted instantiation: void CDBBatch::Write<unsigned char, MuHash3072>(unsigned char const&, MuHash3072 const&)
Unexecuted instantiation: void CDBBatch::Write<std::pair<unsigned char, uint256>, CDiskTxPos>(std::pair<unsigned char, uint256> const&, CDiskTxPos const&)
Unexecuted instantiation: void CDBBatch::Write<char [12], std::pair<unsigned long, unsigned long> >(char const (&) [12], std::pair<unsigned long, unsigned long> const&)
Unexecuted instantiation: void CDBBatch::Write<DBKey, char [1]>(DBKey const&, char const (&) [1])
106
107
    template <typename K>
108
    void Erase(const K& key)
109
0
    {
110
0
        ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
111
0
        ssKey << key;
112
0
        EraseImpl(ssKey);
113
0
        ssKey.clear();
114
0
    }
Unexecuted instantiation: void CDBBatch::Erase<unsigned char>(unsigned char const&)
Unexecuted instantiation: txdb.cpp:void CDBBatch::Erase<(anonymous namespace)::CoinEntry>((anonymous namespace)::CoinEntry const&)
Unexecuted instantiation: void CDBBatch::Erase<DBKey>(DBKey const&)
115
116
    size_t ApproximateSize() const;
117
};
118
119
class CDBIterator
120
{
121
public:
122
    struct IteratorImpl;
123
124
private:
125
    const CDBWrapper &parent;
126
    const std::unique_ptr<IteratorImpl> m_impl_iter;
127
128
    void SeekImpl(std::span<const std::byte> key);
129
    std::span<const std::byte> GetKeyImpl() const;
130
    std::span<const std::byte> GetValueImpl() const;
131
132
public:
133
134
    /**
135
     * @param[in] _parent          Parent CDBWrapper instance.
136
     * @param[in] _piter           The original leveldb iterator.
137
     */
138
    CDBIterator(const CDBWrapper& _parent, std::unique_ptr<IteratorImpl> _piter);
139
    ~CDBIterator();
140
141
    bool Valid() const;
142
143
    void SeekToFirst();
144
145
0
    template<typename K> void Seek(const K& key) {
146
0
        DataStream ssKey{};
147
0
        ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
148
0
        ssKey << key;
149
0
        SeekImpl(ssKey);
150
0
    }
Unexecuted instantiation: void CDBIterator::Seek<std::pair<unsigned char, uint256> >(std::pair<unsigned char, uint256> const&)
Unexecuted instantiation: void CDBIterator::Seek<unsigned char>(unsigned char const&)
Unexecuted instantiation: void CDBIterator::Seek<index_util::DBHeightKey>(index_util::DBHeightKey const&)
Unexecuted instantiation: void CDBIterator::Seek<std::pair<unsigned char, unsigned long> >(std::pair<unsigned char, unsigned long> const&)
151
152
    void Next();
153
154
0
    template<typename K> bool GetKey(K& key) {
155
0
        try {
156
0
            DataStream ssKey{GetKeyImpl()};
157
0
            ssKey >> key;
158
0
        } catch (const std::exception&) {
159
0
            return false;
160
0
        }
161
0
        return true;
162
0
    }
Unexecuted instantiation: bool CDBIterator::GetKey<std::pair<unsigned char, uint256> >(std::pair<unsigned char, uint256>&)
Unexecuted instantiation: txdb.cpp:bool CDBIterator::GetKey<(anonymous namespace)::CoinEntry>((anonymous namespace)::CoinEntry&)
Unexecuted instantiation: bool CDBIterator::GetKey<index_util::DBHeightKey>(index_util::DBHeightKey&)
Unexecuted instantiation: bool CDBIterator::GetKey<DBKey>(DBKey&)
163
164
0
    template<typename V> bool GetValue(V& value) {
165
0
        try {
166
0
            DataStream ssValue{GetValueImpl()};
167
0
            dbwrapper_private::GetObfuscation(parent)(ssValue);
168
0
            ssValue >> value;
169
0
        } catch (const std::exception&) {
170
0
            return false;
171
0
        }
172
0
        return true;
173
0
    }
Unexecuted instantiation: bool CDBIterator::GetValue<CDiskBlockIndex>(CDiskBlockIndex&)
Unexecuted instantiation: bool CDBIterator::GetValue<Coin>(Coin&)
Unexecuted instantiation: blockfilterindex.cpp:bool CDBIterator::GetValue<std::pair<uint256, (anonymous namespace)::DBVal> >(std::pair<uint256, (anonymous namespace)::DBVal>&)
Unexecuted instantiation: coinstatsindex.cpp:bool CDBIterator::GetValue<std::pair<uint256, (anonymous namespace)::DBVal> >(std::pair<uint256, (anonymous namespace)::DBVal>&)
174
};
175
176
struct LevelDBContext;
177
178
class CDBWrapper
179
{
180
    friend const Obfuscation& dbwrapper_private::GetObfuscation(const CDBWrapper&);
181
private:
182
    //! holds all leveldb-specific fields of this class
183
    std::unique_ptr<LevelDBContext> m_db_context;
184
185
    //! the name of this database
186
    std::string m_name;
187
188
    //! optional XOR-obfuscation of the database
189
    Obfuscation m_obfuscation;
190
191
    //! obfuscation key storage key, null-prefixed to avoid collisions
192
    inline static const std::string OBFUSCATION_KEY{"\000obfuscate_key", 14}; // explicit size to avoid truncation at leading \0
193
194
    std::optional<std::string> ReadImpl(std::span<const std::byte> key) const;
195
    bool ExistsImpl(std::span<const std::byte> key) const;
196
    size_t EstimateSizeImpl(std::span<const std::byte> key1, std::span<const std::byte> key2) const;
197
0
    auto& DBContext() const LIFETIMEBOUND { return *Assert(m_db_context); }
Line
Count
Source
113
0
#define Assert(val) inline_assertion_check<true>(val, std::source_location::current(), #val)
198
199
public:
200
    CDBWrapper(const DBParams& params);
201
    ~CDBWrapper();
202
203
    CDBWrapper(const CDBWrapper&) = delete;
204
    CDBWrapper& operator=(const CDBWrapper&) = delete;
205
206
    template <typename K, typename V>
207
    bool Read(const K& key, V& value) const
208
0
    {
209
0
        DataStream ssKey{};
210
0
        ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
211
0
        ssKey << key;
212
0
        std::optional<std::string> strValue{ReadImpl(ssKey)};
213
0
        if (!strValue) {
214
0
            return false;
215
0
        }
216
0
        try {
217
0
            std::span ssValue{MakeWritableByteSpan(*strValue)};
218
0
            m_obfuscation(ssValue);
219
0
            SpanReader{ssValue} >> value;
220
0
        } catch (const std::exception&) {
221
0
            return false;
222
0
        }
223
0
        return true;
224
0
    }
Unexecuted instantiation: bool CDBWrapper::Read<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, Obfuscation>(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Obfuscation&) const
Unexecuted instantiation: bool CDBWrapper::Read<std::pair<unsigned char, int>, kernel::CBlockFileInfo>(std::pair<unsigned char, int> const&, kernel::CBlockFileInfo&) const
Unexecuted instantiation: bool CDBWrapper::Read<unsigned char, int>(unsigned char const&, int&) const
Unexecuted instantiation: bool CDBWrapper::Read<std::pair<unsigned char, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, unsigned char>(std::pair<unsigned char, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > const&, unsigned char&) const
Unexecuted instantiation: txdb.cpp:bool CDBWrapper::Read<(anonymous namespace)::CoinEntry, Coin>((anonymous namespace)::CoinEntry const&, Coin&) const
Unexecuted instantiation: bool CDBWrapper::Read<unsigned char, uint256>(unsigned char const&, uint256&) const
Unexecuted instantiation: bool CDBWrapper::Read<unsigned char, std::vector<uint256, std::allocator<uint256> > >(unsigned char const&, std::vector<uint256, std::allocator<uint256> >&) const
Unexecuted instantiation: bool CDBWrapper::Read<unsigned char, CBlockLocator>(unsigned char const&, CBlockLocator&) const
Unexecuted instantiation: blockfilterindex.cpp:bool CDBWrapper::Read<index_util::DBHashKey, (anonymous namespace)::DBVal>(index_util::DBHashKey const&, (anonymous namespace)::DBVal&) const
Unexecuted instantiation: bool CDBWrapper::Read<unsigned char, FlatFilePos>(unsigned char const&, FlatFilePos&) const
Unexecuted instantiation: blockfilterindex.cpp:bool CDBWrapper::Read<index_util::DBHeightKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHeightKey const&, std::pair<uint256, (anonymous namespace)::DBVal>&) const
Unexecuted instantiation: coinstatsindex.cpp:bool CDBWrapper::Read<index_util::DBHashKey, (anonymous namespace)::DBVal>(index_util::DBHashKey const&, (anonymous namespace)::DBVal&) const
Unexecuted instantiation: bool CDBWrapper::Read<unsigned char, MuHash3072>(unsigned char const&, MuHash3072&) const
Unexecuted instantiation: coinstatsindex.cpp:bool CDBWrapper::Read<index_util::DBHeightKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHeightKey const&, std::pair<uint256, (anonymous namespace)::DBVal>&) const
Unexecuted instantiation: coinstatsindex.cpp:bool CDBWrapper::Read<index_util::DBHashKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHashKey const&, std::pair<uint256, (anonymous namespace)::DBVal>&) const
Unexecuted instantiation: bool CDBWrapper::Read<std::pair<unsigned char, uint256>, CDiskTxPos>(std::pair<unsigned char, uint256> const&, CDiskTxPos&) const
Unexecuted instantiation: bool CDBWrapper::Read<char [12], std::pair<unsigned long, unsigned long> >(char const (&) [12], std::pair<unsigned long, unsigned long>&) const
225
226
    template <typename K, typename V>
227
    void Write(const K& key, const V& value, bool fSync = false)
228
0
    {
229
0
        CDBBatch batch(*this);
230
0
        batch.Write(key, value);
231
0
        WriteBatch(batch, fSync);
232
0
    }
Unexecuted instantiation: void CDBWrapper::Write<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, Obfuscation>(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Obfuscation const&, bool)
Unexecuted instantiation: void CDBWrapper::Write<unsigned char, unsigned char>(unsigned char const&, unsigned char const&, bool)
Unexecuted instantiation: void CDBWrapper::Write<std::pair<unsigned char, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, unsigned char>(std::pair<unsigned char, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > const&, unsigned char const&, bool)
Unexecuted instantiation: blockfilterindex.cpp:void CDBWrapper::Write<index_util::DBHeightKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHeightKey const&, std::pair<uint256, (anonymous namespace)::DBVal> const&, bool)
Unexecuted instantiation: coinstatsindex.cpp:void CDBWrapper::Write<index_util::DBHeightKey, std::pair<uint256, (anonymous namespace)::DBVal> >(index_util::DBHeightKey const&, std::pair<uint256, (anonymous namespace)::DBVal> const&, bool)
Unexecuted instantiation: void CDBWrapper::Write<char [12], std::pair<unsigned long, unsigned long> >(char const (&) [12], std::pair<unsigned long, unsigned long> const&, bool)
233
234
    template <typename K>
235
    bool Exists(const K& key) const
236
0
    {
237
0
        DataStream ssKey{};
238
0
        ssKey.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
239
0
        ssKey << key;
240
0
        return ExistsImpl(ssKey);
241
0
    }
Unexecuted instantiation: bool CDBWrapper::Exists<unsigned char>(unsigned char const&) const
Unexecuted instantiation: txdb.cpp:bool CDBWrapper::Exists<(anonymous namespace)::CoinEntry>((anonymous namespace)::CoinEntry const&) const
242
243
    template <typename K>
244
    void Erase(const K& key, bool fSync = false)
245
0
    {
246
0
        CDBBatch batch(*this);
247
0
        batch.Erase(key);
248
0
        WriteBatch(batch, fSync);
249
0
    }
250
251
    void WriteBatch(CDBBatch& batch, bool fSync = false);
252
253
    // Get an estimate of LevelDB memory usage (in bytes).
254
    size_t DynamicMemoryUsage() const;
255
256
    CDBIterator* NewIterator();
257
258
    /**
259
     * Return true if the database managed by this class contains no entries.
260
     */
261
    bool IsEmpty();
262
263
    template<typename K>
264
    size_t EstimateSize(const K& key_begin, const K& key_end) const
265
0
    {
266
0
        DataStream ssKey1{}, ssKey2{};
267
0
        ssKey1.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
268
0
        ssKey2.reserve(DBWRAPPER_PREALLOC_KEY_SIZE);
269
0
        ssKey1 << key_begin;
270
0
        ssKey2 << key_end;
271
0
        return EstimateSizeImpl(ssKey1, ssKey2);
272
0
    }
273
};
274
275
#endif // BITCOIN_DBWRAPPER_H