mirror of
https://github.com/Retropex/bitcoin.git
synced 2025-05-13 03:30:42 +02:00
Policy: Restore support for mining based on coin-age priority
This reverts commit 7036cf52aa
(and much more).
This commit is contained in:
parent
3b7028b4f6
commit
678978f6f6
@ -243,6 +243,7 @@ BITCOIN_CORE_H = \
|
|||||||
node/warnings.h \
|
node/warnings.h \
|
||||||
noui.h \
|
noui.h \
|
||||||
outputtype.h \
|
outputtype.h \
|
||||||
|
policy/coin_age_priority.h \
|
||||||
policy/feerate.h \
|
policy/feerate.h \
|
||||||
policy/fees.h \
|
policy/fees.h \
|
||||||
policy/fees_args.h \
|
policy/fees_args.h \
|
||||||
@ -445,6 +446,7 @@ libbitcoin_node_a_SOURCES = \
|
|||||||
node/utxo_snapshot.cpp \
|
node/utxo_snapshot.cpp \
|
||||||
node/warnings.cpp \
|
node/warnings.cpp \
|
||||||
noui.cpp \
|
noui.cpp \
|
||||||
|
policy/coin_age_priority.cpp \
|
||||||
policy/fees.cpp \
|
policy/fees.cpp \
|
||||||
policy/fees_args.cpp \
|
policy/fees_args.cpp \
|
||||||
policy/packages.cpp \
|
policy/packages.cpp \
|
||||||
|
@ -12,13 +12,15 @@
|
|||||||
static void AddTx(const CTransactionRef& tx, const CAmount& nFee, CTxMemPool& pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main, pool.cs)
|
static void AddTx(const CTransactionRef& tx, const CAmount& nFee, CTxMemPool& pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main, pool.cs)
|
||||||
{
|
{
|
||||||
int64_t nTime = 0;
|
int64_t nTime = 0;
|
||||||
|
double dPriority = 10.0;
|
||||||
unsigned int nHeight = 1;
|
unsigned int nHeight = 1;
|
||||||
uint64_t sequence = 0;
|
uint64_t sequence = 0;
|
||||||
bool spendsCoinbase = false;
|
bool spendsCoinbase = false;
|
||||||
unsigned int sigOpCost = 4;
|
unsigned int sigOpCost = 4;
|
||||||
LockPoints lp;
|
LockPoints lp;
|
||||||
pool.addUnchecked(CTxMemPoolEntry(
|
pool.addUnchecked(CTxMemPoolEntry(
|
||||||
tx, nFee, nTime, nHeight, sequence,
|
tx, nFee, nTime, dPriority, nHeight, sequence,
|
||||||
|
tx->GetValueOut(),
|
||||||
spendsCoinbase, sigOpCost, lp));
|
spendsCoinbase, sigOpCost, lp));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -16,12 +16,13 @@
|
|||||||
static void AddTx(const CTransactionRef& tx, CTxMemPool& pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main, pool.cs)
|
static void AddTx(const CTransactionRef& tx, CTxMemPool& pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main, pool.cs)
|
||||||
{
|
{
|
||||||
int64_t nTime = 0;
|
int64_t nTime = 0;
|
||||||
|
const double coinage_priority = 10.0;
|
||||||
unsigned int nHeight = 1;
|
unsigned int nHeight = 1;
|
||||||
uint64_t sequence = 0;
|
uint64_t sequence = 0;
|
||||||
bool spendsCoinbase = false;
|
bool spendsCoinbase = false;
|
||||||
unsigned int sigOpCost = 4;
|
unsigned int sigOpCost = 4;
|
||||||
LockPoints lp;
|
LockPoints lp;
|
||||||
pool.addUnchecked(CTxMemPoolEntry(tx, 1000, nTime, nHeight, sequence, spendsCoinbase, sigOpCost, lp));
|
pool.addUnchecked(CTxMemPoolEntry(tx, 1000, nTime, coinage_priority, nHeight, sequence, tx->GetValueOut(), spendsCoinbase, sigOpCost, lp));
|
||||||
}
|
}
|
||||||
|
|
||||||
struct Available {
|
struct Available {
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
static void AddTx(const CTransactionRef& tx, const CAmount& fee, CTxMemPool& pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main, pool.cs)
|
static void AddTx(const CTransactionRef& tx, const CAmount& fee, CTxMemPool& pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main, pool.cs)
|
||||||
{
|
{
|
||||||
LockPoints lp;
|
LockPoints lp;
|
||||||
pool.addUnchecked(CTxMemPoolEntry(tx, fee, /*time=*/0, /*entry_height=*/1, /*entry_sequence=*/0, /*spends_coinbase=*/false, /*sigops_cost=*/4, lp));
|
pool.addUnchecked(CTxMemPoolEntry(tx, fee, /*time=*/0, /*entry_priority=*/0, /*entry_height=*/0, /*entry_sequence=*/0, /*in_chain_input_value=*/0, /*spends_coinbase=*/false, /*sigops_cost=*/4, lp));
|
||||||
}
|
}
|
||||||
|
|
||||||
static void RpcMempool(benchmark::Bench& bench)
|
static void RpcMempool(benchmark::Bench& bench)
|
||||||
|
@ -631,7 +631,7 @@ void SetupServerArgs(ArgsManager& argsman)
|
|||||||
strprintf("Maximum tip age in seconds to consider node in initial block download (default: %u)",
|
strprintf("Maximum tip age in seconds to consider node in initial block download (default: %u)",
|
||||||
Ticks<std::chrono::seconds>(DEFAULT_MAX_TIP_AGE)),
|
Ticks<std::chrono::seconds>(DEFAULT_MAX_TIP_AGE)),
|
||||||
ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
|
ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
|
||||||
argsman.AddArg("-printpriority", strprintf("Log transaction fee rate in " + CURRENCY_UNIT + "/kvB when mining blocks (default: %u)", DEFAULT_PRINT_MODIFIED_FEE), ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
|
argsman.AddArg("-printpriority", strprintf("Log transaction priority and fee rate in " + CURRENCY_UNIT + "/kvB when mining blocks (default: %u)", DEFAULT_PRINT_MODIFIED_FEE), ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::DEBUG_TEST);
|
||||||
argsman.AddArg("-uaappend=<cmt>", "Append literal to the user agent string (should only be used for software embedding)", ArgsManager::ALLOW_ANY, OptionsCategory::CONNECTION);
|
argsman.AddArg("-uaappend=<cmt>", "Append literal to the user agent string (should only be used for software embedding)", ArgsManager::ALLOW_ANY, OptionsCategory::CONNECTION);
|
||||||
argsman.AddArg("-uacomment=<cmt>", "Append comment to the user agent string", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
|
argsman.AddArg("-uacomment=<cmt>", "Append comment to the user agent string", ArgsManager::ALLOW_ANY, OptionsCategory::DEBUG_TEST);
|
||||||
|
|
||||||
@ -660,6 +660,7 @@ void SetupServerArgs(ArgsManager& argsman)
|
|||||||
argsman.AddArg("-blockmaxsize=<n>", strprintf("Set maximum block size in bytes (default: %d)", DEFAULT_BLOCK_MAX_SIZE), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
argsman.AddArg("-blockmaxsize=<n>", strprintf("Set maximum block size in bytes (default: %d)", DEFAULT_BLOCK_MAX_SIZE), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
||||||
argsman.AddArg("-blockmaxweight=<n>", strprintf("Set maximum BIP141 block weight (default: %d)", DEFAULT_BLOCK_MAX_WEIGHT), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
argsman.AddArg("-blockmaxweight=<n>", strprintf("Set maximum BIP141 block weight (default: %d)", DEFAULT_BLOCK_MAX_WEIGHT), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
||||||
argsman.AddArg("-blockmintxfee=<amt>", strprintf("Set lowest fee rate (in %s/kvB) for transactions to be included in block creation. (default: %s)", CURRENCY_UNIT, FormatMoney(DEFAULT_BLOCK_MIN_TX_FEE)), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
argsman.AddArg("-blockmintxfee=<amt>", strprintf("Set lowest fee rate (in %s/kvB) for transactions to be included in block creation. (default: %s)", CURRENCY_UNIT, FormatMoney(DEFAULT_BLOCK_MIN_TX_FEE)), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
||||||
|
argsman.AddArg("-blockprioritysize=<n>", strprintf("Set maximum size of high-priority/low-fee transactions in bytes (default: %d)", DEFAULT_BLOCK_PRIORITY_SIZE), ArgsManager::ALLOW_ANY, OptionsCategory::BLOCK_CREATION);
|
||||||
argsman.AddArg("-blockversion=<n>", "Override block version to test forking scenarios", ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::BLOCK_CREATION);
|
argsman.AddArg("-blockversion=<n>", "Override block version to test forking scenarios", ArgsManager::ALLOW_ANY | ArgsManager::DEBUG_ONLY, OptionsCategory::BLOCK_CREATION);
|
||||||
|
|
||||||
argsman.AddArg("-rest", strprintf("Accept public REST requests (default: %u)", DEFAULT_REST_ENABLE), ArgsManager::ALLOW_ANY, OptionsCategory::RPC);
|
argsman.AddArg("-rest", strprintf("Accept public REST requests (default: %u)", DEFAULT_REST_ENABLE), ArgsManager::ALLOW_ANY, OptionsCategory::RPC);
|
||||||
|
@ -8,12 +8,14 @@
|
|||||||
#include <consensus/amount.h>
|
#include <consensus/amount.h>
|
||||||
#include <consensus/validation.h>
|
#include <consensus/validation.h>
|
||||||
#include <core_memusage.h>
|
#include <core_memusage.h>
|
||||||
|
#include <policy/coin_age_priority.h>
|
||||||
#include <policy/policy.h>
|
#include <policy/policy.h>
|
||||||
#include <policy/settings.h>
|
#include <policy/settings.h>
|
||||||
#include <primitives/transaction.h>
|
#include <primitives/transaction.h>
|
||||||
#include <util/epochguard.h>
|
#include <util/epochguard.h>
|
||||||
#include <util/overflow.h>
|
#include <util/overflow.h>
|
||||||
|
|
||||||
|
#include <cassert>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
@ -84,9 +86,14 @@ private:
|
|||||||
const size_t nUsageSize; //!< ... and total memory usage
|
const size_t nUsageSize; //!< ... and total memory usage
|
||||||
const int64_t nTime; //!< Local time when entering the mempool
|
const int64_t nTime; //!< Local time when entering the mempool
|
||||||
const uint64_t entry_sequence; //!< Sequence number used to determine whether this transaction is too recent for relay
|
const uint64_t entry_sequence; //!< Sequence number used to determine whether this transaction is too recent for relay
|
||||||
|
const double entryPriority; //!< Priority when entering the mempool
|
||||||
const unsigned int entryHeight; //!< Chain height when entering the mempool
|
const unsigned int entryHeight; //!< Chain height when entering the mempool
|
||||||
|
double cachedPriority; //!< Last calculated priority
|
||||||
|
unsigned int cachedHeight; //!< Height at which priority was last calculated
|
||||||
|
CAmount inChainInputValue; //!< Sum of all txin values that are already in blockchain
|
||||||
const bool spendsCoinbase; //!< keep track of transactions that spend a coinbase
|
const bool spendsCoinbase; //!< keep track of transactions that spend a coinbase
|
||||||
const int64_t sigOpCost; //!< Total sigop cost
|
const int64_t sigOpCost; //!< Total sigop cost
|
||||||
|
const size_t nModSize; //!< Cached modified size for priority
|
||||||
CAmount m_modified_fee; //!< Used for determining the priority of the transaction for mining in a block
|
CAmount m_modified_fee; //!< Used for determining the priority of the transaction for mining in a block
|
||||||
mutable LockPoints lockPoints; //!< Track the height and time at which tx was final
|
mutable LockPoints lockPoints; //!< Track the height and time at which tx was final
|
||||||
|
|
||||||
@ -107,7 +114,8 @@ private:
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
CTxMemPoolEntry(const CTransactionRef& tx, CAmount fee,
|
CTxMemPoolEntry(const CTransactionRef& tx, CAmount fee,
|
||||||
int64_t time, unsigned int entry_height, uint64_t entry_sequence,
|
int64_t time, double entry_priority, unsigned int entry_height, uint64_t entry_sequence,
|
||||||
|
CAmount in_chain_input_value,
|
||||||
bool spends_coinbase,
|
bool spends_coinbase,
|
||||||
int64_t sigops_cost, LockPoints lp)
|
int64_t sigops_cost, LockPoints lp)
|
||||||
: tx{tx},
|
: tx{tx},
|
||||||
@ -116,16 +124,25 @@ public:
|
|||||||
nUsageSize{RecursiveDynamicUsage(tx)},
|
nUsageSize{RecursiveDynamicUsage(tx)},
|
||||||
nTime{time},
|
nTime{time},
|
||||||
entry_sequence{entry_sequence},
|
entry_sequence{entry_sequence},
|
||||||
|
entryPriority{entry_priority},
|
||||||
entryHeight{entry_height},
|
entryHeight{entry_height},
|
||||||
|
cachedPriority{entry_priority},
|
||||||
|
// Since entries arrive *after* the tip's height, their entry priority is for the height+1
|
||||||
|
cachedHeight{entry_height + 1},
|
||||||
|
inChainInputValue{in_chain_input_value},
|
||||||
spendsCoinbase{spends_coinbase},
|
spendsCoinbase{spends_coinbase},
|
||||||
sigOpCost{sigops_cost},
|
sigOpCost{sigops_cost},
|
||||||
|
nModSize{CalculateModifiedSize(*tx, GetTxSize())},
|
||||||
m_modified_fee{nFee},
|
m_modified_fee{nFee},
|
||||||
lockPoints{lp},
|
lockPoints{lp},
|
||||||
nSizeWithDescendants{GetTxSize()},
|
nSizeWithDescendants{GetTxSize()},
|
||||||
nModFeesWithDescendants{nFee},
|
nModFeesWithDescendants{nFee},
|
||||||
nSizeWithAncestors{GetTxSize()},
|
nSizeWithAncestors{GetTxSize()},
|
||||||
nModFeesWithAncestors{nFee},
|
nModFeesWithAncestors{nFee},
|
||||||
nSigOpCostWithAncestors{sigOpCost} {}
|
nSigOpCostWithAncestors{sigOpCost} {
|
||||||
|
CAmount nValueIn = tx->GetValueOut() + nFee;
|
||||||
|
assert(inChainInputValue <= nValueIn);
|
||||||
|
}
|
||||||
|
|
||||||
CTxMemPoolEntry(ExplicitCopyTag, const CTxMemPoolEntry& entry) : CTxMemPoolEntry(entry) {}
|
CTxMemPoolEntry(ExplicitCopyTag, const CTxMemPoolEntry& entry) : CTxMemPoolEntry(entry) {}
|
||||||
CTxMemPoolEntry& operator=(const CTxMemPoolEntry&) = delete;
|
CTxMemPoolEntry& operator=(const CTxMemPoolEntry&) = delete;
|
||||||
@ -136,6 +153,17 @@ public:
|
|||||||
|
|
||||||
const CTransaction& GetTx() const { return *this->tx; }
|
const CTransaction& GetTx() const { return *this->tx; }
|
||||||
CTransactionRef GetSharedTx() const { return this->tx; }
|
CTransactionRef GetSharedTx() const { return this->tx; }
|
||||||
|
double GetStartingPriority() const {return entryPriority; }
|
||||||
|
/**
|
||||||
|
* Fast calculation of priority as update from cached value, but only valid if
|
||||||
|
* currentHeight is greater than last height it was recalculated.
|
||||||
|
*/
|
||||||
|
double GetPriority(unsigned int currentHeight) const;
|
||||||
|
/**
|
||||||
|
* Recalculate the cached priority as of currentHeight and adjust inChainInputValue by
|
||||||
|
* valueInCurrentBlock which represents input that was just added to or removed from the blockchain.
|
||||||
|
*/
|
||||||
|
void UpdateCachedPriority(unsigned int currentHeight, CAmount valueInCurrentBlock);
|
||||||
const CAmount& GetFee() const { return nFee; }
|
const CAmount& GetFee() const { return nFee; }
|
||||||
int32_t GetTxSize() const
|
int32_t GetTxSize() const
|
||||||
{
|
{
|
||||||
|
@ -726,7 +726,7 @@ public:
|
|||||||
{
|
{
|
||||||
if (!m_node.mempool) return {};
|
if (!m_node.mempool) return {};
|
||||||
LockPoints lp;
|
LockPoints lp;
|
||||||
CTxMemPoolEntry entry(tx, 0, 0, 0, 0, false, 0, lp);
|
CTxMemPoolEntry entry(tx, 0, 0, 0, 0, 0, 0, false, 0, lp);
|
||||||
LOCK(m_node.mempool->cs);
|
LOCK(m_node.mempool->cs);
|
||||||
return m_node.mempool->CheckPackageLimits({tx}, entry.GetTxSize());
|
return m_node.mempool->CheckPackageLimits({tx}, entry.GetTxSize());
|
||||||
}
|
}
|
||||||
|
@ -124,6 +124,9 @@ void BlockAssembler::resetBlock()
|
|||||||
// These counters do not include coinbase tx
|
// These counters do not include coinbase tx
|
||||||
nBlockTx = 0;
|
nBlockTx = 0;
|
||||||
nFees = 0;
|
nFees = 0;
|
||||||
|
|
||||||
|
lastFewTxs = 0;
|
||||||
|
blockFinished = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::unique_ptr<CBlockTemplate> BlockAssembler::CreateNewBlock(const CScript& scriptPubKeyIn)
|
std::unique_ptr<CBlockTemplate> BlockAssembler::CreateNewBlock(const CScript& scriptPubKeyIn)
|
||||||
@ -143,6 +146,9 @@ std::unique_ptr<CBlockTemplate> BlockAssembler::CreateNewBlock(const CScript& sc
|
|||||||
pblock->vtx.emplace_back();
|
pblock->vtx.emplace_back();
|
||||||
pblocktemplate->vTxFees.push_back(-1); // updated at end
|
pblocktemplate->vTxFees.push_back(-1); // updated at end
|
||||||
pblocktemplate->vTxSigOpsCost.push_back(-1); // updated at end
|
pblocktemplate->vTxSigOpsCost.push_back(-1); // updated at end
|
||||||
|
if (m_options.print_modified_fee) {
|
||||||
|
pblocktemplate->vTxPriorities.push_back(-1); // n/a
|
||||||
|
}
|
||||||
|
|
||||||
LOCK(::cs_main);
|
LOCK(::cs_main);
|
||||||
CBlockIndex* pindexPrev = m_chainstate.m_chain.Tip();
|
CBlockIndex* pindexPrev = m_chainstate.m_chain.Tip();
|
||||||
@ -163,6 +169,7 @@ std::unique_ptr<CBlockTemplate> BlockAssembler::CreateNewBlock(const CScript& sc
|
|||||||
int nDescendantsUpdated = 0;
|
int nDescendantsUpdated = 0;
|
||||||
if (m_mempool) {
|
if (m_mempool) {
|
||||||
LOCK(m_mempool->cs);
|
LOCK(m_mempool->cs);
|
||||||
|
addPriorityTxs(*m_mempool, nPackagesSelected);
|
||||||
addPackageTxs(*m_mempool, nPackagesSelected, nDescendantsUpdated);
|
addPackageTxs(*m_mempool, nPackagesSelected, nDescendantsUpdated);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -273,9 +280,12 @@ void BlockAssembler::AddToBlock(CTxMemPool::txiter iter)
|
|||||||
inBlock.insert(iter);
|
inBlock.insert(iter);
|
||||||
|
|
||||||
if (m_options.print_modified_fee) {
|
if (m_options.print_modified_fee) {
|
||||||
LogPrintf("fee rate %s txid %s\n",
|
double dPriority = iter->GetPriority(nHeight);
|
||||||
|
LogPrintf("priority %.1f fee rate %s txid %s\n",
|
||||||
|
dPriority,
|
||||||
CFeeRate(iter->GetModifiedFee(), iter->GetTxSize()).ToString(),
|
CFeeRate(iter->GetModifiedFee(), iter->GetTxSize()).ToString(),
|
||||||
iter->GetTx().GetHash().ToString());
|
iter->GetTx().GetHash().ToString());
|
||||||
|
pblocktemplate->vTxPriorities.push_back(dPriority);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -340,6 +350,9 @@ void BlockAssembler::addPackageTxs(const CTxMemPool& mempool, int& nPackagesSele
|
|||||||
// Keep track of entries that failed inclusion, to avoid duplicate work
|
// Keep track of entries that failed inclusion, to avoid duplicate work
|
||||||
CTxMemPool::setEntries failedTx;
|
CTxMemPool::setEntries failedTx;
|
||||||
|
|
||||||
|
// Start by adding all descendants of previously added txs to mapModifiedTx
|
||||||
|
// and modifying them for their already included ancestors
|
||||||
|
nDescendantsUpdated += UpdatePackagesForAdded(mempool, inBlock, mapModifiedTx);
|
||||||
CTxMemPool::indexed_transaction_set::index<ancestor_score>::type::iterator mi = mempool.mapTx.get<ancestor_score>().begin();
|
CTxMemPool::indexed_transaction_set::index<ancestor_score>::type::iterator mi = mempool.mapTx.get<ancestor_score>().begin();
|
||||||
CTxMemPool::txiter iter;
|
CTxMemPool::txiter iter;
|
||||||
|
|
||||||
|
@ -38,6 +38,7 @@ struct CBlockTemplate
|
|||||||
CBlock block;
|
CBlock block;
|
||||||
std::vector<CAmount> vTxFees;
|
std::vector<CAmount> vTxFees;
|
||||||
std::vector<int64_t> vTxSigOpsCost;
|
std::vector<int64_t> vTxSigOpsCost;
|
||||||
|
std::vector<double> vTxPriorities;
|
||||||
std::vector<unsigned char> vchCoinbaseCommitment;
|
std::vector<unsigned char> vchCoinbaseCommitment;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -160,6 +161,10 @@ private:
|
|||||||
const CTxMemPool* const m_mempool;
|
const CTxMemPool* const m_mempool;
|
||||||
Chainstate& m_chainstate;
|
Chainstate& m_chainstate;
|
||||||
|
|
||||||
|
// Variables used for addPriorityTxs
|
||||||
|
int lastFewTxs;
|
||||||
|
bool blockFinished;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
struct Options : BlockCreateOptions {
|
struct Options : BlockCreateOptions {
|
||||||
// Configuration parameters for the block size
|
// Configuration parameters for the block size
|
||||||
@ -190,11 +195,19 @@ private:
|
|||||||
void AddToBlock(CTxMemPool::txiter iter);
|
void AddToBlock(CTxMemPool::txiter iter);
|
||||||
|
|
||||||
// Methods for how to add transactions to a block.
|
// Methods for how to add transactions to a block.
|
||||||
|
/** Add transactions based on tx "priority" */
|
||||||
|
void addPriorityTxs(const CTxMemPool& mempool, int &nPackagesSelected) EXCLUSIVE_LOCKS_REQUIRED(mempool.cs);
|
||||||
/** Add transactions based on feerate including unconfirmed ancestors
|
/** Add transactions based on feerate including unconfirmed ancestors
|
||||||
* Increments nPackagesSelected / nDescendantsUpdated with corresponding
|
* Increments nPackagesSelected / nDescendantsUpdated with corresponding
|
||||||
* statistics from the package selection (for logging statistics). */
|
* statistics from the package selection (for logging statistics). */
|
||||||
void addPackageTxs(const CTxMemPool& mempool, int& nPackagesSelected, int& nDescendantsUpdated) EXCLUSIVE_LOCKS_REQUIRED(mempool.cs);
|
void addPackageTxs(const CTxMemPool& mempool, int& nPackagesSelected, int& nDescendantsUpdated) EXCLUSIVE_LOCKS_REQUIRED(mempool.cs);
|
||||||
|
|
||||||
|
// helper function for addPriorityTxs
|
||||||
|
/** Test if tx will still "fit" in the block */
|
||||||
|
bool TestForBlock(CTxMemPool::txiter iter);
|
||||||
|
/** Test if tx still has unconfirmed parents not yet in block */
|
||||||
|
bool isStillDependent(const CTxMemPool& mempool, CTxMemPool::txiter iter) EXCLUSIVE_LOCKS_REQUIRED(mempool.cs);
|
||||||
|
|
||||||
// helper functions for addPackageTxs()
|
// helper functions for addPackageTxs()
|
||||||
/** Remove confirmed (inBlock) entries from given set */
|
/** Remove confirmed (inBlock) entries from given set */
|
||||||
void onlyUnconfirmed(CTxMemPool::setEntries& testSet);
|
void onlyUnconfirmed(CTxMemPool::setEntries& testSet);
|
||||||
|
253
src/policy/coin_age_priority.cpp
Normal file
253
src/policy/coin_age_priority.cpp
Normal file
@ -0,0 +1,253 @@
|
|||||||
|
// Copyright (c) 2012-2017 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 <policy/coin_age_priority.h>
|
||||||
|
|
||||||
|
#include <coins.h>
|
||||||
|
#include <common/args.h>
|
||||||
|
#include <consensus/validation.h>
|
||||||
|
#include <node/miner.h>
|
||||||
|
#include <policy/policy.h>
|
||||||
|
#include <primitives/transaction.h>
|
||||||
|
#include <txmempool.h>
|
||||||
|
#include <validation.h>
|
||||||
|
|
||||||
|
using node::BlockAssembler;
|
||||||
|
|
||||||
|
unsigned int CalculateModifiedSize(const CTransaction& tx, unsigned int nTxSize)
|
||||||
|
{
|
||||||
|
// In order to avoid disincentivizing cleaning up the UTXO set we don't count
|
||||||
|
// the constant overhead for each txin and up to 110 bytes of scriptSig (which
|
||||||
|
// is enough to cover a compressed pubkey p2sh redemption) for priority.
|
||||||
|
// Providing any more cleanup incentive than making additional inputs free would
|
||||||
|
// risk encouraging people to create junk outputs to redeem later.
|
||||||
|
if (nTxSize == 0)
|
||||||
|
nTxSize = (GetTransactionWeight(tx) + WITNESS_SCALE_FACTOR - 1) / WITNESS_SCALE_FACTOR;
|
||||||
|
for (std::vector<CTxIn>::const_iterator it(tx.vin.begin()); it != tx.vin.end(); ++it)
|
||||||
|
{
|
||||||
|
unsigned int offset = 41U + std::min(110U, (unsigned int)it->scriptSig.size());
|
||||||
|
if (nTxSize > offset)
|
||||||
|
nTxSize -= offset;
|
||||||
|
}
|
||||||
|
return nTxSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
double ComputePriority(const CTransaction& tx, double dPriorityInputs, unsigned int nTxSize)
|
||||||
|
{
|
||||||
|
nTxSize = CalculateModifiedSize(tx, nTxSize);
|
||||||
|
if (nTxSize == 0) return 0.0;
|
||||||
|
|
||||||
|
return dPriorityInputs / nTxSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
double GetPriority(const CTransaction &tx, const CCoinsViewCache& view, int nHeight, CAmount &inChainInputValue)
|
||||||
|
{
|
||||||
|
inChainInputValue = 0;
|
||||||
|
if (tx.IsCoinBase())
|
||||||
|
return 0.0;
|
||||||
|
double dResult = 0.0;
|
||||||
|
for (const CTxIn& txin : tx.vin)
|
||||||
|
{
|
||||||
|
const Coin& coin = view.AccessCoin(txin.prevout);
|
||||||
|
if (coin.IsSpent()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (coin.nHeight <= nHeight) {
|
||||||
|
dResult += (double)(coin.out.nValue) * (nHeight - coin.nHeight);
|
||||||
|
inChainInputValue += coin.out.nValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ComputePriority(tx, dResult);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CTxMemPoolEntry::UpdateCachedPriority(unsigned int currentHeight, CAmount valueInCurrentBlock)
|
||||||
|
{
|
||||||
|
int heightDiff = int(currentHeight) - int(cachedHeight);
|
||||||
|
double deltaPriority = ((double)heightDiff*inChainInputValue)/nModSize;
|
||||||
|
cachedPriority += deltaPriority;
|
||||||
|
cachedHeight = currentHeight;
|
||||||
|
inChainInputValue += valueInCurrentBlock;
|
||||||
|
assert(MoneyRange(inChainInputValue));
|
||||||
|
}
|
||||||
|
|
||||||
|
struct update_priority
|
||||||
|
{
|
||||||
|
update_priority(unsigned int _height, CAmount _value) :
|
||||||
|
height(_height), value(_value)
|
||||||
|
{}
|
||||||
|
|
||||||
|
void operator() (CTxMemPoolEntry &e)
|
||||||
|
{ e.UpdateCachedPriority(height, value); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
unsigned int height;
|
||||||
|
CAmount value;
|
||||||
|
};
|
||||||
|
|
||||||
|
void CTxMemPool::UpdateDependentPriorities(const CTransaction &tx, unsigned int nBlockHeight, bool addToChain)
|
||||||
|
{
|
||||||
|
LOCK(cs);
|
||||||
|
for (unsigned int i = 0; i < tx.vout.size(); i++) {
|
||||||
|
auto it = mapNextTx.find(COutPoint(tx.GetHash(), i));
|
||||||
|
if (it == mapNextTx.end())
|
||||||
|
continue;
|
||||||
|
uint256 hash = it->second->GetHash();
|
||||||
|
txiter iter = mapTx.find(hash);
|
||||||
|
mapTx.modify(iter, update_priority(nBlockHeight, addToChain ? tx.vout[i].nValue : -tx.vout[i].nValue));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
double
|
||||||
|
CTxMemPoolEntry::GetPriority(unsigned int currentHeight) const
|
||||||
|
{
|
||||||
|
// This will only return accurate results when currentHeight >= the heights
|
||||||
|
// at which all the in-chain inputs of the tx were included in blocks.
|
||||||
|
// Typical usage of GetPriority with chainActive.Height() will ensure this.
|
||||||
|
int heightDiff = currentHeight - cachedHeight;
|
||||||
|
double deltaPriority = ((double)heightDiff*inChainInputValue)/nModSize;
|
||||||
|
double dResult = cachedPriority + deltaPriority;
|
||||||
|
if (dResult < 0) // This should only happen if it was called with an invalid height
|
||||||
|
dResult = 0;
|
||||||
|
return dResult;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We want to sort transactions by coin age priority
|
||||||
|
typedef std::pair<double, CTxMemPool::txiter> TxCoinAgePriority;
|
||||||
|
|
||||||
|
struct TxCoinAgePriorityCompare
|
||||||
|
{
|
||||||
|
bool operator()(const TxCoinAgePriority& a, const TxCoinAgePriority& b)
|
||||||
|
{
|
||||||
|
if (a.first == b.first)
|
||||||
|
return CompareTxMemPoolEntryByScore()(*(b.second), *(a.second)); //Reverse order to make sort less than
|
||||||
|
return a.first < b.first;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
bool BlockAssembler::isStillDependent(const CTxMemPool& mempool, CTxMemPool::txiter iter)
|
||||||
|
{
|
||||||
|
assert(iter != mempool.mapTx.end());
|
||||||
|
for (const auto& parent : iter->GetMemPoolParentsConst()) {
|
||||||
|
auto parent_it = mempool.mapTx.iterator_to(parent);
|
||||||
|
if (!inBlock.count(parent_it)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool BlockAssembler::TestForBlock(CTxMemPool::txiter iter)
|
||||||
|
{
|
||||||
|
uint64_t packageSize = iter->GetSizeWithAncestors();
|
||||||
|
int64_t packageSigOps = iter->GetSigOpCostWithAncestors();
|
||||||
|
if (!TestPackage(packageSize, packageSigOps)) {
|
||||||
|
// If the block is so close to full that no more txs will fit
|
||||||
|
// or if we've tried more than 50 times to fill remaining space
|
||||||
|
// then flag that the block is finished
|
||||||
|
if (nBlockWeight > m_options.nBlockMaxWeight - 400 || nBlockSigOpsCost > MAX_BLOCK_SIGOPS_COST - 8 || lastFewTxs > 50) {
|
||||||
|
blockFinished = true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
// Once we're within 4000 weight of a full block, only look at 50 more txs
|
||||||
|
// to try to fill the remaining space.
|
||||||
|
if (nBlockWeight > m_options.nBlockMaxWeight - 4000) {
|
||||||
|
++lastFewTxs;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
CTxMemPool::setEntries package;
|
||||||
|
package.insert(iter);
|
||||||
|
if (!TestPackageTransactions(package)) {
|
||||||
|
if (nBlockSize > m_options.nBlockMaxSize - 100 || lastFewTxs > 50) {
|
||||||
|
blockFinished = true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (nBlockSize > m_options.nBlockMaxSize - 1000) {
|
||||||
|
++lastFewTxs;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void BlockAssembler::addPriorityTxs(const CTxMemPool& mempool, int &nPackagesSelected)
|
||||||
|
{
|
||||||
|
AssertLockHeld(mempool.cs);
|
||||||
|
|
||||||
|
// How much of the block should be dedicated to high-priority transactions,
|
||||||
|
// included regardless of the fees they pay
|
||||||
|
uint64_t nBlockPrioritySize = gArgs.GetIntArg("-blockprioritysize", DEFAULT_BLOCK_PRIORITY_SIZE);
|
||||||
|
nBlockPrioritySize = std::min(m_options.nBlockMaxSize, nBlockPrioritySize);
|
||||||
|
|
||||||
|
if (nBlockPrioritySize == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool fSizeAccounting = fNeedSizeAccounting;
|
||||||
|
fNeedSizeAccounting = true;
|
||||||
|
|
||||||
|
// This vector will be sorted into a priority queue:
|
||||||
|
std::vector<TxCoinAgePriority> vecPriority;
|
||||||
|
TxCoinAgePriorityCompare pricomparer;
|
||||||
|
std::map<CTxMemPool::txiter, double, CompareIteratorByHash> waitPriMap;
|
||||||
|
typedef std::map<CTxMemPool::txiter, double, CompareIteratorByHash>::iterator waitPriIter;
|
||||||
|
double actualPriority = -1;
|
||||||
|
|
||||||
|
vecPriority.reserve(mempool.mapTx.size());
|
||||||
|
for (auto mi = mempool.mapTx.begin(); mi != mempool.mapTx.end(); ++mi) {
|
||||||
|
double dPriority = mi->GetPriority(nHeight);
|
||||||
|
vecPriority.emplace_back(dPriority, mi);
|
||||||
|
}
|
||||||
|
std::make_heap(vecPriority.begin(), vecPriority.end(), pricomparer);
|
||||||
|
|
||||||
|
CTxMemPool::txiter iter;
|
||||||
|
while (!vecPriority.empty() && !blockFinished) { // add a tx from priority queue to fill the blockprioritysize
|
||||||
|
iter = vecPriority.front().second;
|
||||||
|
actualPriority = vecPriority.front().first;
|
||||||
|
std::pop_heap(vecPriority.begin(), vecPriority.end(), pricomparer);
|
||||||
|
vecPriority.pop_back();
|
||||||
|
|
||||||
|
// If tx already in block, skip
|
||||||
|
if (inBlock.count(iter)) {
|
||||||
|
assert(false); // shouldn't happen for priority txs
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If tx is dependent on other mempool txs which haven't yet been included
|
||||||
|
// then put it in the waitSet
|
||||||
|
if (isStillDependent(mempool, iter)) {
|
||||||
|
waitPriMap.insert(std::make_pair(iter, actualPriority));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this tx fits in the block add it, otherwise keep looping
|
||||||
|
if (TestForBlock(iter)) {
|
||||||
|
AddToBlock(iter);
|
||||||
|
|
||||||
|
++nPackagesSelected;
|
||||||
|
|
||||||
|
// If now that this txs is added we've surpassed our desired priority size
|
||||||
|
// or have dropped below the minimum priority threshold, then we're done adding priority txs
|
||||||
|
if (nBlockSize >= nBlockPrioritySize || actualPriority <= MINIMUM_TX_PRIORITY) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This tx was successfully added, so
|
||||||
|
// add transactions that depend on this one to the priority queue to try again
|
||||||
|
for (const auto& child : iter->GetMemPoolChildrenConst())
|
||||||
|
{
|
||||||
|
auto child_it = mempool.mapTx.iterator_to(child);
|
||||||
|
waitPriIter wpiter = waitPriMap.find(child_it);
|
||||||
|
if (wpiter != waitPriMap.end()) {
|
||||||
|
vecPriority.emplace_back(wpiter->second, child_it);
|
||||||
|
std::push_heap(vecPriority.begin(), vecPriority.end(), pricomparer);
|
||||||
|
waitPriMap.erase(wpiter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fNeedSizeAccounting = fSizeAccounting;
|
||||||
|
}
|
26
src/policy/coin_age_priority.h
Normal file
26
src/policy/coin_age_priority.h
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
// Copyright (c) 2012-2017 The Bitcoin Core developers
|
||||||
|
// Distributed under the MIT software license, see the accompanying
|
||||||
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||||
|
|
||||||
|
#ifndef BITCOIN_POLICY_COIN_AGE_PRIORITY_H
|
||||||
|
#define BITCOIN_POLICY_COIN_AGE_PRIORITY_H
|
||||||
|
|
||||||
|
#include <consensus/amount.h>
|
||||||
|
|
||||||
|
class CCoinsViewCache;
|
||||||
|
class CTransaction;
|
||||||
|
|
||||||
|
// Compute modified tx size for priority calculation (optionally given tx size)
|
||||||
|
unsigned int CalculateModifiedSize(const CTransaction& tx, unsigned int nTxSize=0);
|
||||||
|
|
||||||
|
// Compute priority, given sum coin-age of inputs and (optionally) tx size
|
||||||
|
double ComputePriority(const CTransaction& tx, double dPriorityInputs, unsigned int nTxSize=0);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return priority of tx at height nHeight. Also calculate the sum of the values of the inputs
|
||||||
|
* that are already in the chain. These are the inputs that will age and increase priority as
|
||||||
|
* new blocks are added to the chain.
|
||||||
|
*/
|
||||||
|
double GetPriority(const CTransaction &tx, const CCoinsViewCache& view, int nHeight, CAmount &inChainInputValue);
|
||||||
|
|
||||||
|
#endif // BITCOIN_POLICY_COIN_AGE_PRIORITY_H
|
@ -22,6 +22,10 @@ class CScript;
|
|||||||
|
|
||||||
/** Default for -blockmaxsize, which controls the maximum size of block the mining code will create **/
|
/** Default for -blockmaxsize, which controls the maximum size of block the mining code will create **/
|
||||||
static const unsigned int DEFAULT_BLOCK_MAX_SIZE = MAX_BLOCK_SERIALIZED_SIZE;
|
static const unsigned int DEFAULT_BLOCK_MAX_SIZE = MAX_BLOCK_SERIALIZED_SIZE;
|
||||||
|
/** Default for -blockprioritysize, maximum space for zero/low-fee transactions **/
|
||||||
|
static const unsigned int DEFAULT_BLOCK_PRIORITY_SIZE = 0;
|
||||||
|
/** Minimum priority for transactions to be accepted into the priority area **/
|
||||||
|
static const double MINIMUM_TX_PRIORITY = COIN * 144 / 250;
|
||||||
/** Default for -blockmaxweight, which controls the range of block weights the mining code will create **/
|
/** Default for -blockmaxweight, which controls the range of block weights the mining code will create **/
|
||||||
static constexpr unsigned int DEFAULT_BLOCK_MAX_WEIGHT{MAX_BLOCK_WEIGHT - 4000};
|
static constexpr unsigned int DEFAULT_BLOCK_MAX_WEIGHT{MAX_BLOCK_WEIGHT - 4000};
|
||||||
/** Default for -blockmintxfee, which sets the minimum feerate for a transaction in blocks created by mining code **/
|
/** Default for -blockmintxfee, which sets the minimum feerate for a transaction in blocks created by mining code **/
|
||||||
|
@ -76,7 +76,7 @@ FUZZ_TARGET(partially_downloaded_block, .init = initialize_pdb)
|
|||||||
available.insert(i);
|
available.insert(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (add_to_mempool && !pool.exists(GenTxid::Txid(tx->GetHash()))) {
|
if (add_to_mempool && (!pool.exists(GenTxid::Txid(tx->GetHash()))) && SanityCheckForConsumeTxMemPoolEntry(*tx)) {
|
||||||
LOCK2(cs_main, pool.cs);
|
LOCK2(cs_main, pool.cs);
|
||||||
pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, *tx));
|
pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, *tx));
|
||||||
available.insert(i);
|
available.insert(i);
|
||||||
|
@ -44,6 +44,7 @@ FUZZ_TARGET(policy_estimator, .init = initialize_policy_estimator)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const CTransaction tx{*mtx};
|
const CTransaction tx{*mtx};
|
||||||
|
if (!SanityCheckForConsumeTxMemPoolEntry(tx)) return;
|
||||||
const CTxMemPoolEntry& entry = ConsumeTxMemPoolEntry(fuzzed_data_provider, tx);
|
const CTxMemPoolEntry& entry = ConsumeTxMemPoolEntry(fuzzed_data_provider, tx);
|
||||||
const auto tx_submitted_in_package = fuzzed_data_provider.ConsumeBool();
|
const auto tx_submitted_in_package = fuzzed_data_provider.ConsumeBool();
|
||||||
const auto tx_has_mempool_parents = fuzzed_data_provider.ConsumeBool();
|
const auto tx_has_mempool_parents = fuzzed_data_provider.ConsumeBool();
|
||||||
@ -68,6 +69,7 @@ FUZZ_TARGET(policy_estimator, .init = initialize_policy_estimator)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
const CTransaction tx{*mtx};
|
const CTransaction tx{*mtx};
|
||||||
|
if (!SanityCheckForConsumeTxMemPoolEntry(tx)) return;
|
||||||
mempool_entries.emplace_back(CTxMemPoolEntry::ExplicitCopy, ConsumeTxMemPoolEntry(fuzzed_data_provider, tx));
|
mempool_entries.emplace_back(CTxMemPoolEntry::ExplicitCopy, ConsumeTxMemPoolEntry(fuzzed_data_provider, tx));
|
||||||
}
|
}
|
||||||
std::vector<RemovedMempoolTransactionInfo> txs;
|
std::vector<RemovedMempoolTransactionInfo> txs;
|
||||||
|
@ -57,6 +57,8 @@ FUZZ_TARGET(rbf, .init = initialize_rbf)
|
|||||||
if (!mtx) {
|
if (!mtx) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const CTransaction tx{*mtx};
|
||||||
|
if (!SanityCheckForConsumeTxMemPoolEntry(tx)) return;
|
||||||
|
|
||||||
bilingual_str error;
|
bilingual_str error;
|
||||||
CTxMemPool pool{MemPoolOptionsForTest(g_setup->m_node), error};
|
CTxMemPool pool{MemPoolOptionsForTest(g_setup->m_node), error};
|
||||||
@ -69,13 +71,13 @@ FUZZ_TARGET(rbf, .init = initialize_rbf)
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
const CTransaction another_tx{*another_mtx};
|
const CTransaction another_tx{*another_mtx};
|
||||||
|
if (!SanityCheckForConsumeTxMemPoolEntry(another_tx)) break;
|
||||||
if (fuzzed_data_provider.ConsumeBool() && !mtx->vin.empty()) {
|
if (fuzzed_data_provider.ConsumeBool() && !mtx->vin.empty()) {
|
||||||
mtx->vin[0].prevout = COutPoint{another_tx.GetHash(), 0};
|
mtx->vin[0].prevout = COutPoint{another_tx.GetHash(), 0};
|
||||||
}
|
}
|
||||||
LOCK2(cs_main, pool.cs);
|
LOCK2(cs_main, pool.cs);
|
||||||
pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, another_tx));
|
pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, another_tx));
|
||||||
}
|
}
|
||||||
const CTransaction tx{*mtx};
|
|
||||||
if (fuzzed_data_provider.ConsumeBool()) {
|
if (fuzzed_data_provider.ConsumeBool()) {
|
||||||
LOCK2(cs_main, pool.cs);
|
LOCK2(cs_main, pool.cs);
|
||||||
pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, tx));
|
pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, tx));
|
||||||
|
@ -14,6 +14,17 @@
|
|||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
|
||||||
|
bool SanityCheckForConsumeTxMemPoolEntry(const CTransaction& tx) noexcept
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
(void)tx.GetValueOut();
|
||||||
|
return true;
|
||||||
|
} catch (const std::runtime_error&) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NOTE: Transaction must pass SanityCheckForConsumeTxMemPoolEntry first
|
||||||
CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider& fuzzed_data_provider, const CTransaction& tx) noexcept
|
CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider& fuzzed_data_provider, const CTransaction& tx) noexcept
|
||||||
{
|
{
|
||||||
// Avoid:
|
// Avoid:
|
||||||
@ -24,8 +35,9 @@ CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider& fuzzed_data_provider,
|
|||||||
assert(MoneyRange(fee));
|
assert(MoneyRange(fee));
|
||||||
const int64_t time = fuzzed_data_provider.ConsumeIntegral<int64_t>();
|
const int64_t time = fuzzed_data_provider.ConsumeIntegral<int64_t>();
|
||||||
const uint64_t entry_sequence{fuzzed_data_provider.ConsumeIntegral<uint64_t>()};
|
const uint64_t entry_sequence{fuzzed_data_provider.ConsumeIntegral<uint64_t>()};
|
||||||
const unsigned int entry_height = fuzzed_data_provider.ConsumeIntegral<unsigned int>();
|
const double coinage_priority = fuzzed_data_provider.ConsumeFloatingPoint<double>();
|
||||||
|
const unsigned int entry_height = fuzzed_data_provider.ConsumeIntegralInRange<unsigned int>(0, std::numeric_limits<unsigned int>::max() - 1);
|
||||||
const bool spends_coinbase = fuzzed_data_provider.ConsumeBool();
|
const bool spends_coinbase = fuzzed_data_provider.ConsumeBool();
|
||||||
const unsigned int sig_op_cost = fuzzed_data_provider.ConsumeIntegralInRange<unsigned int>(0, MAX_BLOCK_SIGOPS_COST);
|
const unsigned int sig_op_cost = fuzzed_data_provider.ConsumeIntegralInRange<unsigned int>(0, MAX_BLOCK_SIGOPS_COST);
|
||||||
return CTxMemPoolEntry{MakeTransactionRef(tx), fee, time, entry_height, entry_sequence, spends_coinbase, sig_op_cost, {}};
|
return CTxMemPoolEntry{MakeTransactionRef(tx), fee, time, coinage_priority, entry_height, entry_sequence, tx.GetValueOut(), spends_coinbase, sig_op_cost, {}};
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,7 @@ public:
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
[[nodiscard]] bool SanityCheckForConsumeTxMemPoolEntry(const CTransaction& tx) noexcept;
|
||||||
[[nodiscard]] CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider& fuzzed_data_provider, const CTransaction& tx) noexcept;
|
[[nodiscard]] CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider& fuzzed_data_provider, const CTransaction& tx) noexcept;
|
||||||
|
|
||||||
#endif // BITCOIN_TEST_FUZZ_UTIL_MEMPOOL_H
|
#endif // BITCOIN_TEST_FUZZ_UTIL_MEMPOOL_H
|
||||||
|
@ -558,7 +558,8 @@ std::vector<CTransactionRef> TestChain100Setup::PopulateMempool(FastRandomContex
|
|||||||
LOCK2(cs_main, m_node.mempool->cs);
|
LOCK2(cs_main, m_node.mempool->cs);
|
||||||
LockPoints lp;
|
LockPoints lp;
|
||||||
m_node.mempool->addUnchecked(CTxMemPoolEntry(ptx, /*fee=*/(total_in - num_outputs * amount_per_output),
|
m_node.mempool->addUnchecked(CTxMemPoolEntry(ptx, /*fee=*/(total_in - num_outputs * amount_per_output),
|
||||||
/*time=*/0, /*entry_height=*/1, /*entry_sequence=*/0,
|
/*time=*/0, /*entry_priority=*/0, /*entry_height=*/1, /*entry_sequence=*/0,
|
||||||
|
/*in_chain_input_value=*/0,
|
||||||
/*spends_coinbase=*/false, /*sigops_cost=*/4, lp));
|
/*spends_coinbase=*/false, /*sigops_cost=*/4, lp));
|
||||||
}
|
}
|
||||||
--num_transactions;
|
--num_transactions;
|
||||||
@ -588,7 +589,8 @@ void TestChain100Setup::MockMempoolMinFee(const CFeeRate& target_feerate)
|
|||||||
const auto tx_fee = target_feerate.GetFee(GetVirtualTransactionSize(*tx)) -
|
const auto tx_fee = target_feerate.GetFee(GetVirtualTransactionSize(*tx)) -
|
||||||
m_node.mempool->m_opts.incremental_relay_feerate.GetFee(GetVirtualTransactionSize(*tx));
|
m_node.mempool->m_opts.incremental_relay_feerate.GetFee(GetVirtualTransactionSize(*tx));
|
||||||
m_node.mempool->addUnchecked(CTxMemPoolEntry(tx, /*fee=*/tx_fee,
|
m_node.mempool->addUnchecked(CTxMemPoolEntry(tx, /*fee=*/tx_fee,
|
||||||
/*time=*/0, /*entry_height=*/1, /*entry_sequence=*/0,
|
/*time=*/0, /*entry_priority=*/0, /*entry_height=*/1, /*entry_sequence=*/0,
|
||||||
|
/*in_chain_input_value=*/0,
|
||||||
/*spends_coinbase=*/true, /*sigops_cost=*/1, lp));
|
/*spends_coinbase=*/true, /*sigops_cost=*/1, lp));
|
||||||
m_node.mempool->TrimToSize(0);
|
m_node.mempool->TrimToSize(0);
|
||||||
assert(m_node.mempool->GetMinFee() == target_feerate);
|
assert(m_node.mempool->GetMinFee() == target_feerate);
|
||||||
|
@ -37,7 +37,9 @@ CTxMemPoolEntry TestMemPoolEntryHelper::FromTx(const CMutableTransaction& tx) co
|
|||||||
|
|
||||||
CTxMemPoolEntry TestMemPoolEntryHelper::FromTx(const CTransactionRef& tx) const
|
CTxMemPoolEntry TestMemPoolEntryHelper::FromTx(const CTransactionRef& tx) const
|
||||||
{
|
{
|
||||||
return CTxMemPoolEntry{tx, nFee, TicksSinceEpoch<std::chrono::seconds>(time), nHeight, m_sequence, spendsCoinbase, sigOpCost, lp};
|
const double dPriority = 0;
|
||||||
|
const CAmount inChainValue = 0;
|
||||||
|
return CTxMemPoolEntry{tx, nFee, TicksSinceEpoch<std::chrono::seconds>(time), dPriority, nHeight, m_sequence, inChainValue, spendsCoinbase, sigOpCost, lp};
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<std::string> CheckPackageMempoolAcceptResult(const Package& txns,
|
std::optional<std::string> CheckPackageMempoolAcceptResult(const Package& txns,
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
#include <consensus/tx_verify.h>
|
#include <consensus/tx_verify.h>
|
||||||
#include <consensus/validation.h>
|
#include <consensus/validation.h>
|
||||||
#include <logging.h>
|
#include <logging.h>
|
||||||
|
#include <policy/coin_age_priority.h>
|
||||||
#include <policy/policy.h>
|
#include <policy/policy.h>
|
||||||
#include <policy/settings.h>
|
#include <policy/settings.h>
|
||||||
#include <random.h>
|
#include <random.h>
|
||||||
@ -636,6 +637,7 @@ void CTxMemPool::removeForBlock(const std::vector<CTransactionRef>& vtx, unsigne
|
|||||||
txs_removed_for_block.reserve(vtx.size());
|
txs_removed_for_block.reserve(vtx.size());
|
||||||
for (const auto& tx : vtx)
|
for (const auto& tx : vtx)
|
||||||
{
|
{
|
||||||
|
UpdateDependentPriorities(*tx, nBlockHeight, true);
|
||||||
txiter it = mapTx.find(tx->GetHash());
|
txiter it = mapTx.find(tx->GetHash());
|
||||||
if (it != mapTx.end()) {
|
if (it != mapTx.end()) {
|
||||||
setEntries stage;
|
setEntries stage;
|
||||||
@ -672,6 +674,13 @@ void CTxMemPool::check(const CCoinsViewCache& active_coins_tip, int64_t spendhei
|
|||||||
|
|
||||||
for (const auto& it : GetSortedDepthAndScore()) {
|
for (const auto& it : GetSortedDepthAndScore()) {
|
||||||
checkTotal += it->GetTxSize();
|
checkTotal += it->GetTxSize();
|
||||||
|
CAmount dummyValue;
|
||||||
|
double freshPriority = GetPriority(it->GetTx(), active_coins_tip, spendheight, dummyValue);
|
||||||
|
double cachePriority = it->GetPriority(spendheight);
|
||||||
|
double priDiff = cachePriority > freshPriority ? cachePriority - freshPriority : freshPriority - cachePriority;
|
||||||
|
// Verify that the difference between the on the fly calculation and a fresh calculation
|
||||||
|
// is small enough to be a result of double imprecision.
|
||||||
|
assert(priDiff < .0001 * freshPriority + 1);
|
||||||
check_total_fee += it->GetFee();
|
check_total_fee += it->GetFee();
|
||||||
innerUsage += it->DynamicMemoryUsage();
|
innerUsage += it->DynamicMemoryUsage();
|
||||||
const CTransaction& tx = it->GetTx();
|
const CTransaction& tx = it->GetTx();
|
||||||
|
@ -484,6 +484,12 @@ public:
|
|||||||
* the tx is not dependent on other mempool transactions to be included in a block.
|
* the tx is not dependent on other mempool transactions to be included in a block.
|
||||||
*/
|
*/
|
||||||
bool HasNoInputsOf(const CTransaction& tx) const EXCLUSIVE_LOCKS_REQUIRED(cs);
|
bool HasNoInputsOf(const CTransaction& tx) const EXCLUSIVE_LOCKS_REQUIRED(cs);
|
||||||
|
/**
|
||||||
|
* Update all transactions in the mempool which depend on tx to recalculate their priority
|
||||||
|
* and adjust the input value that will age to reflect that the inputs from this transaction have
|
||||||
|
* either just been added to the chain or just been removed.
|
||||||
|
*/
|
||||||
|
void UpdateDependentPriorities(const CTransaction &tx, unsigned int nBlockHeight, bool addToChain);
|
||||||
|
|
||||||
/** Affect CreateNewBlock prioritisation of transactions */
|
/** Affect CreateNewBlock prioritisation of transactions */
|
||||||
void PrioritiseTransaction(const uint256& hash, const CAmount& nFeeDelta);
|
void PrioritiseTransaction(const uint256& hash, const CAmount& nFeeDelta);
|
||||||
|
@ -32,6 +32,7 @@
|
|||||||
#include <logging/timer.h>
|
#include <logging/timer.h>
|
||||||
#include <node/blockstorage.h>
|
#include <node/blockstorage.h>
|
||||||
#include <node/utxo_snapshot.h>
|
#include <node/utxo_snapshot.h>
|
||||||
|
#include <policy/coin_age_priority.h>
|
||||||
#include <policy/policy.h>
|
#include <policy/policy.h>
|
||||||
#include <policy/rbf.h>
|
#include <policy/rbf.h>
|
||||||
#include <policy/settings.h>
|
#include <policy/settings.h>
|
||||||
@ -934,6 +935,10 @@ bool MemPoolAccept::PreChecks(ATMPArgs& args, Workspace& ws)
|
|||||||
ws.m_modified_fees = ws.m_base_fees;
|
ws.m_modified_fees = ws.m_base_fees;
|
||||||
m_pool.ApplyDelta(hash, ws.m_modified_fees);
|
m_pool.ApplyDelta(hash, ws.m_modified_fees);
|
||||||
|
|
||||||
|
CAmount inChainInputValue;
|
||||||
|
// Since entries arrive *after* the tip's height, their priority is for the height+1
|
||||||
|
double dPriority = GetPriority(tx, m_view, m_active_chainstate.m_chain.Height() + 1, inChainInputValue);
|
||||||
|
|
||||||
// Keep track of transactions that spend a coinbase, which we re-scan
|
// Keep track of transactions that spend a coinbase, which we re-scan
|
||||||
// during reorgs to ensure COINBASE_MATURITY is still met.
|
// during reorgs to ensure COINBASE_MATURITY is still met.
|
||||||
bool fSpendsCoinbase = false;
|
bool fSpendsCoinbase = false;
|
||||||
@ -948,7 +953,8 @@ bool MemPoolAccept::PreChecks(ATMPArgs& args, Workspace& ws)
|
|||||||
// Set entry_sequence to 0 when rejectmsg_zero_mempool_entry_seq is used; this allows txs from a block
|
// Set entry_sequence to 0 when rejectmsg_zero_mempool_entry_seq is used; this allows txs from a block
|
||||||
// reorg to be marked earlier than any child txs that were already in the mempool.
|
// reorg to be marked earlier than any child txs that were already in the mempool.
|
||||||
const uint64_t entry_sequence = args.m_ignore_rejects.count(rejectmsg_zero_mempool_entry_seq) ? 0 : m_pool.GetSequence();
|
const uint64_t entry_sequence = args.m_ignore_rejects.count(rejectmsg_zero_mempool_entry_seq) ? 0 : m_pool.GetSequence();
|
||||||
entry.reset(new CTxMemPoolEntry(ptx, ws.m_base_fees, nAcceptTime, m_active_chainstate.m_chain.Height(), entry_sequence,
|
entry.reset(new CTxMemPoolEntry(ptx, ws.m_base_fees, nAcceptTime, dPriority, m_active_chainstate.m_chain.Height(), entry_sequence,
|
||||||
|
inChainInputValue,
|
||||||
fSpendsCoinbase, nSigOpsCost, lock_points.value()));
|
fSpendsCoinbase, nSigOpsCost, lock_points.value()));
|
||||||
ws.m_vsize = entry->GetTxSize();
|
ws.m_vsize = entry->GetTxSize();
|
||||||
|
|
||||||
@ -3108,6 +3114,9 @@ bool Chainstate::DisconnectTip(BlockValidationState& state, DisconnectedBlockTra
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (disconnectpool && m_mempool) {
|
if (disconnectpool && m_mempool) {
|
||||||
|
for (auto it = block.vtx.rbegin(); it != block.vtx.rend(); ++it) {
|
||||||
|
m_mempool->UpdateDependentPriorities(*(*it), pindexDelete->nHeight, false);
|
||||||
|
}
|
||||||
// Save transactions to re-add to mempool at end of reorg. If any entries are evicted for
|
// Save transactions to re-add to mempool at end of reorg. If any entries are evicted for
|
||||||
// exceeding memory limits, remove them and their descendants from the mempool.
|
// exceeding memory limits, remove them and their descendants from the mempool.
|
||||||
for (auto&& evicted_tx : disconnectpool->AddTransactionsFromBlock(block.vtx)) {
|
for (auto&& evicted_tx : disconnectpool->AddTransactionsFromBlock(block.vtx)) {
|
||||||
|
Loading…
Reference in New Issue
Block a user