Coverage Report

Created: 2026-04-29 19:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/bitcoin/src/wallet/fees.cpp
Line
Count
Source
1
// Copyright (c) 2009-2010 Satoshi Nakamoto
2
// Copyright (c) 2009-present The Bitcoin Core developers
3
// Distributed under the MIT software license, see the accompanying
4
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
5
6
#include <wallet/fees.h>
7
8
#include <wallet/coincontrol.h>
9
#include <wallet/wallet.h>
10
11
12
namespace wallet {
13
CAmount GetRequiredFee(const CWallet& wallet, unsigned int nTxBytes)
14
21
{
15
21
    return GetRequiredFeeRate(wallet).GetFee(static_cast<int32_t>(nTxBytes));
16
21
}
17
18
19
CAmount GetMinimumFee(const CWallet& wallet, unsigned int nTxBytes, const CCoinControl& coin_control, FeeCalculation* feeCalc)
20
0
{
21
0
    return GetMinimumFeeRate(wallet, coin_control, feeCalc).GetFee(static_cast<int32_t>(nTxBytes));
22
0
}
23
24
CFeeRate GetRequiredFeeRate(const CWallet& wallet)
25
3.53k
{
26
3.53k
    return std::max(wallet.m_min_fee, wallet.chain().relayMinFee());
27
3.53k
}
28
29
CFeeRate GetMinimumFeeRate(const CWallet& wallet, const CCoinControl& coin_control, FeeCalculation* feeCalc)
30
3.89k
{
31
    /* User control of how to calculate fee uses the following parameter precedence:
32
       1. coin_control.m_feerate
33
       2. coin_control.m_confirm_target
34
       3. m_confirm_target (user-set member variable of wallet)
35
       The first parameter that is set is used.
36
    */
37
3.89k
    CFeeRate feerate_needed;
38
3.89k
    if (coin_control.m_feerate) { // 1.
39
1.07k
        feerate_needed = *(coin_control.m_feerate);
40
        // Allow to override automatic min/max check over coin control instance
41
1.07k
        if (coin_control.fOverrideFeeRate) return feerate_needed;
42
1.07k
    }
43
2.81k
    else { // 2. or 3.
44
        // We will use smart fee estimation
45
2.81k
        unsigned int target = coin_control.m_confirm_target ? *coin_control.m_confirm_target : wallet.m_confirm_target;
46
        // By default estimates are economical iff we are signaling opt-in-RBF
47
2.81k
        bool conservative_estimate = !coin_control.m_signal_bip125_rbf.value_or(wallet.m_signal_rbf);
48
        // Allow to override the default fee estimate mode over the CoinControl instance
49
2.81k
        if (coin_control.m_fee_mode == FeeEstimateMode::CONSERVATIVE) conservative_estimate = true;
50
2.81k
        else if (coin_control.m_fee_mode == FeeEstimateMode::ECONOMICAL) conservative_estimate = false;
51
52
2.81k
        feerate_needed = wallet.chain().estimateSmartFee(target, conservative_estimate, feeCalc);
53
2.81k
        if (feerate_needed == CFeeRate(0)) {
54
            // if we don't have enough data for estimateSmartFee, then use fallback fee
55
1.48k
            feerate_needed = wallet.m_fallback_fee;
56
1.48k
            if (feeCalc) feeCalc->reason = FeeReason::FALLBACK;
57
58
            // directly return if fallback fee is disabled (feerate 0 == disabled)
59
1.48k
            if (wallet.m_fallback_fee == CFeeRate(0)) return feerate_needed;
60
1.48k
        }
61
        // Obey mempool min fee when using smart fee estimation
62
2.80k
        CFeeRate min_mempool_feerate = wallet.chain().mempoolMinFee();
63
2.80k
        if (feerate_needed < min_mempool_feerate) {
64
0
            feerate_needed = min_mempool_feerate;
65
0
            if (feeCalc) feeCalc->reason = FeeReason::MEMPOOL_MIN;
66
0
        }
67
2.80k
    }
68
69
    // prevent user from paying a fee below the required fee rate
70
3.51k
    CFeeRate required_feerate = GetRequiredFeeRate(wallet);
71
3.51k
    if (required_feerate > feerate_needed) {
72
93
        feerate_needed = required_feerate;
73
93
        if (feeCalc) feeCalc->reason = FeeReason::REQUIRED;
74
93
    }
75
3.51k
    return feerate_needed;
76
3.89k
}
77
78
CFeeRate GetDiscardRate(const CWallet& wallet)
79
3.63k
{
80
3.63k
    unsigned int highest_target = wallet.chain().estimateMaxBlocks();
81
3.63k
    CFeeRate discard_rate = wallet.chain().estimateSmartFee(highest_target, /*conservative=*/false);
82
    // Don't let discard_rate be greater than longest possible fee estimate if we get a valid fee estimate
83
3.63k
    discard_rate = (discard_rate == CFeeRate(0)) ? wallet.m_discard_rate : std::min(discard_rate, wallet.m_discard_rate);
84
    // Discard rate must be at least dust relay feerate
85
3.63k
    discard_rate = std::max(discard_rate, wallet.chain().relayDustFee());
86
3.63k
    return discard_rate;
87
3.63k
}
88
} // namespace wallet