Skip to content

Commit 9b6167f

Browse files
committed
Rework mempoolreplacement option handling
1 parent 9029cbb commit 9b6167f

File tree

8 files changed

+66
-37
lines changed

8 files changed

+66
-37
lines changed

src/init.cpp

Lines changed: 3 additions & 26 deletions
Original file line numberDiff line numberDiff line change
@@ -648,8 +648,8 @@ void SetupServerArgs(ArgsManager& argsman)
648648
"is of this size or less (default: %u)",
649649
MAX_OP_RETURN_RELAY),
650650
ArgsManager::ALLOW_ANY, OptionsCategory::NODE_RELAY);
651-
argsman.AddArg("-mempoolfullrbf", strprintf("Accept transaction replace-by-fee without requiring replaceability signaling (default: %u)", DEFAULT_MEMPOOL_FULL_RBF), ArgsManager::ALLOW_ANY, OptionsCategory::NODE_RELAY);
652-
argsman.AddArg("-mempoolreplacement", strprintf("Enable transaction replacement in the memory pool (\"fee,-optin\" = ignore opt-out flag, default: %u)", DEFAULT_ENABLE_REPLACEMENT), ArgsManager::ALLOW_ANY, OptionsCategory::NODE_RELAY);
651+
argsman.AddArg("-mempoolfullrbf", strprintf("Accept transaction replace-by-fee without requiring replaceability signaling (default: %u)", (DEFAULT_MEMPOOL_RBF_POLICY == RBFPolicy::Always)), ArgsManager::ALLOW_ANY, OptionsCategory::NODE_RELAY);
652+
argsman.AddArg("-mempoolreplacement", strprintf("Enable transaction replacement in the memory pool (\"fee,-optin\" = ignore opt-out flag, default: %u)", "fee,-optin"), ArgsManager::ALLOW_ANY, OptionsCategory::NODE_RELAY);
653653
argsman.AddArg("-permitbaremultisig", strprintf("Relay transactions creating non-P2SH multisig outputs (default: %u)", DEFAULT_PERMIT_BAREMULTISIG), ArgsManager::ALLOW_ANY,
654654
OptionsCategory::NODE_RELAY);
655655
argsman.AddArg("-minrelaytxfee=<amt>", strprintf("Fees (in %s/kvB) smaller than this are considered zero fee for relaying, mining and transaction creation (default: %s)",
@@ -894,8 +894,6 @@ bool AppInitBasicSetup(const ArgsManager& args, std::atomic<int>& exit_status)
894894
return true;
895895
}
896896

897-
static bool gReplacementHonourOptOut; // FIXME: Get rid of this
898-
899897
bool AppInitParameterInteraction(const ArgsManager& args)
900898
{
901899
const CChainParams& chainparams = Params();
@@ -1061,26 +1059,6 @@ bool AppInitParameterInteraction(const ArgsManager& args)
10611059
}
10621060
}
10631061

1064-
gEnableReplacement = args.GetBoolArg("-mempoolreplacement", DEFAULT_ENABLE_REPLACEMENT);
1065-
gReplacementHonourOptOut = !args.GetBoolArg("-mempoolfullrbf", DEFAULT_MEMPOOL_FULL_RBF);
1066-
if ((!gEnableReplacement) && args.IsArgSet("-mempoolreplacement")) {
1067-
// Minimal effort at forwards compatibility
1068-
std::string replacement_opt = args.GetArg("-mempoolreplacement", ""); // default is impossible
1069-
std::vector<std::string> replacement_modes = util::SplitString(replacement_opt, ",+");
1070-
gEnableReplacement = (std::find(replacement_modes.begin(), replacement_modes.end(), "fee") != replacement_modes.end());
1071-
if (gEnableReplacement) {
1072-
for (auto& opt : replacement_modes) {
1073-
if (opt == "optin") {
1074-
gReplacementHonourOptOut = true;
1075-
} else if (opt == "-optin") {
1076-
gReplacementHonourOptOut = false;
1077-
}
1078-
}
1079-
} else {
1080-
gReplacementHonourOptOut = true;
1081-
}
1082-
}
1083-
10841062
// Also report errors from parsing before daemonization
10851063
{
10861064
kernel::Notifications notifications{};
@@ -1554,7 +1532,6 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info)
15541532

15551533
CTxMemPool::Options mempool_opts{
15561534
.check_ratio = chainparams.DefaultConsistencyChecks() ? 1 : 0,
1557-
.full_rbf = !gReplacementHonourOptOut,
15581535
.signals = &validation_signals,
15591536
};
15601537
auto result{ApplyArgsManOptions(args, chainparams, mempool_opts)};
@@ -1572,7 +1549,7 @@ bool AppInitMain(NodeContext& node, interfaces::BlockAndHeaderTipInfo* tip_info)
15721549
LogPrintf("* Flushing caches if available system memory drops below %s MiB\n", g_low_memory_threshold / 1024 / 1024);
15731550
}
15741551

