// Copyright (c) 2022 The Bitcoin Core developers // Distributed under the MIT software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using common::AmountErrMsg; using kernel::MemPoolLimits; using kernel::MemPoolOptions; //! Maximum mempool size on 32-bit systems. static constexpr int MAX_32BIT_MEMPOOL_MB{500}; namespace { void ApplyArgsManOptions(const ArgsManager& argsman, MemPoolLimits& mempool_limits) { mempool_limits.ancestor_count = argsman.GetIntArg("-limitancestorcount", mempool_limits.ancestor_count); if (auto vkb = argsman.GetIntArg("-limitancestorsize")) mempool_limits.ancestor_size_vbytes = *vkb * 1'000; mempool_limits.descendant_count = argsman.GetIntArg("-limitdescendantcount", mempool_limits.descendant_count); if (auto vkb = argsman.GetIntArg("-limitdescendantsize")) mempool_limits.descendant_size_vbytes = *vkb * 1'000; } } util::Result ApplyArgsManOptions(const ArgsManager& argsman, const CChainParams& chainparams, MemPoolOptions& mempool_opts) { mempool_opts.check_ratio = argsman.GetIntArg("-checkmempool", mempool_opts.check_ratio); if (auto mb = argsman.GetIntArg("-maxmempool")) { constexpr bool is_32bit{sizeof(void*) == 4}; if (is_32bit && *mb > MAX_32BIT_MEMPOOL_MB) { return util::Error{Untranslated(strprintf("-maxmempool is set to %i but can't be over %i MB on 32-bit systems", *mb, MAX_32BIT_MEMPOOL_MB))}; } mempool_opts.max_size_bytes = *mb * 1'000'000; } if (auto hours = argsman.GetIntArg("-mempoolexpiry")) mempool_opts.expiry = std::chrono::hours{*hours}; // incremental relay fee sets the minimum feerate increase necessary for replacement in the mempool // and the amount the mempool min fee increases above the feerate of txs evicted due to mempool limiting. if (const auto arg{argsman.GetArg("-incrementalrelayfee")}) { if (std::optional inc_relay_fee = ParseMoney(*arg)) { mempool_opts.incremental_relay_feerate = CFeeRate{inc_relay_fee.value()}; } else { return util::Error{AmountErrMsg("incrementalrelayfee", *arg)}; } } static_assert(DEFAULT_MIN_RELAY_TX_FEE == DEFAULT_INCREMENTAL_RELAY_FEE); if (const auto arg{argsman.GetArg("-minrelaytxfee")}) { if (std::optional min_relay_feerate = ParseMoney(*arg)) { // High fee check is done afterward in CWallet::Create() mempool_opts.min_relay_feerate = CFeeRate{min_relay_feerate.value()}; } else { return util::Error{AmountErrMsg("minrelaytxfee", *arg)}; } } else if (mempool_opts.incremental_relay_feerate > mempool_opts.min_relay_feerate) { // Allow only setting incremental fee to control both mempool_opts.min_relay_feerate = mempool_opts.incremental_relay_feerate; LogInfo("Increasing minrelaytxfee to %s to match incrementalrelayfee", mempool_opts.min_relay_feerate.ToString()); } // Feerate used to define dust. Shouldn't be changed lightly as old // implementations may inadvertently create non-standard transactions if (const auto arg{argsman.GetArg("-dustrelayfee")}) { if (std::optional parsed = ParseMoney(*arg)) { mempool_opts.dust_relay_feerate = CFeeRate{parsed.value()}; } else { return util::Error{AmountErrMsg("dustrelayfee", *arg)}; } } mempool_opts.permit_bare_multisig = argsman.GetBoolArg("-permitbaremultisig", DEFAULT_PERMIT_BAREMULTISIG); if (argsman.GetBoolArg("-datacarrier", DEFAULT_ACCEPT_DATACARRIER)) { mempool_opts.max_datacarrier_bytes = argsman.GetIntArg("-datacarriersize", MAX_OP_RETURN_RELAY); } else { mempool_opts.max_datacarrier_bytes = std::nullopt; } mempool_opts.require_standard = !argsman.GetBoolArg("-acceptnonstdtxn", DEFAULT_ACCEPT_NON_STD_TXN); if (!chainparams.IsTestChain() && !mempool_opts.require_standard) { return util::Error{Untranslated(strprintf("acceptnonstdtxn is not currently supported for %s chain", chainparams.GetChainTypeString()))}; } mempool_opts.persist_v1_dat = argsman.GetBoolArg("-persistmempoolv1", mempool_opts.persist_v1_dat); ApplyArgsManOptions(argsman, mempool_opts.limits); return {}; }