// Copyright (c) 2009-2010 Satoshi Nakamoto // Copyright (c) 2009-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 #include #include #include #include #include #include #include #include #include #include #include #include #include TRACEPOINT_SEMAPHORE(mempool, added); TRACEPOINT_SEMAPHORE(mempool, removed); bool TestLockPointValidity(CChain& active_chain, const LockPoints& lp) { AssertLockHeld(cs_main); // If there are relative lock times then the maxInputBlock will be set // If there are no relative lock times, the LockPoints don't depend on the chain if (lp.maxInputBlock) { // Check whether active_chain is an extension of the block at which the LockPoints // calculation was valid. If not LockPoints are no longer valid if (!active_chain.Contains(lp.maxInputBlock)) { return false; } } // LockPoints still valid return true; } std::vector CTxMemPool::GetChildren(const CTxMemPoolEntry& entry) const { LOCK(cs); std::vector ret; WITH_FRESH_EPOCH(m_epoch); auto iter = mapNextTx.lower_bound(COutPoint(entry.GetTx().GetHash(), 0)); for (; iter != mapNextTx.end() && iter->first->hash == entry.GetTx().GetHash(); ++iter) { if (!visited(iter->second)) { ret.emplace_back(*(iter->second)); } } return ret; } std::vector CTxMemPool::GetParents(const CTxMemPoolEntry& entry) const { LOCK(cs); std::vector ret; std::set inputs; for (const auto& txin : entry.GetTx().vin) { inputs.insert(txin.prevout.hash); } for (const auto& hash : inputs) { std::optional piter = GetIter(hash); if (piter) { ret.emplace_back(**piter); } } return ret; } void CTxMemPool::UpdateTransactionsFromBlock(const std::vector& vHashesToUpdate) { AssertLockHeld(cs); // Iterate in reverse, so that whenever we are looking at a transaction // we are sure that all in-mempool descendants have already been processed. for (const Txid& hash : vHashesToUpdate | std::views::reverse) { // calculate children from mapNextTx txiter it = mapTx.find(hash); if (it == mapTx.end()) { continue; } auto iter = mapNextTx.lower_bound(COutPoint(hash, 0)); { for (; iter != mapNextTx.end() && iter->first->hash == hash; ++iter) { txiter childIter = iter->second; assert(childIter != mapTx.end()); // Add dependencies that are discovered between transactions in the // block and transactions that were in the mempool to txgraph. m_txgraph->AddDependency(/*parent=*/*it, /*child=*/*childIter); } } } auto txs_to_remove = m_txgraph->Trim(); // Enforce cluster size limits. for (auto txptr : txs_to_remove) { const CTxMemPoolEntry& entry = *(static_cast(txptr)); removeUnchecked(mapTx.iterator_to(entry), MemPoolRemovalReason::SIZELIMIT); } } bool CTxMemPool::HasDescendants(const Txid& txid) const { LOCK(cs); auto entry = GetEntry(txid); if (!entry) return false; return m_txgraph->GetDescendants(*entry, TxGraph::Level::MAIN).size() > 1; } CTxMemPool::setEntries CTxMemPool::CalculateMemPoolAncestors(const CTxMemPoolEntry &entry) const { auto ancestors = m_txgraph->GetAncestors(entry, TxGraph::Level::MAIN); setEntries ret; if (ancestors.size() > 0) { for (auto ancestor : ancestors) { if (ancestor != &entry) { ret.insert(mapTx.iterator_to(static_cast(*ancestor))); } } return ret; } // If we didn't get anything back, the transaction is not in the graph. // Find each parent and call GetAncestors on each. setEntries staged_parents; const CTransaction &tx = entry.GetTx(); // Get parents of this transaction that are in the mempool for (unsigned int i = 0; i < tx.vin.size(); i++) { std::optional piter = GetIter(tx.vin[i].prevout.hash); if (piter) { staged_parents.insert(*piter); } } for (const auto& parent : staged_parents) { auto parent_ancestors = m_txgraph->GetAncestors(*parent, TxGraph::Level::MAIN); for (auto ancestor : parent_ancestors) { ret.insert(mapTx.iterator_to(static_cast(*ancestor))); } } return ret; } static CTxMemPool::Options&& Flatten(CTxMemPool::Options&& opts, bilingual_str& error) { opts.check_ratio = std::clamp(opts.check_ratio, 0, 1'000'000); int64_t cluster_limit_bytes = opts.limits.cluster_size_vbytes * 40; if (opts.max_size_bytes < 0 || (opts.max_size_bytes > 0 && opts.max_size_bytes < cluster_limit_bytes)) { error = strprintf(_("-maxmempool must be at least %d MB"), std::ceil(cluster_limit_bytes / 1'000'000.0)); } return std::move(opts); } CTxMemPool::CTxMemPool(Options opts, bilingual_str& error) : m_opts{Flatten(std::move(opts), error)} { m_txgraph = MakeTxGraph(m_opts.limits.cluster_count, m_opts.limits.cluster_size_vbytes * WITNESS_SCALE_FACTOR, ACCEPTABLE_ITERS); } bool CTxMemPool::isSpent(const COutPoint& outpoint) const { LOCK(cs); return mapNextTx.count(outpoint); } unsigned int CTxMemPool::GetTransactionsUpdated() const { return nTransactionsUpdated; } void CTxMemPool::AddTransactionsUpdated(unsigned int n) { nTransactionsUpdated += n; } void CTxMemPool::Apply(ChangeSet* changeset) { AssertLockHeld(cs); m_txgraph->CommitStaging(); RemoveStaged(changeset->m_to_remove, MemPoolRemovalReason::REPLACED); for (size_t i=0; im_entry_vec.size(); ++i) { auto tx_entry = changeset->m_entry_vec[i]; // First splice this entry into mapTx. auto node_handle = changeset->m_to_add.extract(tx_entry); auto result = mapTx.insert(std::move(node_handle)); Assume(result.inserted); txiter it = result.position; addNewTransaction(it); } m_txgraph->DoWork(POST_CHANGE_WORK); } void CTxMemPool::addNewTransaction(CTxMemPool::txiter newit) { const CTxMemPoolEntry& entry = *newit; // Update cachedInnerUsage to include contained transaction's usage. // (When we update the entry for in-mempool parents, memory usage will be // further updated.) cachedInnerUsage += entry.DynamicMemoryUsage(); const CTransaction& tx = newit->GetTx(); for (unsigned int i = 0; i < tx.vin.size(); i++) { mapNextTx.insert(std::make_pair(&tx.vin[i].prevout, newit)); } // Don't bother worrying about child transactions of this one. // Normal case of a new transaction arriving is that there can't be any // children, because such children would be orphans. // An exception to that is if a transaction enters that used to be in a block. // In that case, our disconnect block logic will call UpdateTransactionsFromBlock // to clean up the mess we're leaving here. nTransactionsUpdated++; totalTxSize += entry.GetTxSize(); m_total_fee += entry.GetFee(); txns_randomized.emplace_back(tx.GetWitnessHash(), newit); newit->idx_randomized = txns_randomized.size() - 1; TRACEPOINT(mempool, added, entry.GetTx().GetHash().data(), entry.GetTxSize(), entry.GetFee() ); } void CTxMemPool::removeUnchecked(txiter it, MemPoolRemovalReason reason) { // We increment mempool sequence value no matter removal reason // even if not directly reported below. uint64_t mempool_sequence = GetAndIncrementSequence(); if (reason != MemPoolRemovalReason::BLOCK && m_opts.signals) { // Notify clients that a transaction has been removed from the mempool // for any reason except being included in a block. Clients interested // in transactions included in blocks can subscribe to the BlockConnected // notification. m_opts.signals->TransactionRemovedFromMempool(it->GetSharedTx(), reason, mempool_sequence); } TRACEPOINT(mempool, removed, it->GetTx().GetHash().data(), RemovalReasonToString(reason).c_str(), it->GetTxSize(), it->GetFee(), std::chrono::duration_cast>(it->GetTime()).count() ); for (const CTxIn& txin : it->GetTx().vin) mapNextTx.erase(txin.prevout); RemoveUnbroadcastTx(it->GetTx().GetHash(), true /* add logging because unchecked */); if (txns_randomized.size() > 1) { // Remove entry from txns_randomized by replacing it with the back and deleting the back. txns_randomized[it->idx_randomized] = std::move(txns_randomized.back()); txns_randomized[it->idx_randomized].second->idx_randomized = it->idx_randomized; txns_randomized.pop_back(); if (txns_randomized.size() * 2 < txns_randomized.capacity()) { txns_randomized.shrink_to_fit(); } } else { txns_randomized.clear(); } totalTxSize -= it->GetTxSize(); m_total_fee -= it->GetFee(); cachedInnerUsage -= it->DynamicMemoryUsage(); mapTx.erase(it); nTransactionsUpdated++; } // Calculates descendants of given entry and adds to setDescendants. void CTxMemPool::CalculateDescendants(txiter entryit, setEntries& setDescendants) const { (void)CalculateDescendants(*entryit, setDescendants); return; } CTxMemPool::txiter CTxMemPool::CalculateDescendants(const CTxMemPoolEntry& entry, setEntries& setDescendants) const { for (auto tx : m_txgraph->GetDescendants(entry, TxGraph::Level::MAIN)) { setDescendants.insert(mapTx.iterator_to(static_cast(*tx))); } return mapTx.iterator_to(entry); } void CTxMemPool::removeRecursive(CTxMemPool::txiter to_remove, MemPoolRemovalReason reason) { AssertLockHeld(cs); Assume(!m_have_changeset); auto descendants = m_txgraph->GetDescendants(*to_remove, TxGraph::Level::MAIN); for (auto tx: descendants) { removeUnchecked(mapTx.iterator_to(static_cast(*tx)), reason); } } void CTxMemPool::removeRecursive(const CTransaction &origTx, MemPoolRemovalReason reason) { // Remove transaction from memory pool AssertLockHeld(cs); Assume(!m_have_changeset); txiter origit = mapTx.find(origTx.GetHash()); if (origit != mapTx.end()) { removeRecursive(origit, reason); } else { // When recursively removing but origTx isn't in the mempool // be sure to remove any descendants that are in the pool. This can // happen during chain re-orgs if origTx isn't re-accepted into // the mempool for any reason. auto iter = mapNextTx.lower_bound(COutPoint(origTx.GetHash(), 0)); std::vector to_remove; while (iter != mapNextTx.end() && iter->first->hash == origTx.GetHash()) { to_remove.emplace_back(&*(iter->second)); ++iter; } auto all_removes = m_txgraph->GetDescendantsUnion(to_remove, TxGraph::Level::MAIN); for (auto ref : all_removes) { auto tx = mapTx.iterator_to(static_cast(*ref)); removeUnchecked(tx, reason); } } } void CTxMemPool::removeForReorg(CChain& chain, std::function check_final_and_mature) { // Remove transactions spending a coinbase which are now immature and no-longer-final transactions AssertLockHeld(cs); AssertLockHeld(::cs_main); Assume(!m_have_changeset); std::vector to_remove; for (txiter it = mapTx.begin(); it != mapTx.end(); it++) { if (check_final_and_mature(it)) { to_remove.emplace_back(&*it); } } auto all_to_remove = m_txgraph->GetDescendantsUnion(to_remove, TxGraph::Level::MAIN); for (auto ref : all_to_remove) { auto it = mapTx.iterator_to(static_cast(*ref)); removeUnchecked(it, MemPoolRemovalReason::REORG); } for (indexed_transaction_set::const_iterator it = mapTx.begin(); it != mapTx.end(); it++) { assert(TestLockPointValidity(chain, it->GetLockPoints())); } m_txgraph->DoWork(POST_CHANGE_WORK); } void CTxMemPool::removeConflicts(const CTransaction &tx) { // Remove transactions which depend on inputs of tx, recursively AssertLockHeld(cs); for (const CTxIn &txin : tx.vin) { auto it = mapNextTx.find(txin.prevout); if (it != mapNextTx.end()) { const CTransaction &txConflict = it->second->GetTx(); if (Assume(txConflict.GetHash() != tx.GetHash())) { ClearPrioritisation(txConflict.GetHash()); removeRecursive(it->second, MemPoolRemovalReason::CONFLICT); } } } } void CTxMemPool::removeForBlock(const std::vector& vtx, unsigned int nBlockHeight) { // Remove confirmed txs and conflicts when a new block is connected, updating the fee logic AssertLockHeld(cs); Assume(!m_have_changeset); std::vector txs_removed_for_block; if (mapTx.size() || mapNextTx.size() || mapDeltas.size()) { txs_removed_for_block.reserve(vtx.size()); for (const auto& tx : vtx) { txiter it = mapTx.find(tx->GetHash()); if (it != mapTx.end()) { txs_removed_for_block.emplace_back(*it); removeUnchecked(it, MemPoolRemovalReason::BLOCK); } removeConflicts(*tx); ClearPrioritisation(tx->GetHash()); } } if (m_opts.signals) { m_opts.signals->MempoolTransactionsRemovedForBlock(txs_removed_for_block, nBlockHeight); } lastRollingFeeUpdate = GetTime(); blockSinceLastRollingFeeBump = true; m_txgraph->DoWork(POST_CHANGE_WORK); } void CTxMemPool::check(const CCoinsViewCache& active_coins_tip, int64_t spendheight) const { if (m_opts.check_ratio == 0) return; if (FastRandomContext().randrange(m_opts.check_ratio) >= 1) return; AssertLockHeld(::cs_main); LOCK(cs); LogDebug(BCLog::MEMPOOL, "Checking mempool with %u transactions and %u inputs\n", (unsigned int)mapTx.size(), (unsigned int)mapNextTx.size()); uint64_t checkTotal = 0; CAmount check_total_fee{0}; CAmount check_total_modified_fee{0}; int64_t check_total_adjusted_weight{0}; uint64_t innerUsage = 0; assert(!m_txgraph->IsOversized(TxGraph::Level::MAIN)); m_txgraph->SanityCheck(); CCoinsViewCache mempoolDuplicate(const_cast(&active_coins_tip)); const auto score_with_topo{GetSortedScoreWithTopology()}; // Number of chunks is bounded by number of transactions. const auto diagram{GetFeerateDiagram()}; assert(diagram.size() <= score_with_topo.size() + 1); assert(diagram.size() >= 1); std::optional last_wtxid = std::nullopt; auto diagram_iter = diagram.cbegin(); for (const auto& it : score_with_topo) { // GetSortedScoreWithTopology() contains the same chunks as the feerate // diagram. We do not know where the chunk boundaries are, but we can // check that there are points at which they match the cumulative fee // and weight. // The feerate diagram should never get behind the current transaction // size totals. assert(diagram_iter->size >= check_total_adjusted_weight); if (diagram_iter->fee == check_total_modified_fee && diagram_iter->size == check_total_adjusted_weight) { ++diagram_iter; } checkTotal += it->GetTxSize(); check_total_adjusted_weight += it->GetAdjustedWeight(); check_total_fee += it->GetFee(); check_total_modified_fee += it->GetModifiedFee(); innerUsage += it->DynamicMemoryUsage(); const CTransaction& tx = it->GetTx(); // CompareMiningScoreWithTopology should agree with GetSortedScoreWithTopology() if (last_wtxid) { assert(CompareMiningScoreWithTopology(*last_wtxid, tx.GetWitnessHash())); } last_wtxid = tx.GetWitnessHash(); std::set setParentCheck; std::set setParentsStored; for (const CTxIn &txin : tx.vin) { // Check that every mempool transaction's inputs refer to available coins, or other mempool tx's. indexed_transaction_set::const_iterator it2 = mapTx.find(txin.prevout.hash); if (it2 != mapTx.end()) { const CTransaction& tx2 = it2->GetTx(); assert(tx2.vout.size() > txin.prevout.n && !tx2.vout[txin.prevout.n].IsNull()); setParentCheck.insert(*it2); } // We are iterating through the mempool entries sorted // topologically and by mining score. All parents must have been // checked before their children and their coins added to the // mempoolDuplicate coins cache. assert(mempoolDuplicate.HaveCoin(txin.prevout)); // Check whether its inputs are marked in mapNextTx. auto it3 = mapNextTx.find(txin.prevout); assert(it3 != mapNextTx.end()); assert(it3->first == &txin.prevout); assert(&it3->second->GetTx() == &tx); } auto comp = [](const CTxMemPoolEntry& a, const CTxMemPoolEntry& b) -> bool { return a.GetTx().GetHash() == b.GetTx().GetHash(); }; for (auto &txentry : GetParents(*it)) { setParentsStored.insert(dynamic_cast(txentry.get())); } assert(setParentCheck.size() == setParentsStored.size()); assert(std::equal(setParentCheck.begin(), setParentCheck.end(), setParentsStored.begin(), comp)); // Check children against mapNextTx std::set setChildrenCheck; std::set setChildrenStored; auto iter = mapNextTx.lower_bound(COutPoint(it->GetTx().GetHash(), 0)); for (; iter != mapNextTx.end() && iter->first->hash == it->GetTx().GetHash(); ++iter) { txiter childit = iter->second; assert(childit != mapTx.end()); // mapNextTx points to in-mempool transactions setChildrenCheck.insert(*childit); } for (auto &txentry : GetChildren(*it)) { setChildrenStored.insert(dynamic_cast(txentry.get())); } assert(setChildrenCheck.size() == setChildrenStored.size()); assert(std::equal(setChildrenCheck.begin(), setChildrenCheck.end(), setChildrenStored.begin(), comp)); TxValidationState dummy_state; // Not used. CheckTxInputs() should always pass CAmount txfee = 0; assert(!tx.IsCoinBase()); assert(Consensus::CheckTxInputs(tx, dummy_state, mempoolDuplicate, spendheight, txfee)); for (const auto& input: tx.vin) mempoolDuplicate.SpendCoin(input.prevout); AddCoins(mempoolDuplicate, tx, std::numeric_limits::max()); } for (auto it = mapNextTx.cbegin(); it != mapNextTx.cend(); it++) { indexed_transaction_set::const_iterator it2 = it->second; assert(it2 != mapTx.end()); } ++diagram_iter; assert(diagram_iter == diagram.cend()); assert(totalTxSize == checkTotal); assert(m_total_fee == check_total_fee); assert(diagram.back().fee == check_total_modified_fee); assert(diagram.back().size == check_total_adjusted_weight); assert(innerUsage == cachedInnerUsage); } bool CTxMemPool::CompareMiningScoreWithTopology(const Wtxid& hasha, const Wtxid& hashb) const { /* Return `true` if hasha should be considered sooner than hashb, namely when: * a is not in the mempool but b is, or * both are in the mempool but a is sorted before b in the total mempool ordering * (which takes dependencies and (chunk) feerates into account). */ LOCK(cs); auto j{GetIter(hashb)}; if (!j.has_value()) return false; auto i{GetIter(hasha)}; if (!i.has_value()) return true; return m_txgraph->CompareMainOrder(*i.value(), *j.value()) < 0; } std::vector CTxMemPool::GetSortedScoreWithTopology() const { std::vector iters; AssertLockHeld(cs); iters.reserve(mapTx.size()); for (indexed_transaction_set::iterator mi = mapTx.begin(); mi != mapTx.end(); ++mi) { iters.push_back(mi); } std::sort(iters.begin(), iters.end(), [this](const auto& a, const auto& b) EXCLUSIVE_LOCKS_REQUIRED(cs) noexcept { return m_txgraph->CompareMainOrder(*a, *b) < 0; }); return iters; } std::vector CTxMemPool::entryAll() const { AssertLockHeld(cs); std::vector ret; ret.reserve(mapTx.size()); for (const auto& it : GetSortedScoreWithTopology()) { ret.emplace_back(*it); } return ret; } std::vector CTxMemPool::infoAll() const { LOCK(cs); auto iters = GetSortedScoreWithTopology(); std::vector ret; ret.reserve(mapTx.size()); for (auto it : iters) { ret.push_back(GetInfo(it)); } return ret; } const CTxMemPoolEntry* CTxMemPool::GetEntry(const Txid& txid) const { AssertLockHeld(cs); const auto i = mapTx.find(txid); return i == mapTx.end() ? nullptr : &(*i); } CTransactionRef CTxMemPool::get(const Txid& hash) const { LOCK(cs); indexed_transaction_set::const_iterator i = mapTx.find(hash); if (i == mapTx.end()) return nullptr; return i->GetSharedTx(); } void CTxMemPool::PrioritiseTransaction(const Txid& hash, const CAmount& nFeeDelta) { { LOCK(cs); CAmount &delta = mapDeltas[hash]; delta = SaturatingAdd(delta, nFeeDelta); txiter it = mapTx.find(hash); if (it != mapTx.end()) { // PrioritiseTransaction calls stack on previous ones. Set the new // transaction fee to be current modified fee + feedelta. it->UpdateModifiedFee(nFeeDelta); m_txgraph->SetTransactionFee(*it, it->GetModifiedFee()); ++nTransactionsUpdated; } if (delta == 0) { mapDeltas.erase(hash); LogInfo("PrioritiseTransaction: %s (%sin mempool) delta cleared\n", hash.ToString(), it == mapTx.end() ? "not " : ""); } else { LogInfo("PrioritiseTransaction: %s (%sin mempool) fee += %s, new delta=%s\n", hash.ToString(), it == mapTx.end() ? "not " : "", FormatMoney(nFeeDelta), FormatMoney(delta)); } } } void CTxMemPool::ApplyDelta(const Txid& hash, CAmount &nFeeDelta) const { AssertLockHeld(cs); std::map::const_iterator pos = mapDeltas.find(hash); if (pos == mapDeltas.end()) return; const CAmount &delta = pos->second; nFeeDelta += delta; } void CTxMemPool::ClearPrioritisation(const Txid& hash) { AssertLockHeld(cs); mapDeltas.erase(hash); } std::vector CTxMemPool::GetPrioritisedTransactions() const { AssertLockNotHeld(cs); LOCK(cs); std::vector result; result.reserve(mapDeltas.size()); for (const auto& [txid, delta] : mapDeltas) { const auto iter{mapTx.find(txid)}; const bool in_mempool{iter != mapTx.end()}; std::optional modified_fee; if (in_mempool) modified_fee = iter->GetModifiedFee(); result.emplace_back(delta_info{in_mempool, delta, modified_fee, txid}); } return result; } const CTransaction* CTxMemPool::GetConflictTx(const COutPoint& prevout) const { const auto it = mapNextTx.find(prevout); return it == mapNextTx.end() ? nullptr : &(it->second->GetTx()); } std::optional CTxMemPool::GetIter(const Txid& txid) const { AssertLockHeld(cs); auto it = mapTx.find(txid); return it != mapTx.end() ? std::make_optional(it) : std::nullopt; } std::optional CTxMemPool::GetIter(const Wtxid& wtxid) const { AssertLockHeld(cs); auto it{mapTx.project<0>(mapTx.get().find(wtxid))}; return it != mapTx.end() ? std::make_optional(it) : std::nullopt; } CTxMemPool::setEntries CTxMemPool::GetIterSet(const std::set& hashes) const { CTxMemPool::setEntries ret; for (const auto& h : hashes) { const auto mi = GetIter(h); if (mi) ret.insert(*mi); } return ret; } std::vector CTxMemPool::GetIterVec(const std::vector& txids) const { AssertLockHeld(cs); std::vector ret; ret.reserve(txids.size()); for (const auto& txid : txids) { const auto it{GetIter(txid)}; if (!it) return {}; ret.push_back(*it); } return ret; } bool CTxMemPool::HasNoInputsOf(const CTransaction &tx) const { for (unsigned int i = 0; i < tx.vin.size(); i++) if (exists(tx.vin[i].prevout.hash)) return false; return true; } CCoinsViewMemPool::CCoinsViewMemPool(CCoinsView* baseIn, const CTxMemPool& mempoolIn) : CCoinsViewBacked(baseIn), mempool(mempoolIn) { } std::optional CCoinsViewMemPool::GetCoin(const COutPoint& outpoint) const { // Check to see if the inputs are made available by another tx in the package. // These Coins would not be available in the underlying CoinsView. if (auto it = m_temp_added.find(outpoint); it != m_temp_added.end()) { return it->second; } // If an entry in the mempool exists, always return that one, as it's guaranteed to never // conflict with the underlying cache, and it cannot have pruned entries (as it contains full) // transactions. First checking the underlying cache risks returning a pruned entry instead. CTransactionRef ptx = mempool.get(outpoint.hash); if (ptx) { if (outpoint.n < ptx->vout.size()) { Coin coin(ptx->vout[outpoint.n], MEMPOOL_HEIGHT, false); m_non_base_coins.emplace(outpoint); return coin; } return std::nullopt; } return base->GetCoin(outpoint); } void CCoinsViewMemPool::PackageAddTransaction(const CTransactionRef& tx) { for (unsigned int n = 0; n < tx->vout.size(); ++n) { m_temp_added.emplace(COutPoint(tx->GetHash(), n), Coin(tx->vout[n], MEMPOOL_HEIGHT, false)); m_non_base_coins.emplace(tx->GetHash(), n); } } void CCoinsViewMemPool::Reset() { m_temp_added.clear(); m_non_base_coins.clear(); } size_t CTxMemPool::DynamicMemoryUsage() const { LOCK(cs); // Estimate the overhead of mapTx to be 9 pointers (3 pointers per index) + an allocation, as no exact formula for boost::multi_index_contained is implemented. return memusage::MallocUsage(sizeof(CTxMemPoolEntry) + 9 * sizeof(void*)) * mapTx.size() + memusage::DynamicUsage(mapNextTx) + memusage::DynamicUsage(mapDeltas) + memusage::DynamicUsage(txns_randomized) + m_txgraph->GetMainMemoryUsage() + cachedInnerUsage; } void CTxMemPool::RemoveUnbroadcastTx(const Txid& txid, const bool unchecked) { LOCK(cs); if (m_unbroadcast_txids.erase(txid)) { LogDebug(BCLog::MEMPOOL, "Removed %i from set of unbroadcast txns%s\n", txid.GetHex(), (unchecked ? " before confirmation that txn was sent out" : "")); } } void CTxMemPool::RemoveStaged(setEntries &stage, MemPoolRemovalReason reason) { AssertLockHeld(cs); for (txiter it : stage) { removeUnchecked(it, reason); } } bool CTxMemPool::CheckPolicyLimits(const CTransactionRef& tx) { LOCK(cs); // Use ChangeSet interface to check whether the cluster count // limits would be violated. Note that the changeset will be destroyed // when it goes out of scope. auto changeset = GetChangeSet(); (void) changeset->StageAddition(tx, /*fee=*/0, /*time=*/0, /*entry_height=*/0, /*entry_sequence=*/0, /*spends_coinbase=*/false, /*sigops_cost=*/0, LockPoints{}); return changeset->CheckMemPoolPolicyLimits(); } int CTxMemPool::Expire(std::chrono::seconds time) { AssertLockHeld(cs); Assume(!m_have_changeset); indexed_transaction_set::index::type::iterator it = mapTx.get().begin(); setEntries toremove; while (it != mapTx.get().end() && it->GetTime() < time) { toremove.insert(mapTx.project<0>(it)); it++; } setEntries stage; for (txiter removeit : toremove) { CalculateDescendants(removeit, stage); } RemoveStaged(stage, MemPoolRemovalReason::EXPIRY); return stage.size(); } CFeeRate CTxMemPool::GetMinFee(size_t sizelimit) const { LOCK(cs); if (!blockSinceLastRollingFeeBump || rollingMinimumFeeRate == 0) return CFeeRate(llround(rollingMinimumFeeRate)); int64_t time = GetTime(); if (time > lastRollingFeeUpdate + 10) { double halflife = ROLLING_FEE_HALFLIFE; if (DynamicMemoryUsage() < sizelimit / 4) halflife /= 4; else if (DynamicMemoryUsage() < sizelimit / 2) halflife /= 2; rollingMinimumFeeRate = rollingMinimumFeeRate / pow(2.0, (time - lastRollingFeeUpdate) / halflife); lastRollingFeeUpdate = time; if (rollingMinimumFeeRate < (double)m_opts.incremental_relay_feerate.GetFeePerK() / 2) { rollingMinimumFeeRate = 0; return CFeeRate(0); } } return std::max(CFeeRate(llround(rollingMinimumFeeRate)), m_opts.incremental_relay_feerate); } void CTxMemPool::trackPackageRemoved(const CFeeRate& rate) { AssertLockHeld(cs); if (rate.GetFeePerK() > rollingMinimumFeeRate) { rollingMinimumFeeRate = rate.GetFeePerK(); blockSinceLastRollingFeeBump = false; } } void CTxMemPool::TrimToSize(size_t sizelimit, std::vector* pvNoSpendsRemaining) { AssertLockHeld(cs); Assume(!m_have_changeset); unsigned nTxnRemoved = 0; CFeeRate maxFeeRateRemoved(0); while (!mapTx.empty() && DynamicMemoryUsage() > sizelimit) { const auto &[worst_chunk, feeperweight] = m_txgraph->GetWorstMainChunk(); FeePerVSize feerate = ToFeePerVSize(feeperweight); CFeeRate removed{feerate.fee, feerate.size}; // We set the new mempool min fee to the feerate of the removed set, plus the // "minimum reasonable fee rate" (ie some value under which we consider txn // to have 0 fee). This way, we don't allow txn to enter mempool with feerate // equal to txn which were removed with no block in between. removed += m_opts.incremental_relay_feerate; trackPackageRemoved(removed); maxFeeRateRemoved = std::max(maxFeeRateRemoved, removed); nTxnRemoved += worst_chunk.size(); std::vector txn; if (pvNoSpendsRemaining) { txn.reserve(worst_chunk.size()); for (auto ref : worst_chunk) { txn.emplace_back(static_cast(*ref).GetTx()); } } setEntries stage; for (auto ref : worst_chunk) { stage.insert(mapTx.iterator_to(static_cast(*ref))); } for (auto e : stage) { removeUnchecked(e, MemPoolRemovalReason::SIZELIMIT); } if (pvNoSpendsRemaining) { for (const CTransaction& tx : txn) { for (const CTxIn& txin : tx.vin) { if (exists(txin.prevout.hash)) continue; pvNoSpendsRemaining->push_back(txin.prevout); } } } } if (maxFeeRateRemoved > CFeeRate(0)) { LogDebug(BCLog::MEMPOOL, "Removed %u txn, rolling minimum fee bumped to %s\n", nTxnRemoved, maxFeeRateRemoved.ToString()); } } std::tuple CTxMemPool::CalculateAncestorData(const CTxMemPoolEntry& entry) const { auto ancestors = m_txgraph->GetAncestors(entry, TxGraph::Level::MAIN); size_t ancestor_count = ancestors.size(); size_t ancestor_size = 0; CAmount ancestor_fees = 0; for (auto tx: ancestors) { const CTxMemPoolEntry& anc = static_cast(*tx); ancestor_size += anc.GetTxSize(); ancestor_fees += anc.GetModifiedFee(); } return {ancestor_count, ancestor_size, ancestor_fees}; } std::tuple CTxMemPool::CalculateDescendantData(const CTxMemPoolEntry& entry) const { auto descendants = m_txgraph->GetDescendants(entry, TxGraph::Level::MAIN); size_t descendant_count = descendants.size(); size_t descendant_size = 0; CAmount descendant_fees = 0; for (auto tx: descendants) { const CTxMemPoolEntry &desc = static_cast(*tx); descendant_size += desc.GetTxSize(); descendant_fees += desc.GetModifiedFee(); } return {descendant_count, descendant_size, descendant_fees}; } void CTxMemPool::GetTransactionAncestry(const Txid& txid, size_t& ancestors, size_t& cluster_count, size_t* const ancestorsize, CAmount* const ancestorfees) const { LOCK(cs); auto it = mapTx.find(txid); ancestors = cluster_count = 0; if (it != mapTx.end()) { auto [ancestor_count, ancestor_size, ancestor_fees] = CalculateAncestorData(*it); ancestors = ancestor_count; if (ancestorsize) *ancestorsize = ancestor_size; if (ancestorfees) *ancestorfees = ancestor_fees; cluster_count = m_txgraph->GetCluster(*it, TxGraph::Level::MAIN).size(); } } bool CTxMemPool::GetLoadTried() const { LOCK(cs); return m_load_tried; } void CTxMemPool::SetLoadTried(bool load_tried) { LOCK(cs); m_load_tried = load_tried; } std::vector CTxMemPool::GatherClusters(const std::vector& txids) const { AssertLockHeld(cs); std::vector ret; std::set unique_cluster_representatives; for (auto txid : txids) { auto it = mapTx.find(txid); if (it != mapTx.end()) { // Note that TxGraph::GetCluster will return results in graph // order, which is deterministic (as long as we are not modifying // the graph). auto cluster = m_txgraph->GetCluster(*it, TxGraph::Level::MAIN); if (unique_cluster_representatives.insert(static_cast(&(**cluster.begin()))).second) { for (auto tx : cluster) { ret.emplace_back(mapTx.iterator_to(static_cast(*tx))); } } } } if (ret.size() > 500) { return {}; } return ret; } util::Result, std::vector>> CTxMemPool::ChangeSet::CalculateChunksForRBF() { LOCK(m_pool->cs); if (!CheckMemPoolPolicyLimits()) { return util::Error{Untranslated("cluster size limit exceeded")}; } return m_pool->m_txgraph->GetMainStagingDiagrams(); } CTxMemPool::ChangeSet::TxHandle CTxMemPool::ChangeSet::StageAddition(const CTransactionRef& tx, const CAmount fee, int64_t time, unsigned int entry_height, uint64_t entry_sequence, bool spends_coinbase, int64_t sigops_cost, LockPoints lp) { LOCK(m_pool->cs); Assume(m_to_add.find(tx->GetHash()) == m_to_add.end()); Assume(!m_dependencies_processed); // We need to process dependencies after adding a new transaction. m_dependencies_processed = false; CAmount delta{0}; m_pool->ApplyDelta(tx->GetHash(), delta); TxGraph::Ref ref(m_pool->m_txgraph->AddTransaction(FeePerWeight(fee, GetSigOpsAdjustedWeight(GetTransactionWeight(*tx), sigops_cost, ::nBytesPerSigOp)))); auto newit = m_to_add.emplace(std::move(ref), tx, fee, time, entry_height, entry_sequence, spends_coinbase, sigops_cost, lp).first; if (delta) { newit->UpdateModifiedFee(delta); m_pool->m_txgraph->SetTransactionFee(*newit, newit->GetModifiedFee()); } m_entry_vec.push_back(newit); return newit; } void CTxMemPool::ChangeSet::StageRemoval(CTxMemPool::txiter it) { LOCK(m_pool->cs); m_pool->m_txgraph->RemoveTransaction(*it); m_to_remove.insert(it); } void CTxMemPool::ChangeSet::Apply() { LOCK(m_pool->cs); if (!m_dependencies_processed) { ProcessDependencies(); } m_pool->Apply(this); m_to_add.clear(); m_to_remove.clear(); m_entry_vec.clear(); m_ancestors.clear(); } void CTxMemPool::ChangeSet::ProcessDependencies() { LOCK(m_pool->cs); Assume(!m_dependencies_processed); // should only call this once. for (const auto& entryptr : m_entry_vec) { for (const auto &txin : entryptr->GetSharedTx()->vin) { std::optional piter = m_pool->GetIter(txin.prevout.hash); if (!piter) { auto it = m_to_add.find(txin.prevout.hash); if (it != m_to_add.end()) { piter = std::make_optional(it); } } if (piter) { m_pool->m_txgraph->AddDependency(/*parent=*/**piter, /*child=*/*entryptr); } } } m_dependencies_processed = true; return; } bool CTxMemPool::ChangeSet::CheckMemPoolPolicyLimits() { LOCK(m_pool->cs); if (!m_dependencies_processed) { ProcessDependencies(); } return !m_pool->m_txgraph->IsOversized(TxGraph::Level::TOP); } std::vector CTxMemPool::GetFeerateDiagram() const { FeePerWeight zero{}; std::vector ret; ret.emplace_back(zero); StartBlockBuilding(); std::vector dummy; FeePerWeight last_selection = GetBlockBuilderChunk(dummy); while (last_selection != FeePerWeight{}) { last_selection += ret.back(); ret.emplace_back(last_selection); IncludeBuilderChunk(); last_selection = GetBlockBuilderChunk(dummy); } StopBlockBuilding(); return ret; }