1575-
if (mempool_opts.full_rbf) {
1552+
if (mempool_opts.rbf_policy == RBFPolicy::Always) {
15761553
nLocalServices = ServiceFlags(nLocalServices | NODE_REPLACE_BY_FEE);
15771554
}
15781555

src/kernel/mempool_options.h

Lines changed: 5 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,16 @@
1515

1616
class ValidationSignals;
1717

18+
enum class RBFPolicy { Never, OptIn, Always };
19+
1820
/** Default for -maxmempool, maximum megabytes of mempool memory usage */
1921
static constexpr unsigned int DEFAULT_MAX_MEMPOOL_SIZE_MB{300};
2022
/** Default for -maxmempool when blocksonly is set */
2123
static constexpr unsigned int DEFAULT_BLOCKSONLY_MAX_MEMPOOL_SIZE_MB{5};
2224
/** Default for -mempoolexpiry, expiration time for mempool transactions in hours */
2325
static constexpr unsigned int DEFAULT_MEMPOOL_EXPIRY_HOURS{336};
24-
/** Default for -mempoolfullrbf, if the transaction replaceability signaling is ignored */
25-
static constexpr bool DEFAULT_MEMPOOL_FULL_RBF{true};
26+
/** Default for -mempoolreplacement; must update docs in init.cpp manually */
27+
static constexpr RBFPolicy DEFAULT_MEMPOOL_RBF_POLICY{RBFPolicy::Always};
2628
/** Whether to fall back to legacy V1 serialization when writing mempool.dat */
2729
static constexpr bool DEFAULT_PERSIST_V1_DAT{false};
2830
/** Default for -acceptnonstdtxn */
@@ -55,7 +57,7 @@ struct MemPoolOptions {
5557
std::optional<unsigned> max_datacarrier_bytes{DEFAULT_ACCEPT_DATACARRIER ? std::optional{MAX_OP_RETURN_RELAY} : std::nullopt};
5658
bool permit_bare_multisig{DEFAULT_PERMIT_BAREMULTISIG};
5759
bool require_standard{true};
58-
bool full_rbf{DEFAULT_MEMPOOL_FULL_RBF};
60+
RBFPolicy rbf_policy{DEFAULT_MEMPOOL_RBF_POLICY};
5961
bool persist_v1_dat{DEFAULT_PERSIST_V1_DAT};
6062
MemPoolLimits limits{};
6163

src/node/mempool_args.cpp

Lines changed: 55 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -12,10 +12,12 @@
1212
#include <consensus/amount.h>
1313
#include <kernel/chainparams.h>
1414
#include <logging.h>
15+
#include <node/interface_ui.h>
1516
#include <policy/feerate.h>
1617
#include <policy/policy.h>
1718
#include <tinyformat.h>
1819
#include <util/moneystr.h>
20+
#include <util/string.h>
1921
#include <util/translation.h>
2022

2123
#include <chrono>
@@ -92,7 +94,59 @@ util::Result<void> ApplyArgsManOptions(const ArgsManager& argsman, const CChainP
9294
return util::Error{strprintf(Untranslated("acceptnonstdtxn is not currently supported for %s chain"), chainparams.GetChainTypeString())};
9395
}
9496

95-
mempool_opts.full_rbf = argsman.GetBoolArg("-mempoolfullrbf", mempool_opts.full_rbf);
97+
if (argsman.IsArgSet("-mempoolreplacement") || argsman.IsArgSet("-mempoolfullrbf")) {
98+
// Generally, mempoolreplacement overrides mempoolfullrbf, but the latter is used to infer intent in some cases
99+
std::optional<bool> optin_flag;
100+
bool fee_flag{false};
101+
if (argsman.GetBoolArg("-mempoolreplacement", false)) {
102+
fee_flag = true;
103+
} else {
104+
for (auto& opt : util::SplitString(argsman.GetArg("-mempoolreplacement", ""), ",+")) {
105+
if (opt == "optin") {
106+
optin_flag = true;
107+
} else if (opt == "-optin") {
108+
optin_flag = false;
109+
} else if (opt == "fee") {
110+
fee_flag = true;
111+
}
112+
}
113+
}
114+
if (optin_flag.value_or(false)) {
115+
// "optin" is explicitly specified
116+
mempool_opts.rbf_policy = RBFPolicy::OptIn;
117+
} else if (argsman.GetBoolArg("-mempoolfullrbf", false)) {
118+
const bool mempoolreplacement_false{argsman.IsArgSet("-mempoolreplacement") && !(fee_flag || optin_flag.has_value())};
119+
if (mempoolreplacement_false) {
120+
// This is a contradiction, but override rather than error
121+
InitWarning(_("False mempoolreplacement option contradicts true mempoolfullrbf; disallowing all RBF"));
122+
mempool_opts.rbf_policy = RBFPolicy::Never;
123+
} else {
124+
mempool_opts.rbf_policy = RBFPolicy::Always;
125+
}
126+
} else if (!optin_flag.value_or(true)) {
127+
// "-optin" is explicitly specified
128+
mempool_opts.rbf_policy = fee_flag ? RBFPolicy::Always : RBFPolicy::Never;
129+
} else if (fee_flag) {
130+
// Just "fee" by itself
131+
if (!argsman.GetBoolArg("-mempoolfullrbf", true)) {
132+
mempool_opts.rbf_policy = RBFPolicy::OptIn;
133+
} else {
134+
// Fallback to default, unless it's been changed to Never
135+
if (mempool_opts.rbf_policy == RBFPolicy::Never) {
136+
mempool_opts.rbf_policy = RBFPolicy::Always;
137+
}
138+
}
139+
} else if (!argsman.IsArgSet("-mempoolreplacement")) {
140+
// mempoolfullrbf is always explicitly false here
141+
// Fallback to default, as long as it isn't Always
142+
if (mempool_opts.rbf_policy == RBFPolicy::Always) {
143+
mempool_opts.rbf_policy = RBFPolicy::OptIn;
144+
}
145+
} else {
146+
// mempoolreplacement is explicitly false here
147+
mempool_opts.rbf_policy = RBFPolicy::Never;
148+
}
149+
}
96150

97151
mempool_opts.persist_v1_dat = argsman.GetBoolArg("-persistmempoolv1", mempool_opts.persist_v1_dat);
98152

src/policy/policy.h

Lines changed: 0 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -36,8 +36,6 @@ static constexpr unsigned int MAX_STANDARD_TX_SIGOPS_COST{MAX_BLOCK_SIGOPS_COST/
3636
static constexpr unsigned int DEFAULT_INCREMENTAL_RELAY_FEE{1000};
3737
/** Default for -bytespersigop */
3838
static constexpr unsigned int DEFAULT_BYTES_PER_SIGOP{20};
39-
/** Default for -mempoolreplacement */
40-
static constexpr bool DEFAULT_ENABLE_REPLACEMENT{true};
4139
/** Default for -permitbaremultisig */
4240
static constexpr bool DEFAULT_PERMIT_BAREMULTISIG{true};
4341
/** The maximum number of witness stack items in a standard P2WSH script */

src/policy/settings.cpp

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,4 @@
77

88
#include <policy/policy.h>
99

10-
bool gEnableReplacement = DEFAULT_ENABLE_REPLACEMENT;
1110
unsigned int nBytesPerSigOp = DEFAULT_BYTES_PER_SIGOP;

src/policy/settings.h

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@
66
#ifndef BITCOIN_POLICY_SETTINGS_H
77
#define BITCOIN_POLICY_SETTINGS_H
88

9-
extern bool gEnableReplacement;
109
extern unsigned int nBytesPerSigOp;
1110

1211
#endif // BITCOIN_POLICY_SETTINGS_H

src/rpc/mempool.cpp

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -726,7 +726,7 @@ UniValue MempoolInfoToJSON(const CTxMemPool& pool, const std::optional<MempoolHi
726726
ret.pushKV("minrelaytxfee", ValueFromAmount(pool.m_opts.min_relay_feerate.GetFeePerK()));
727727
ret.pushKV("incrementalrelayfee", ValueFromAmount(pool.m_opts.incremental_relay_feerate.GetFeePerK()));
728728
ret.pushKV("unbroadcastcount", uint64_t{pool.GetUnbroadcastTxs().size()});
729-
ret.pushKV("fullrbf", pool.m_opts.full_rbf);
729+
ret.pushKV("fullrbf", (pool.m_opts.rbf_policy == RBFPolicy::Always));
730730

731731
if (histogram_floors) {
732732
const MempoolHistogramFeeRates& floors{histogram_floors.value()};

src/validation.cpp

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -859,9 +859,9 @@ bool MemPoolAccept::PreChecks(ATMPArgs& args, Workspace& ws)
859859
// Replaceability signaling of the original transactions may be
860860
// ignored due to node setting.
861861
bool allow_rbf;
862-
if (m_pool.m_opts.full_rbf || ignore_rejects.count("txn-mempool-conflict")) {
862+
if (m_pool.m_opts.rbf_policy == RBFPolicy::Always || ignore_rejects.count("txn-mempool-conflict")) {
863863
allow_rbf = true;
864-
} else if (!gEnableReplacement) {
864+
} else if (m_pool.m_opts.rbf_policy == RBFPolicy::Never) {
865865
allow_rbf = false;
866866
} else {
867867
allow_rbf = SignalsOptInRBF(*ptxConflicting) || ptxConflicting->version == TRUC_VERSION;

0 commit comments

Comments
 (0)