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/wallet/test/fuzz/fees.cpp
Line
Count
Source
1
// Copyright (c) 2022-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
#include <test/fuzz/FuzzedDataProvider.h>
6
#include <test/fuzz/fuzz.h>
7
#include <test/fuzz/util.h>
8
#include <test/util/setup_common.h>
9
#include <test/util/time.h>
10
#include <test/util/txmempool.h>
11
#include <validation.h>
12
#include <wallet/coincontrol.h>
13
#include <wallet/fees.h>
14
#include <wallet/test/util.h>
15
#include <wallet/wallet.h>
16
17
namespace wallet {
18
namespace {
19
20
struct FeeEstimatorTestingSetup : public TestingSetup {
21
0
    FeeEstimatorTestingSetup(const ChainType chain_type, TestOpts opts) : TestingSetup{chain_type, opts}
22
0
    {
23
0
    }
24
25
0
    ~FeeEstimatorTestingSetup() {
26
0
        m_node.fee_estimator.reset();
27
0
    }
28
29
    void SetFeeEstimator(std::unique_ptr<CBlockPolicyEstimator> fee_estimator)
30
0
    {
31
0
        m_node.fee_estimator = std::move(fee_estimator);
32
0
    }
33
};
34
35
FeeEstimatorTestingSetup* g_setup;
36
37
class FuzzedBlockPolicyEstimator : public CBlockPolicyEstimator
38
{
39
    FuzzedDataProvider& fuzzed_data_provider;
40
41
public:
42
    FuzzedBlockPolicyEstimator(FuzzedDataProvider& provider)
43
0
        : CBlockPolicyEstimator(fs::path{}, false), fuzzed_data_provider(provider) {}
44
45
    CFeeRate estimateSmartFee(int confTarget, FeeCalculation* feeCalc, bool conservative) const override
46
0
    {
47
0
        return CFeeRate{ConsumeMoney(fuzzed_data_provider, /*max=*/1'000'000)};
48
0
    }
49
50
    unsigned int HighestTargetTracked(FeeEstimateHorizon horizon) const override
51
0
    {
52
0
        return fuzzed_data_provider.ConsumeIntegralInRange<unsigned int>(1, 1000);
53
0
    }
54
};
55
56
void initialize_setup()
57
0
{
58
0
    static const auto testing_setup = MakeNoLogFileContext<FeeEstimatorTestingSetup>();
59
0
    g_setup = testing_setup.get();
60
0
}
61
62
FUZZ_TARGET(wallet_fees, .init = initialize_setup)
63
0
{
64
0
    SeedRandomStateForTest(SeedRand::ZEROS);
65
0
    FuzzedDataProvider fuzzed_data_provider{buffer.data(), buffer.size()};
66
0
    NodeClockContext clock_ctx{ConsumeTime(fuzzed_data_provider)};
67
0
    auto& node{g_setup->m_node};
68
0
    Chainstate* chainstate = &node.chainman->ActiveChainstate();
69
70
0
    bilingual_str error;
71
0
    CTxMemPool::Options mempool_opts{
72
0
        .incremental_relay_feerate = CFeeRate{ConsumeMoney(fuzzed_data_provider, 1'000'000)},
73
0
        .min_relay_feerate = CFeeRate{ConsumeMoney(fuzzed_data_provider, 1'000'000)},
74
0
        .dust_relay_feerate = CFeeRate{ConsumeMoney(fuzzed_data_provider, 1'000'000)}
75
0
    };
76
0
    node.mempool = std::make_unique<CTxMemPool>(mempool_opts, error);
77
0
    std::unique_ptr<CBlockPolicyEstimator> fee_estimator = std::make_unique<FuzzedBlockPolicyEstimator>(fuzzed_data_provider);
78
0
    g_setup->SetFeeEstimator(std::move(fee_estimator));
79
0
    auto target_feerate{CFeeRate{ConsumeMoney(fuzzed_data_provider, /*max=*/1'000'000)}};
80
0
    if (target_feerate > node.mempool->m_opts.incremental_relay_feerate &&
81
0
        target_feerate > node.mempool->m_opts.min_relay_feerate) {
82
0
        MockMempoolMinFee(target_feerate, *node.mempool);
83
0
    }
84
0
    std::unique_ptr<CWallet> wallet_ptr{std::make_unique<CWallet>(node.chain.get(), "", CreateMockableWalletDatabase())};
85
0
    CWallet& wallet{*wallet_ptr};
86
0
    {
87
0
        LOCK(wallet.cs_wallet);
Line
Count
Source
266
0
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
0
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
0
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
0
#define PASTE(x, y) x ## y
88
0
        wallet.SetLastBlockProcessed(chainstate->m_chain.Height(), chainstate->m_chain.Tip()->GetBlockHash());
89
0
    }
90
91
0
    if (fuzzed_data_provider.ConsumeBool()) {
92
0
        wallet.m_fallback_fee = CFeeRate{ConsumeMoney(fuzzed_data_provider, /*max=*/COIN)};
93
0
    }
94
95
0
    if (fuzzed_data_provider.ConsumeBool()) {
96
0
        wallet.m_discard_rate = CFeeRate{ConsumeMoney(fuzzed_data_provider, /*max=*/COIN)};
97
0
    }
98
0
    (void)GetDiscardRate(wallet);
99
100
0
    const auto tx_bytes{fuzzed_data_provider.ConsumeIntegralInRange(0, std::numeric_limits<int32_t>::max())};
101
0
    if (fuzzed_data_provider.ConsumeBool()) {
102
0
        wallet.m_min_fee = CFeeRate{ConsumeMoney(fuzzed_data_provider, /*max=*/COIN)};
103
0
    }
104
105
0
    (void)GetRequiredFee(wallet, tx_bytes);
106
0
    (void)GetRequiredFeeRate(wallet);
107
108
0
    CCoinControl coin_control;
109
0
    if (fuzzed_data_provider.ConsumeBool()) {
110
0
        coin_control.m_feerate = CFeeRate{ConsumeMoney(fuzzed_data_provider, /*max=*/COIN)};
111
0
    }
112
0
    if (fuzzed_data_provider.ConsumeBool()) {
113
0
        coin_control.m_confirm_target = fuzzed_data_provider.ConsumeIntegralInRange<unsigned int>(0, 999'000);
114
0
    }
115
0
    if (fuzzed_data_provider.ConsumeBool()) {
116
0
        coin_control.m_fee_mode = fuzzed_data_provider.ConsumeBool() ? FeeEstimateMode::CONSERVATIVE : FeeEstimateMode::ECONOMICAL;
117
0
    }
118
119
0
    FeeCalculation fee_calculation;
120
0
    FeeCalculation* maybe_fee_calculation{fuzzed_data_provider.ConsumeBool() ? nullptr : &fee_calculation};
121
0
    (void)GetMinimumFeeRate(wallet, coin_control, maybe_fee_calculation);
122
0
    (void)GetMinimumFee(wallet, tx_bytes, coin_control, maybe_fee_calculation);
123
0
}
124
} // namespace
125
} // namespace wallet