6#include <bitcoin-build-config.h>
102 "level 0 reads the blocks from disk",
103 "level 1 verifies block validity",
104 "level 2 verifies undo data",
105 "level 3 checks disconnection of tip blocks",
106 "level 4 tries to reconnect the blocks",
107 "each level includes the checks of the previous levels",
119 static constexpr uint32_t flush_ratio{320};
152 std::vector<CScriptCheck>* pvChecks =
nullptr)
165 const int nBlockHeight = active_chain_tip.nHeight + 1;
172 const int64_t nBlockTime{active_chain_tip.GetMedianTimePast()};
174 return IsFinalTx(tx, nBlockHeight, nBlockTime);
188std::optional<std::vector<int>> CalculatePrevHeights(
193 std::vector<int> prev_heights;
194 prev_heights.resize(tx.
vin.size());
195 for (
size_t i = 0; i < tx.
vin.size(); ++i) {
196 if (
auto coin{coins.
GetCoin(tx.
vin[i].prevout)}) {
201 LogInfo(
"ERROR: %s: Missing input %d in transaction \'%s\'\n", __func__, i, tx.
GetHash().
GetHex());
216 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
217 if (!prev_heights.has_value())
return std::nullopt;
220 next_tip.
pprev = tip;
238 int max_input_height{0};
239 for (
const int height : prev_heights.value()) {
241 if (height != next_tip.
nHeight) {
242 max_input_height = std::max(max_input_height, height);
277 int expired = pool.Expire(GetTime<std::chrono::seconds>() - pool.m_opts.expiry);
282 std::vector<COutPoint> vNoSpendsRemaining;
283 pool.TrimToSize(pool.m_opts.max_size_bytes, &vNoSpendsRemaining);
284 for (
const COutPoint& removed : vNoSpendsRemaining)
285 coins_cache.Uncache(removed);
291 if (active_chainstate.m_chainman.IsInitialBlockDownload()) {
296 if (active_chainstate.m_chain.Height() < active_chainstate.m_chainman.m_best_header->nHeight - 1) {
310 std::vector<Txid> vHashUpdate;
317 const auto queuedTx = disconnectpool.
take();
318 auto it = queuedTx.rbegin();
319 while (it != queuedTx.rend()) {
321 if (!fAddToMempool || (*it)->IsCoinBase() ||
323 true,
false).m_result_type !=
329 vHashUpdate.push_back((*it)->GetHash());
370 it->UpdateLockPoints(*new_lock_points);
377 if (it->GetSpendsCoinbase()) {
383 if (coin.IsCoinBase() && mempool_spend_height - coin.nHeight <
COINBASE_MATURITY) {
419 if (coin.
IsSpent())
return false;
449 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
450 m_active_chainstate(active_chainstate)
458 const int64_t m_accept_time;
459 const bool m_bypass_limits;
467 std::vector<COutPoint>& m_coins_to_uncache;
469 const bool m_test_accept;
473 const bool m_allow_replacement;
475 const bool m_allow_sibling_eviction;
478 const bool m_package_submission;
482 const bool m_package_feerates;
487 const std::optional<CFeeRate> m_client_maxfeerate;
490 static ATMPArgs SingleAccept(
const CChainParams& chainparams, int64_t accept_time,
491 bool bypass_limits, std::vector<COutPoint>& coins_to_uncache,
493 return ATMPArgs{ chainparams,
507 static ATMPArgs PackageTestAccept(
const CChainParams& chainparams, int64_t accept_time,
508 std::vector<COutPoint>& coins_to_uncache) {
509 return ATMPArgs{ chainparams,
523 static ATMPArgs PackageChildWithParents(
const CChainParams& chainparams, int64_t accept_time,
524 std::vector<COutPoint>& coins_to_uncache,
const std::optional<CFeeRate>& client_maxfeerate) {
525 return ATMPArgs{ chainparams,
539 static ATMPArgs SingleInPackageAccept(
const ATMPArgs& package_args) {
540 return ATMPArgs{ package_args.m_chainparams,
541 package_args.m_accept_time,
543 package_args.m_coins_to_uncache,
544 package_args.m_test_accept,
549 package_args.m_client_maxfeerate,
559 std::vector<COutPoint>& coins_to_uncache,
561 bool allow_replacement,
562 bool allow_sibling_eviction,
563 bool package_submission,
564 bool package_feerates,
565 std::optional<CFeeRate> client_maxfeerate)
566 : m_chainparams{chainparams},
567 m_accept_time{accept_time},
568 m_bypass_limits{bypass_limits},
569 m_coins_to_uncache{coins_to_uncache},
570 m_test_accept{test_accept},
571 m_allow_replacement{allow_replacement},
572 m_allow_sibling_eviction{allow_sibling_eviction},
573 m_package_submission{package_submission},
574 m_package_feerates{package_feerates},
575 m_client_maxfeerate{client_maxfeerate}
579 if (m_package_feerates) {
580 Assume(m_package_submission);
581 Assume(!m_allow_sibling_eviction);
583 if (m_allow_sibling_eviction)
Assume(m_allow_replacement);
594 ClearSubPackageState();
607 ClearSubPackageState();
638 std::set<Txid> m_conflicts;
643 std::vector<CTxMemPoolEntry::CTxMemPoolEntryRef> m_parents;
648 bool m_sibling_eviction{
false};
681 bool PackageRBFChecks(
const std::vector<CTransactionRef>& txns,
682 std::vector<Workspace>& workspaces,
703 std::map<Wtxid, MempoolAcceptResult>& results)
711 CAmount mempoolRejectFee = m_pool.GetMinFee().GetFee(package_size);
712 if (mempoolRejectFee > 0 && package_fee < mempoolRejectFee) {
716 if (package_fee < m_pool.m_opts.min_relay_feerate.GetFee(package_size)) {
718 strprintf(
"%d < %d", package_fee, m_pool.m_opts.min_relay_feerate.GetFee(package_size)));
725 return m_active_chainstate.m_chainman.m_validation_cache;
753 struct SubPackageState {
755 CAmount m_total_modified_fees{0};
757 int64_t m_total_vsize{0};
764 std::list<CTransactionRef> m_replaced_transactions;
766 std::unique_ptr<CTxMemPool::ChangeSet> m_changeset;
771 size_t m_conflicting_size{0};
774 struct SubPackageState m_subpackage;
779 m_subpackage = SubPackageState{};
782 CleanupTemporaryCoins();
786bool MemPoolAccept::PreChecks(ATMPArgs&
args, Workspace& ws)
792 const Txid& hash = ws.m_hash;
795 const int64_t nAcceptTime =
args.m_accept_time;
796 const bool bypass_limits =
args.m_bypass_limits;
797 std::vector<COutPoint>& coins_to_uncache =
args.m_coins_to_uncache;
812 if (m_pool.m_opts.require_standard && !
IsStandardTx(tx, m_pool.m_opts.max_datacarrier_bytes, m_pool.m_opts.permit_bare_multisig, m_pool.m_opts.dust_relay_feerate, reason)) {
830 }
else if (m_pool.exists(tx.
GetHash())) {
840 if (ptxConflicting) {
841 if (!
args.m_allow_replacement) {
845 ws.m_conflicts.insert(ptxConflicting->
GetHash());
849 m_view.SetBackend(m_viewmempool);
855 coins_to_uncache.push_back(txin.
prevout);
861 if (!m_view.HaveCoin(txin.
prevout)) {
876 (void)m_view.GetBestBlock();
883 assert(m_active_chainstate.m_blockman.LookupBlockIndex(m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
890 const std::optional<LockPoints> lock_points{
CalculateLockPointsAtTip(m_active_chainstate.m_chain.Tip(), m_view, tx)};
900 if (m_pool.m_opts.require_standard) {
916 bool fSpendsCoinbase =
false;
918 const Coin &coin = m_view.AccessCoin(txin.
prevout);
920 fSpendsCoinbase =
true;
927 const uint64_t entry_sequence = bypass_limits ? 0 : m_pool.GetSequence();
928 if (!m_subpackage.m_changeset) {
929 m_subpackage.m_changeset = m_pool.GetChangeSet();
931 ws.m_tx_handle = m_subpackage.m_changeset->StageAddition(ptx, ws.m_base_fees, nAcceptTime, m_active_chainstate.m_chain.Height(), entry_sequence, fSpendsCoinbase, nSigOpsCost, lock_points.value());
934 ws.m_modified_fees = ws.m_tx_handle->GetModifiedFee();
936 ws.m_vsize = ws.m_tx_handle->GetTxSize();
939 if (m_pool.m_opts.require_standard) {
940 if (!
PreCheckEphemeralTx(*ptx, m_pool.m_opts.dust_relay_feerate, ws.m_base_fees, ws.m_modified_fees, state)) {
952 if (!bypass_limits && !
args.m_package_feerates && !
CheckFeeRate(ws.m_vsize, ws.m_modified_fees, state))
return false;
954 ws.m_iters_conflicting = m_pool.GetIterSet(ws.m_conflicts);
956 ws.m_parents = m_pool.GetParents(*ws.m_tx_handle);
958 if (!
args.m_bypass_limits) {
960 if (
const auto err{
SingleTRUCChecks(m_pool, ws.m_ptx, ws.m_parents, ws.m_conflicts, ws.m_vsize)}) {
962 if (
args.m_allow_sibling_eviction && err->second !=
nullptr) {
967 ws.m_conflicts.insert(err->second->GetHash());
971 ws.m_iters_conflicting.insert(m_pool.GetIter(err->second->GetHash()).value());
972 ws.m_sibling_eviction =
true;
984 m_subpackage.m_rbf |= !ws.m_conflicts.empty();
988bool MemPoolAccept::ReplacementChecks(Workspace& ws)
994 const Txid& hash = ws.m_hash;
997 CFeeRate newFeeRate(ws.m_modified_fees, ws.m_vsize);
1004 strprintf(
"too many potential replacements%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1010 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1011 m_subpackage.m_conflicting_size += it->GetTxSize();
1014 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees, ws.m_modified_fees, ws.m_vsize,
1015 m_pool.m_opts.incremental_relay_feerate, hash)}) {
1018 strprintf(
"insufficient fee%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1022 for (
auto it : all_conflicts) {
1023 m_subpackage.m_changeset->StageRemoval(it);
1027 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1041bool MemPoolAccept::PackageRBFChecks(
const std::vector<CTransactionRef>& txns,
1042 std::vector<Workspace>& workspaces,
1043 const int64_t total_vsize,
1049 assert(std::all_of(txns.cbegin(), txns.cend(), [
this](
const auto& tx)
1050 { return !m_pool.exists(tx->GetHash());}));
1052 assert(txns.size() == workspaces.size());
1067 for (
const auto& ws : workspaces) {
1068 if (!ws.m_parents.empty()) {
1075 for (Workspace& ws : workspaces) {
1077 direct_conflict_iters.merge(ws.m_iters_conflicting);
1080 const auto& parent_ws = workspaces[0];
1081 const auto& child_ws = workspaces[1];
1089 "package RBF failed: too many potential replacements", *err_string);
1093 m_subpackage.m_changeset->StageRemoval(it);
1094 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1095 m_subpackage.m_conflicting_size += it->GetTxSize();
1099 const Txid& child_hash = child_ws.m_ptx->GetHash();
1100 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees,
1101 m_subpackage.m_total_modified_fees,
1102 m_subpackage.m_total_vsize,
1103 m_pool.m_opts.incremental_relay_feerate, child_hash)}) {
1105 "package RBF failed: insufficient anti-DoS fees", *err_string);
1110 const CFeeRate parent_feerate(parent_ws.m_modified_fees, parent_ws.m_vsize);
1111 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1112 if (package_feerate <= parent_feerate) {
1114 "package RBF failed: package feerate is less than or equal to parent feerate",
1115 strprintf(
"package feerate %s <= parent feerate is %s", package_feerate.ToString(), parent_feerate.ToString()));
1119 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1127 "package RBF failed: " + err_tup.value().second,
"");
1130 LogDebug(
BCLog::TXPACKAGES,
"package RBF checks passed: parent %s (wtxid=%s), child %s (wtxid=%s), package hash (%s)\n",
1131 txns.front()->GetHash().ToString(), txns.front()->GetWitnessHash().ToString(),
1132 txns.back()->GetHash().ToString(), txns.back()->GetWitnessHash().ToString(),
1139bool MemPoolAccept::PolicyScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1150 if (!
CheckInputScripts(tx, state, m_view, scriptVerifyFlags,
true,
false, ws.m_precomputed_txdata, GetValidationCache())) {
1162bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1167 const Txid& hash = ws.m_hash;
1187 ws.m_precomputed_txdata, m_active_chainstate.CoinsTip(), GetValidationCache())) {
1188 LogError(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against latest-block but not STANDARD flags %s, %s", hash.
ToString(), state.
ToString());
1195void MemPoolAccept::FinalizeSubpackage(
const ATMPArgs&
args)
1200 if (!m_subpackage.m_changeset->GetRemovals().empty())
Assume(
args.m_allow_replacement);
1204 std::string log_string =
strprintf(
"replacing mempool tx %s (wtxid=%s, fees=%s, vsize=%s). ",
1205 it->GetTx().GetHash().ToString(),
1206 it->GetTx().GetWitnessHash().ToString(),
1209 FeeFrac feerate{m_subpackage.m_total_modified_fees, int32_t(m_subpackage.m_total_vsize)};
1211 const bool replaced_with_tx{m_subpackage.m_changeset->GetTxCount() == 1};
1212 if (replaced_with_tx) {
1213 const CTransaction& tx = m_subpackage.m_changeset->GetAddedTxn(0);
1215 log_string +=
strprintf(
"New tx %s (wtxid=%s, fees=%s, vsize=%s)",
1221 tx_or_package_hash =
GetPackageHash(m_subpackage.m_changeset->GetAddedTxns());
1222 log_string +=
strprintf(
"New package %s with %lu txs, fees=%s, vsize=%s",
1223 tx_or_package_hash.ToString(),
1224 m_subpackage.m_changeset->GetTxCount(),
1231 it->GetTx().GetHash().data(),
1234 std::chrono::duration_cast<std::chrono::duration<std::uint64_t>>(it->GetTime()).count(),
1235 tx_or_package_hash.data(),
1240 m_subpackage.m_replaced_transactions.push_back(it->GetSharedTx());
1242 m_subpackage.m_changeset->Apply();
1243 m_subpackage.m_changeset.reset();
1246bool MemPoolAccept::SubmitPackage(
const ATMPArgs&
args, std::vector<Workspace>& workspaces,
1248 std::map<Wtxid, MempoolAcceptResult>& results)
1254 assert(std::all_of(workspaces.cbegin(), workspaces.cend(), [
this](
const auto& ws) { return !m_pool.exists(ws.m_ptx->GetHash()); }));
1256 bool all_submitted =
true;
1257 FinalizeSubpackage(
args);
1262 for (Workspace& ws : workspaces) {
1263 if (!ConsensusScriptChecks(
args, ws)) {
1267 all_submitted =
false;
1269 strprintf(
"BUG! PolicyScriptChecks succeeded but ConsensusScriptChecks failed: %s",
1270 ws.m_ptx->GetHash().ToString()));
1273 if (!all_submitted) {
1274 if (!m_subpackage.m_changeset) m_subpackage.m_changeset = m_pool.GetChangeSet();
1275 m_subpackage.m_changeset->StageRemoval(m_pool.GetIter(ws.m_ptx->GetHash()).value());
1278 if (!all_submitted) {
1279 Assume(m_subpackage.m_changeset);
1283 m_subpackage.m_changeset->Apply();
1284 m_subpackage.m_changeset.reset();
1288 std::vector<Wtxid> all_package_wtxids;
1289 all_package_wtxids.reserve(workspaces.size());
1290 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1291 [](
const auto& ws) { return ws.m_ptx->GetWitnessHash(); });
1293 if (!m_subpackage.m_replaced_transactions.empty()) {
1294 LogDebug(
BCLog::MEMPOOL,
"replaced %u mempool transactions with %u new one(s) for %s additional fees, %d delta bytes\n",
1295 m_subpackage.m_replaced_transactions.size(), workspaces.size(),
1296 m_subpackage.m_total_modified_fees - m_subpackage.m_conflicting_fees,
1297 m_subpackage.m_total_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1301 for (Workspace& ws : workspaces) {
1302 auto iter = m_pool.GetIter(ws.m_ptx->GetHash());
1303 Assume(iter.has_value());
1304 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1305 CFeeRate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1306 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1307 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1308 results.emplace(ws.m_ptx->GetWitnessHash(),
1310 ws.m_base_fees, effective_feerate, effective_feerate_wtxids));
1311 if (!m_pool.m_opts.signals)
continue;
1314 ws.m_vsize, (*iter)->GetHeight(),
1315 args.m_bypass_limits,
args.m_package_submission,
1317 m_pool.HasNoInputsOf(tx));
1318 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1320 return all_submitted;
1329 const std::vector<Wtxid> single_wtxid{ws.m_ptx->GetWitnessHash()};
1331 if (!PreChecks(
args, ws)) {
1339 if (m_subpackage.m_rbf && !ReplacementChecks(ws)) {
1348 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1355 if (ws.m_conflicts.size()) {
1356 auto ancestors = m_subpackage.m_changeset->CalculateMemPoolAncestors(ws.m_tx_handle);
1370 m_subpackage.m_total_vsize = ws.m_vsize;
1371 m_subpackage.m_total_modified_fees = ws.m_modified_fees;
1374 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1379 if (!
args.m_bypass_limits && m_pool.m_opts.require_standard) {
1381 if (!
CheckEphemeralSpends({ptx}, m_pool.m_opts.dust_relay_feerate, m_pool, ws.m_state, dummy_wtxid)) {
1392 const CFeeRate effective_feerate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1394 if (
args.m_test_accept) {
1396 ws.m_base_fees, effective_feerate, single_wtxid);
1399 FinalizeSubpackage(
args);
1402 if (!
args.m_package_submission && !
args.m_bypass_limits) {
1406 CleanupTemporaryCoins();
1408 if (!m_pool.exists(ws.m_hash)) {
1415 if (m_pool.m_opts.signals) {
1417 auto iter = m_pool.GetIter(tx.
GetHash());
1418 Assume(iter.has_value());
1420 ws.m_vsize, (*iter)->GetHeight(),
1421 args.m_bypass_limits,
args.m_package_submission,
1423 m_pool.HasNoInputsOf(tx));
1424 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1427 if (!m_subpackage.m_replaced_transactions.empty()) {
1428 LogDebug(
BCLog::MEMPOOL,
"replaced %u mempool transactions with 1 new transaction for %s additional fees, %d delta bytes\n",
1429 m_subpackage.m_replaced_transactions.size(),
1430 ws.m_modified_fees - m_subpackage.m_conflicting_fees,
1431 ws.m_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1435 effective_feerate, single_wtxid);
1447 std::vector<Workspace> workspaces{};
1448 workspaces.reserve(txns.size());
1449 std::transform(txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1450 [](
const auto& tx) { return Workspace(tx); });
1451 std::map<Wtxid, MempoolAcceptResult> results;
1454 for (Workspace& ws : workspaces) {
1455 if (!PreChecks(
args, ws)) {
1464 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1482 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1487 for (Workspace& ws : workspaces) {
1488 if (
auto err{
PackageTRUCChecks(m_pool, ws.m_ptx, ws.m_vsize, txns, ws.m_parents)}) {
1503 m_subpackage.m_total_vsize = std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1504 [](int64_t
sum,
auto& ws) { return sum + ws.m_vsize; });
1505 m_subpackage.m_total_modified_fees = std::accumulate(workspaces.cbegin(), workspaces.cend(),
CAmount{0},
1506 [](
CAmount sum,
auto& ws) { return sum + ws.m_modified_fees; });
1507 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1508 std::vector<Wtxid> all_package_wtxids;
1509 all_package_wtxids.reserve(workspaces.size());
1510 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1511 [](
const auto& ws) { return ws.m_ptx->GetWitnessHash(); });
1513 if (
args.m_package_feerates &&
1514 !
CheckFeeRate(m_subpackage.m_total_vsize, m_subpackage.m_total_modified_fees, placeholder_state)) {
1521 if (m_subpackage.m_rbf && !PackageRBFChecks(txns, workspaces, m_subpackage.m_total_vsize, package_state)) {
1526 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1532 if (m_pool.m_opts.require_standard) {
1535 if (!
CheckEphemeralSpends(txns, m_pool.m_opts.dust_relay_feerate, m_pool, child_state, child_wtxid)) {
1542 for (Workspace& ws : workspaces) {
1543 ws.m_package_feerate = package_feerate;
1544 if (!PolicyScriptChecks(
args, ws)) {
1550 if (
args.m_test_accept) {
1551 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1552 CFeeRate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1553 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1554 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1555 results.emplace(ws.m_ptx->GetWitnessHash(),
1557 ws.m_vsize, ws.m_base_fees, effective_feerate,
1558 effective_feerate_wtxids));
1564 if (!SubmitPackage(
args, workspaces, package_state, results)) {
1572void MemPoolAccept::CleanupTemporaryCoins()
1593 for (
const auto& outpoint : m_viewmempool.GetNonBaseCoins()) {
1596 m_view.Uncache(outpoint);
1599 m_viewmempool.Reset();
1607 if (subpackage.size() > 1) {
1608 return AcceptMultipleTransactionsInternal(subpackage,
args);
1610 const auto& tx = subpackage.front();
1611 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(
args);
1612 const auto single_res = AcceptSingleTransactionInternal(tx, single_args);
1623 ClearSubPackageState();
1630 Assert(!package.empty());
1656 std::map<Wtxid, MempoolAcceptResult> results_final;
1660 std::map<Wtxid, MempoolAcceptResult> individual_results_nonfinal;
1662 bool quit_early{
false};
1663 std::vector<CTransactionRef> txns_package_eval;
1664 for (
const auto& tx : package) {
1666 const auto& txid = tx->
GetHash();
1670 if (m_pool.exists(wtxid)) {
1680 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1682 }
else if (m_pool.exists(txid)) {
1690 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1696 const auto single_package_res = AcceptSubPackage({tx},
args);
1697 const auto& single_res = single_package_res.m_tx_results.at(wtxid);
1701 assert(m_pool.exists(wtxid));
1702 results_final.emplace(wtxid, single_res);
1703 }
else if (package.size() == 1 ||
1717 individual_results_nonfinal.emplace(wtxid, single_res);
1719 individual_results_nonfinal.emplace(wtxid, single_res);
1720 txns_package_eval.push_back(tx);
1725 auto multi_submission_result = quit_early || txns_package_eval.empty() ?
PackageMempoolAcceptResult(package_state_quit_early, {}) :
1726 AcceptSubPackage(txns_package_eval,
args);
1732 ClearSubPackageState();
1739 for (
const auto& tx : package) {
1741 if (multi_submission_result.m_tx_results.contains(wtxid)) {
1743 Assume(!results_final.contains(wtxid));
1746 const auto& txresult = multi_submission_result.m_tx_results.at(wtxid);
1753 results_final.emplace(wtxid, txresult);
1755 }
else if (
const auto it{results_final.find(wtxid)}; it != results_final.end()) {
1759 Assume(!individual_results_nonfinal.contains(wtxid));
1761 if (!m_pool.exists(tx->
GetHash())) {
1766 results_final.erase(wtxid);
1769 }
else if (
const auto it{individual_results_nonfinal.find(wtxid)}; it != individual_results_nonfinal.end()) {
1772 results_final.emplace(wtxid, it->second);
1775 Assume(results_final.size() == package.size());
1782 int64_t accept_time,
bool bypass_limits,
bool test_accept)
1789 std::vector<COutPoint> coins_to_uncache;
1791 auto args = MemPoolAccept::ATMPArgs::SingleAccept(chainparams, accept_time, bypass_limits, coins_to_uncache, test_accept);
1792 MempoolAcceptResult result = MemPoolAccept(pool, active_chainstate).AcceptSingleTransactionAndCleanup(tx,
args);
1800 for (
const COutPoint& hashTx : coins_to_uncache)
1803 tx->GetHash().data(),
1814 const Package& package,
bool test_accept,
const std::optional<CFeeRate>& client_maxfeerate)
1817 assert(!package.empty());
1818 assert(std::all_of(package.cbegin(), package.cend(), [](
const auto& tx){return tx != nullptr;}));
1820 std::vector<COutPoint> coins_to_uncache;
1825 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(chainparams,
GetTime(), coins_to_uncache);
1826 return MemPoolAccept(pool, active_chainstate).AcceptMultipleTransactionsAndCleanup(package,
args);
1828 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(chainparams,
GetTime(), coins_to_uncache, client_maxfeerate);
1829 return MemPoolAccept(pool, active_chainstate).AcceptPackage(package,
args);
1834 if (test_accept || result.m_state.IsInvalid()) {
1835 for (
const COutPoint& hashTx : coins_to_uncache) {
1854 nSubsidy >>= halvings;
1859 : m_dbview{
std::move(db_params),
std::move(options)},
1860 m_catcherview(&m_dbview) {}
1862void CoinsViews::InitCache()
1865 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1866 m_connect_block_view = std::make_unique<CoinsViewOverlay>(&*m_cacheview);
1873 std::optional<uint256> from_snapshot_blockhash)
1874 : m_mempool(mempool),
1875 m_blockman(blockman),
1876 m_chainman(chainman),
1878 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1889const CBlockIndex* Chainstate::SnapshotBase()
const
1893 return m_cached_snapshot_base;
1898 if (!m_target_blockhash)
return nullptr;
1900 return m_cached_target_block;
1903void Chainstate::SetTargetBlock(
CBlockIndex* block)
1908 m_target_blockhash.reset();
1910 m_cached_target_block = block;
1913void Chainstate::SetTargetBlockHash(
uint256 block_hash)
1915 m_target_blockhash = block_hash;
1916 m_cached_target_block =
nullptr;
1920 size_t cache_size_bytes,
1927 .cache_bytes = cache_size_bytes,
1928 .memory_only = in_memory,
1929 .wipe_data = should_wipe,
1937void Chainstate::InitCoinsCache(
size_t cache_size_bytes)
1955 if (this->GetRole().historical) {
1960 LogWarning(
"Found invalid chain more than 6 blocks longer than our best chain. This could be due to database corruption or consensus incompatibility with peers.");
1963 _(
"Warning: Found invalid chain more than 6 blocks longer than our best chain. This could be due to database corruption or consensus incompatibility with peers."));
1976 SetBlockFailureFlags(pindexNew);
1981 LogInfo(
"%s: invalid block=%s height=%d log2_work=%f date=%s", __func__,
1986 LogInfo(
"%s: current best=%s height=%d log2_work=%f date=%s", __func__,
2024 if (
VerifyScript(scriptSig,
m_tx_out.
scriptPubKey, witness,
m_flags,
CachingTransactionSignatureChecker(
ptxTo,
nIn,
m_tx_out.
nValue,
cacheStore, *
m_signature_cache, *
txdata), &error)) {
2025 return std::nullopt;
2028 return std::make_pair(error, std::move(debug_str));
2033 : m_signature_cache{signature_cache_bytes}
2044 LogInfo(
"Using %zu MiB out of %zu MiB requested for script execution cache, able to store %zu elements",
2045 approx_size_bytes >> 20, script_execution_cache_bytes >> 20, num_elems);
2071 std::vector<CScriptCheck>* pvChecks)
2076 pvChecks->reserve(tx.
vin.size());
2093 std::vector<CTxOut> spent_outputs;
2094 spent_outputs.reserve(tx.
vin.size());
2096 for (
const auto& txin : tx.
vin) {
2100 spent_outputs.emplace_back(coin.
out);
2102 txdata.
Init(tx, std::move(spent_outputs));
2106 for (
unsigned int i = 0; i < tx.
vin.size(); i++) {
2117 pvChecks->emplace_back(std::move(
check));
2118 }
else if (
auto result =
check(); result.has_value()) {
2133 if (cacheFullScriptStore && !pvChecks) {
2161 if (undo.nHeight == 0) {
2167 undo.nHeight = alternate.
nHeight;
2192 LogError(
"DisconnectBlock(): failure reading undo data\n");
2196 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
2197 LogError(
"DisconnectBlock(): block and undo data inconsistent\n");
2207 bool fEnforceBIP30 = !((pindex->
nHeight==91722 && pindex->
GetBlockHash() ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
2208 (pindex->
nHeight==91812 && pindex->
GetBlockHash() ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"}));
2211 for (
int i = block.
vtx.size() - 1; i >= 0; i--) {
2215 bool is_bip30_exception = (is_coinbase && !fEnforceBIP30);
2219 for (
size_t o = 0; o < tx.
vout.size(); o++) {
2220 if (!tx.
vout[o].scriptPubKey.IsUnspendable()) {
2225 if (!is_bip30_exception) {
2236 LogError(
"DisconnectBlock(): transaction and undo data inconsistent\n");
2239 for (
unsigned int j = tx.
vin.size(); j > 0;) {
2310 const auto time_start{SteadyClock::now()};
2326 if (!
CheckBlock(block, state, params.GetConsensus(), !fJustCheck, !fJustCheck)) {
2338 uint256 hashPrevBlock = pindex->
pprev ==
nullptr ?
uint256() : pindex->pprev->GetBlockHash();
2345 if (block_hash == params.GetConsensus().hashGenesisBlock) {
2351 const char* script_check_reason;
2353 script_check_reason =
"assumevalid=0 (always verify)";
2355 constexpr int64_t TWO_WEEKS_IN_SECONDS{60 * 60 * 24 * 7 * 2};
2363 script_check_reason =
"assumevalid hash not in headers";
2364 }
else if (it->second.GetAncestor(pindex->
nHeight) != pindex) {
2365 script_check_reason = (pindex->
nHeight > it->second.nHeight) ?
"block height above assumevalid height" :
"block not in assumevalid chain";
2367 script_check_reason =
"block not in best header chain";
2369 script_check_reason =
"best header chainwork below minimumchainwork";
2371 script_check_reason =
"block too recent relative to best header";
2387 script_check_reason =
nullptr;
2391 const auto time_1{SteadyClock::now()};
2392 m_chainman.time_check += time_1 - time_start;
2394 Ticks<MillisecondsDouble>(time_1 - time_start),
2436 static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2468 fEnforceBIP30 = fEnforceBIP30 && (!pindexBIP34height || !(pindexBIP34height->
GetBlockHash() == params.GetConsensus().BIP34Hash));
2473 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2474 for (
const auto& tx : block.
vtx) {
2475 for (
size_t o = 0; o < tx->
vout.size(); o++) {
2478 "tried to overwrite transaction");
2485 int nLockTimeFlags = 0;
2493 const auto time_2{SteadyClock::now()};
2496 Ticks<MillisecondsDouble>(time_2 - time_1),
2500 const bool fScriptChecks{!!script_check_reason};
2502 if (script_check_reason != m_last_script_check_reason_logged && role.validated && !role.historical) {
2503 if (fScriptChecks) {
2504 LogInfo(
"Enabling script verification at block #%d (%s): %s.",
2505 pindex->
nHeight, block_hash.ToString(), script_check_reason);
2507 LogInfo(
"Disabling script verification at block #%d (%s).",
2508 pindex->
nHeight, block_hash.ToString());
2510 m_last_script_check_reason_logged = script_check_reason;
2520 std::vector<PrecomputedTransactionData> txsdata(block.
vtx.size());
2521 std::optional<CCheckQueueControl<CScriptCheck>> control;
2524 std::vector<int> prevheights;
2527 int64_t nSigOpsCost = 0;
2528 blockundo.
vtxundo.reserve(block.
vtx.size() - 1);
2529 for (
unsigned int i = 0; i < block.
vtx.size(); i++)
2534 nInputs += tx.
vin.size();
2550 "accumulated fee in the block out of range");
2557 prevheights.resize(tx.
vin.size());
2558 for (
size_t j = 0; j < tx.
vin.size(); j++) {
2562 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2581 bool fCacheResults = fJustCheck;
2587 std::vector<CScriptCheck> vChecks;
2589 if (tx_ok) control->Add(std::move(vChecks));
2603 blockundo.
vtxundo.emplace_back();
2607 const auto time_3{SteadyClock::now()};
2609 LogDebug(
BCLog::BENCH,
" - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) [%.2fs (%.2fms/blk)]\n", (
unsigned)block.
vtx.size(),
2610 Ticks<MillisecondsDouble>(time_3 - time_2), Ticks<MillisecondsDouble>(time_3 - time_2) / block.
vtx.size(),
2611 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_3 - time_2) / (nInputs - 1),
2612 Ticks<SecondsDouble>(
m_chainman.time_connect),
2616 if (block.
vtx[0]->GetValueOut() > blockReward && state.
IsValid()) {
2618 strprintf(
"coinbase pays too much (actual=%d vs limit=%d)", block.
vtx[0]->GetValueOut(), blockReward));
2621 auto parallel_result = control->Complete();
2622 if (parallel_result.has_value() && state.
IsValid()) {
2630 const auto time_4{SteadyClock::now()};
2632 LogDebug(
BCLog::BENCH,
" - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n", nInputs - 1,
2633 Ticks<MillisecondsDouble>(time_4 - time_2),
2634 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_4 - time_2) / (nInputs - 1),
2635 Ticks<SecondsDouble>(
m_chainman.time_verify),
2642 if (!
m_blockman.WriteBlockUndo(blockundo, state, *pindex)) {
2646 const auto time_5{SteadyClock::now()};
2649 Ticks<MillisecondsDouble>(time_5 - time_4),
2661 const auto time_6{SteadyClock::now()};
2664 Ticks<MillisecondsDouble>(time_6 - time_5),
2674 Ticks<std::chrono::nanoseconds>(time_5 - time_start)
2683 return this->GetCoinsCacheSizeState(
2689 size_t max_coins_cache_size_bytes,
2690 size_t max_mempool_size_bytes)
2695 int64_t nTotalSpace =
2696 max_coins_cache_size_bytes + std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2698 if (cacheSize > nTotalSpace) {
2699 LogInfo(
"Cache size (%s) exceeds total space (%s)\n", cacheSize, nTotalSpace);
2710 int nManualPruneHeight)
2714 std::set<int> setFilesToPrune;
2715 bool full_flush_completed =
false;
2722 bool fFlushForPrune =
false;
2729 std::optional<std::string> limiting_lock;
2731 for (
const auto& prune_lock :
m_blockman.m_prune_locks) {
2732 if (prune_lock.second.height_first == std::numeric_limits<int>::max())
continue;
2735 last_prune = std::max(1, std::min(last_prune, lock_height));
2736 if (last_prune == lock_height) {
2737 limiting_lock = prune_lock.first;
2741 if (limiting_lock) {
2742 LogDebug(
BCLog::PRUNE,
"%s limited pruning to height %d\n", limiting_lock.value(), last_prune);
2745 if (nManualPruneHeight > 0) {
2750 std::min(last_prune, nManualPruneHeight),
2758 if (!setFilesToPrune.empty()) {
2759 fFlushForPrune =
true;
2761 m_blockman.m_block_tree_db->WriteFlag(
"prunedblockfiles",
true);
2778 LogDebug(
BCLog::COINDB,
"Writing chainstate to disk: flush mode=%s, prune=%d, large=%d, critical=%d, periodic=%d",
2779 FlushStateModeNames[
size_t(mode)], fFlushForPrune, fCacheLarge, fCacheCritical, fPeriodicWrite);
2792 LogWarning(
"%s: Failed to flush block file.\n", __func__);
2803 if (fFlushForPrune) {
2809 if (!
CoinsTip().GetBestBlock().IsNull()) {
2820 full_flush_completed =
true;
2822 int64_t{Ticks<std::chrono::microseconds>(
NodeClock::now() - nNow)},
2824 (uint64_t)coins_count,
2825 (uint64_t)coins_mem_usage,
2826 (
bool)fFlushForPrune);
2830 if (should_write ||
m_next_write == NodeClock::time_point::max()) {
2835 if (full_flush_completed) {
2844 }
catch (
const std::exception& e) {
2845 LogWarning(
"Failed to start chainstate compaction (%s)", e.what());
2849 }
catch (
const std::runtime_error& e) {
2876 const std::string& func_name,
2877 const std::string&
prefix,
2878 const std::string& warning_messages,
2890 background_validation ? chainman.GetBackgroundVerificationProgress(*tip) : chainman.GuessVerificationProgress(tip),
2893 !warning_messages.empty() ?
strprintf(
" warning='%s'", warning_messages) :
"");
2896void Chainstate::UpdateTip(
const CBlockIndex* pindexNew)
2899 const auto& coins_tip = this->
CoinsTip();
2905 constexpr int BACKGROUND_LOG_INTERVAL = 2000;
2906 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2917 std::vector<bilingual_str> warning_messages;
2920 for (
auto [bit, active] : bits) {
2925 warning_messages.push_back(warning);
2952 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2955 LogError(
"DisconnectTip(): Failed to read block\n");
2959 const auto time_start{SteadyClock::now()};
2963 if (DisconnectBlock(block, pindexDelete, view) !=
DISCONNECT_OK) {
2970 Ticks<MillisecondsDouble>(SteadyClock::now() - time_start));
2974 const int max_height_first{pindexDelete->
nHeight - 1};
2975 for (
auto& prune_lock :
m_blockman.m_prune_locks) {
2976 if (prune_lock.second.height_first <= max_height_first)
continue;
2978 prune_lock.second.height_first = max_height_first;
2979 LogDebug(
BCLog::PRUNE,
"%s prune lock moved back to %d\n", prune_lock.first, max_height_first);
2999 UpdateTip(pindexDelete->
pprev);
3022 std::shared_ptr<const CBlock> block_to_connect,
3023 std::vector<ConnectedBlock>& connected_blocks,
3031 const auto time_1{SteadyClock::now()};
3032 if (!block_to_connect) {
3033 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
3037 block_to_connect = std::move(pblockNew);
3042 const auto time_2{SteadyClock::now()};
3043 SteadyClock::time_point time_3;
3047 Ticks<MillisecondsDouble>(time_2 - time_1));
3051 bool rv =
ConnectBlock(*block_to_connect, state, pindexNew, view);
3061 time_3 = SteadyClock::now();
3062 m_chainman.time_connect_total += time_3 - time_2;
3065 Ticks<MillisecondsDouble>(time_3 - time_2),
3066 Ticks<SecondsDouble>(
m_chainman.time_connect_total),
3070 const auto time_4{SteadyClock::now()};
3073 Ticks<MillisecondsDouble>(time_4 - time_3),
3080 const auto time_5{SteadyClock::now()};
3081 m_chainman.time_chainstate += time_5 - time_4;
3083 Ticks<MillisecondsDouble>(time_5 - time_4),
3084 Ticks<SecondsDouble>(
m_chainman.time_chainstate),
3094 UpdateTip(pindexNew);
3096 const auto time_6{SteadyClock::now()};
3097 m_chainman.time_post_connect += time_6 - time_5;
3100 Ticks<MillisecondsDouble>(time_6 - time_5),
3101 Ticks<SecondsDouble>(
m_chainman.time_post_connect),
3104 Ticks<MillisecondsDouble>(time_6 - time_1),
3118 m_chainman.MaybeValidateSnapshot(*
this, current_cs);
3120 connected_blocks.emplace_back(pindexNew, std::move(block_to_connect));
3144 bool fInvalidAncestor =
false;
3146 assert(pindexTest->HaveNumChainTxs() || pindexTest->nHeight == 0);
3154 if (fFailedChain || fMissingData) {
3160 for (
CBlockIndex *pindexFailed = pindexNew; pindexFailed != pindexTest; pindexFailed = pindexFailed->
pprev) {
3164 if (fMissingData && !fFailedChain) {
3174 fInvalidAncestor =
true;
3178 if (!fInvalidAncestor)
3210 bool fBlocksDisconnected =
false;
3224 fBlocksDisconnected =
true;
3228 std::vector<CBlockIndex*> vpindexToConnect;
3229 bool fContinue =
true;
3234 int nTargetHeight = std::min(
nHeight + 32, index_most_work.
nHeight);
3235 vpindexToConnect.clear();
3236 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3239 vpindexToConnect.push_back(pindexIter);
3240 pindexIter = pindexIter->
pprev;
3245 for (
CBlockIndex* pindexConnect : vpindexToConnect | std::views::reverse) {
3246 if (!
ConnectTip(state, pindexConnect, pindexConnect == &index_most_work ? pblock : std::shared_ptr<const CBlock>(), connected_blocks, disconnectpool)) {
3253 fInvalidFound =
true;
3274 if (fBlocksDisconnected) {
3299 LogInfo(
"Leaving InitialBlockDownload (latching to false)");
3305 bool fNotify =
false;
3306 bool fInitialBlockDownload =
false;
3310 pindexHeader = m_best_header;
3312 if (pindexHeader != m_last_notified_header) {
3315 m_last_notified_header = pindexHeader;
3328 if (signals.CallbacksPending() > 10) {
3329 signals.SyncWithValidationInterfaceQueue();
3333bool Chainstate::ActivateBestChain(
BlockValidationState& state, std::shared_ptr<const CBlock> pblock)
3358 bool exited_ibd{
false};
3375 bool blocks_connected =
false;
3379 std::vector<ConnectedBlock> connected_blocks;
3381 if (pindexMostWork ==
nullptr) {
3386 if (pindexMostWork ==
nullptr || pindexMostWork ==
m_chain.
Tip()) {
3390 bool fInvalidFound =
false;
3391 std::shared_ptr<const CBlock> nullBlockPtr;
3396 if (!
ActivateBestChainStep(state, *pindexMostWork, pblock && pblock->GetHash() == pindexMostWork->
GetBlockHash() ? pblock : nullBlockPtr, fInvalidFound, connected_blocks)) {
3400 blocks_connected =
true;
3402 if (fInvalidFound) {
3404 pindexMostWork =
nullptr;
3408 for (
auto& [index, block] : std::move(connected_blocks)) {
3419 if (!blocks_connected)
return true;
3424 if (was_in_ibd && !still_in_ibd) {
3457 bool reached_target;
3474 if (reached_target) {
3493 }
while (pindexNewTip != pindexMostWork);
3528 return ActivateBestChain(state, std::shared_ptr<const CBlock>());
3538 if (pindex->
nHeight == 0)
return false;
3552 std::multimap<const arith_uint256, CBlockIndex*> highpow_outofchain_headers;
3556 for (
auto& entry :
m_blockman.m_block_index) {
3566 highpow_outofchain_headers.insert({candidate.
nChainWork, &candidate});
3572 bool pindex_was_in_chain =
false;
3573 int disconnected = 0;
3587 pindex_was_in_chain =
true;
3600 if (!
ret)
return false;
3602 assert(disconnected_tip->pprev == new_tip);
3617 auto candidate_it = highpow_outofchain_headers.lower_bound(new_tip->nChainWork);
3619 const bool best_header_needs_update{
m_chainman.m_best_header->GetAncestor(disconnected_tip->nHeight) == disconnected_tip};
3620 if (best_header_needs_update) {
3625 while (candidate_it != highpow_outofchain_headers.end()) {
3627 if (candidate->
GetAncestor(disconnected_tip->nHeight) == disconnected_tip) {
3633 candidate_it = highpow_outofchain_headers.erase(candidate_it);
3643 if (best_header_needs_update &&
3651 to_mark_failed = disconnected_tip;
3677 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3687 if (pindex_was_in_chain) {
3697 *to_mark_failed->
pprev,
3709void Chainstate::SetBlockFailureFlags(
CBlockIndex* invalid_block)
3713 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3714 if (invalid_block != &block_index && block_index.GetAncestor(invalid_block->
nHeight) == invalid_block) {
3727 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3729 block_index.nStatus &= ~BLOCK_FAILED_VALID;
3734 if (&block_index ==
m_chainman.m_best_invalid) {
3761 if (!target_block) {
3768 if (target_block->GetAncestor(pindex->
nHeight) == pindex) {
3778 pindexNew->
nTx = block.
vtx.size();
3784 auto prev_tx_sum = [](
CBlockIndex& block) {
return block.nTx + (block.pprev ? block.pprev->m_chain_tx_count : 0); };
3787 LogWarning(
"Internal bug detected: block %d has unexpected m_chain_tx_count %i that should be %i (%s %s). Please report this issue here: %s\n",
3791 pindexNew->nFile = pos.
nFile;
3792 pindexNew->nDataPos = pos.
nPos;
3793 pindexNew->nUndoPos = 0;
3803 std::deque<CBlockIndex*> queue;
3804 queue.push_back(pindexNew);
3807 while (!queue.empty()) {
3815 LogWarning(
"Internal bug detected: block %d has unexpected m_chain_tx_count %i that should be %i (%s %s). Please report this issue here: %s\n",
3820 for (
const auto& c : m_chainstates) {
3821 c->TryAddBlockIndexCandidate(pindex);
3823 std::pair<std::multimap<CBlockIndex*, CBlockIndex*>::iterator, std::multimap<CBlockIndex*, CBlockIndex*>::iterator> range =
m_blockman.
m_blocks_unlinked.equal_range(pindex);
3824 while (range.first != range.second) {
3825 std::multimap<CBlockIndex*, CBlockIndex*>::iterator it = range.first;
3826 queue.push_back(it->second);
3857 "hashMerkleRoot mismatch");
3866 "bad-txns-duplicate",
3867 "duplicate transaction");
3882 if (expect_witness_commitment) {
3887 assert(!block.
vtx.empty() && !block.
vtx[0]->vin.empty());
3888 const auto& witness_stack{block.
vtx[0]->vin[0].scriptWitness.stack};
3890 if (witness_stack.size() != 1 || witness_stack[0].size() != 32) {
3893 "bad-witness-nonce-size",
3894 strprintf(
"%s : invalid witness reserved value size", __func__));
3903 if (memcmp(hash_witness.
begin(), &block.
vtx[0]->vout[commitpos].scriptPubKey[6], 32)) {
3906 "bad-witness-merkle-match",
3907 strprintf(
"%s : witness merkle commitment mismatch", __func__));
3916 for (
const auto& tx : block.
vtx) {
3920 "unexpected-witness",
3921 strprintf(
"%s : unexpected witness data found", __func__));
3961 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase())
3963 for (
unsigned int i = 1; i < block.
vtx.size(); i++)
3964 if (block.
vtx[i]->IsCoinBase())
3969 for (
const auto& tx : block.
vtx) {
3981 unsigned int nSigOps = 0;
3982 for (
const auto& tx : block.
vtx)
3989 if (fCheckPOW && fCheckMerkleRoot)
3998 static const std::vector<unsigned char>
nonce(32, 0x00);
4001 tx.
vin[0].scriptWitness.stack.resize(1);
4002 tx.
vin[0].scriptWitness.stack[0] =
nonce;
4010 std::vector<unsigned char>
ret(32, 0x00);
4018 out.scriptPubKey[1] = 0x24;
4019 out.scriptPubKey[2] = 0xaa;
4020 out.scriptPubKey[3] = 0x21;
4021 out.scriptPubKey[4] = 0xa9;
4022 out.scriptPubKey[5] = 0xed;
4023 memcpy(&
out.scriptPubKey[6], witnessroot.
begin(), 32);
4033 return std::ranges::all_of(
headers,
4034 [&](
const auto& header) {
return CheckProofOfWork(header.GetHash(), header.nBits, consensusParams); });
4045 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase()) {
4052 return std::any_of(block.
vtx.begin(), block.
vtx.end(),
4053 [](
auto& tx) { return GetSerializeSize(TX_NO_WITNESS(tx)) == 64; });
4093 assert(pindexPrev !=
nullptr);
4094 const int nHeight = pindexPrev->nHeight + 1;
4102 if (block.
GetBlockTime() <= pindexPrev->GetMedianTimePast())
4141 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4144 bool enforce_locktime_median_time_past{
false};
4146 assert(pindexPrev !=
nullptr);
4147 enforce_locktime_median_time_past =
true;
4150 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past ?
4155 for (
const auto& tx : block.
vtx) {
4165 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4166 !std::equal(
expect.begin(),
expect.end(), block.
vtx[0]->vin[0].scriptSig.begin())) {
4202 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4204 if (miSelf !=
m_blockman.m_block_index.end()) {
4229 pindexPrev = &((*mi).second);
4239 if (!min_pow_checked) {
4274 blocks_left = std::max<int64_t>(0, blocks_left);
4275 const double progress{100.0 * last_accepted.nHeight / (last_accepted.nHeight + blocks_left)};
4276 LogInfo(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n", last_accepted.nHeight, progress);
4294 if (now < m_last_presync_update + std::chrono::milliseconds{250})
return;
4295 m_last_presync_update = now;
4299 if (initial_download) {
4301 blocks_left = std::max<int64_t>(0, blocks_left);
4302 const double progress{100.0 * height / (height + blocks_left)};
4303 LogInfo(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n", height, progress);
4310 const CBlock& block = *pblock;
4312 if (fNewBlock) *fNewBlock =
false;
4316 CBlockIndex *&pindex = ppindex ? *ppindex : pindexDummy;
4318 bool accepted_header{
AcceptBlockHeader(block, state, &pindex, min_pow_checked)};
4321 if (!accepted_header)
4345 if (fAlreadyHave)
return true;
4347 if (pindex->
nTx != 0)
return true;
4348 if (!fHasMoreOrSameWork)
return true;
4349 if (fTooFarAhead)
return true;
4360 if (!
CheckBlock(block, state, params.GetConsensus()) ||
4376 if (fNewBlock) *fNewBlock =
true;
4384 if (blockPos.IsNull()) {
4385 state.
Error(
strprintf(
"%s: Failed to find position to write new block to disk", __func__));
4390 }
catch (
const std::runtime_error& e) {
4414 if (new_block) *new_block =
false;
4429 ret =
AcceptBlock(block, state, &pindex, force_processing,
nullptr, new_block, min_pow_checked);
4444 LogError(
"%s: ActivateBestChain failed (%s)\n", __func__, state.
ToString());
4450 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
4451 LogError(
"%s: [background] ActivateBestChain failed (%s)\n", __func__, bg_state.
ToString());
4476 const bool check_pow,
4477 const bool check_merkle_root)
4488 state.
Invalid({},
"inconclusive-not-best-prevblk");
4529 index_dummy.pprev = tip;
4531 index_dummy.phashBlock = &block_hash;
4535 if(!chainstate.
ConnectBlock(block, state, &index_dummy, view_dummy,
true)) {
4579 assert(
cs->setBlockIndexCandidates.empty());
4587 target = target->pprev;
4590 LogInfo(
"Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f",
4597 if (!this->GetRole().historical) {
4625 int nCheckLevel,
int nCheckDepth)
4634 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
4637 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
4638 LogInfo(
"Verifying last %i blocks at level %i", nCheckDepth, nCheckLevel);
4642 int nGoodTransactions = 0;
4645 bool skipped_no_block_data{
false};
4646 bool skipped_l3_checks{
false};
4647 LogInfo(
"Verification progress: 0%%");
4652 const int percentageDone = std::max(1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (
double)nCheckDepth * (nCheckLevel >= 4 ? 50 : 100))));
4653 if (reportDone < percentageDone / 10) {
4655 LogInfo(
"Verification progress: %d%%", percentageDone);
4656 reportDone = percentageDone / 10;
4665 LogInfo(
"Block verification stopping at height %d (no data). This could be due to pruning or use of an assumeutxo snapshot.", pindex->
nHeight);
4666 skipped_no_block_data =
true;
4676 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensus_params)) {
4677 LogError(
"Verification error: found bad block at %d, hash=%s (%s)",
4682 if (nCheckLevel >= 2 && pindex) {
4694 if (nCheckLevel >= 3) {
4703 nGoodTransactions = 0;
4704 pindexFailure = pindex;
4706 nGoodTransactions += block.
vtx.size();
4709 skipped_l3_checks =
true;
4714 if (pindexFailure) {
4715 LogError(
"Verification error: coin database inconsistencies found (last %i blocks, %i good transactions before that)", chainstate.
m_chain.
Height() - pindexFailure->
nHeight + 1, nGoodTransactions);
4718 if (skipped_l3_checks) {
4719 LogWarning(
"Skipped verification of level >=3 (insufficient database cache size). Consider increasing -dbcache.");
4726 if (nCheckLevel >= 4 && !skipped_l3_checks) {
4728 const int percentageDone = std::max(1, std::min(99, 100 - (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (
double)nCheckDepth * 50)));
4729 if (reportDone < percentageDone / 10) {
4731 LogInfo(
"Verification progress: %d%%", percentageDone);
4732 reportDone = percentageDone / 10;
4741 if (!chainstate.
ConnectBlock(block, state, pindex, coins)) {
4749 LogInfo(
"Verification: checked last %i blocks at level %i", block_count, nCheckLevel);
4750 if (nCheckLevel >= 3 && !skipped_l3_checks) {
4751 LogInfo(
"Verification: no coin database inconsistencies (%i transactions)", nGoodTransactions);
4754 if (skipped_l3_checks) {
4757 if (skipped_no_block_data) {
4775 if (!tx->IsCoinBase()) {
4776 for (
const CTxIn &txin : tx->vin) {
4793 std::vector<uint256> hashHeads =
db.GetHeadBlocks();
4794 if (hashHeads.empty())
return true;
4795 if (hashHeads.size() != 2) {
4796 LogError(
"ReplayBlocks(): unknown inconsistent state\n");
4807 if (!
m_blockman.m_block_index.contains(hashHeads[0])) {
4808 LogError(
"ReplayBlocks(): reorganization to unknown block requested\n");
4811 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
4813 if (!hashHeads[1].IsNull()) {
4814 if (!
m_blockman.m_block_index.contains(hashHeads[1])) {
4815 LogError(
"ReplayBlocks(): reorganization from unknown block requested\n");
4818 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
4820 assert(pindexFork !=
nullptr);
4824 const int nForkHeight{pindexFork ? pindexFork->
nHeight : 0};
4825 if (pindexOld != pindexFork) {
4827 while (pindexOld != pindexFork) {
4834 if (pindexOld->
nHeight % 10'000 == 0) {
4847 pindexOld = pindexOld->
pprev;
4853 if (nForkHeight < pindexNew->
nHeight) {
4885 block = block->pprev;
4891void Chainstate::ClearBlockIndexCandidates()
4897void Chainstate::PopulateBlockIndexCandidates()
4905 if (pindex == SnapshotBase() ||
4919 if (!
ret)
return false;
4921 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
4923 std::vector<CBlockIndex*> vSortedByHeight{
m_blockman.GetAllBlockIndices()};
4924 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
4930 m_best_invalid = pindex;
4933 m_best_header = pindex;
4949 if (
m_blockman.m_block_index.contains(params.GenesisBlock().GetHash()))
4953 const CBlock& block = params.GenesisBlock();
4955 if (blockPos.IsNull()) {
4956 LogError(
"%s: writing genesis block to disk failed\n", __func__);
4961 }
catch (
const std::runtime_error& e) {
4962 LogError(
"%s: failed to write genesis block: %s\n", __func__, e.what());
4972 std::multimap<uint256, FlatFilePos>* blocks_with_unknown_parent)
4975 assert(!dbp == !blocks_with_unknown_parent);
4977 const auto start{SteadyClock::now()};
4985 uint64_t nRewind = blkdat.GetPos();
4986 while (!blkdat.eof()) {
4989 blkdat.SetPos(nRewind);
4992 unsigned int nSize = 0;
4996 blkdat.FindByte(std::byte(params.MessageStart()[0]));
4997 nRewind = blkdat.GetPos() + 1;
4999 if (buf != params.MessageStart()) {
5006 }
catch (
const std::exception&) {
5013 const uint64_t nBlockPos{blkdat.GetPos()};
5015 dbp->
nPos = nBlockPos;
5016 blkdat.SetLimit(nBlockPos + nSize);
5022 nRewind = nBlockPos + nSize;
5023 blkdat.SkipTo(nRewind);
5025 std::shared_ptr<CBlock> pblock{};
5033 if (dbp && blocks_with_unknown_parent) {
5034 blocks_with_unknown_parent->emplace(header.
hashPrevBlock, *dbp);
5043 blkdat.SetPos(nBlockPos);
5044 pblock = std::make_shared<CBlock>();
5046 nRewind = blkdat.GetPos();
5049 if (
AcceptBlock(pblock, state,
nullptr,
true, dbp,
nullptr,
true)) {
5055 }
else if (hash != params.GetConsensus().hashGenesisBlock && pindex->
nHeight % 1000 == 0) {
5082 if (
auto result{ActivateBestChains()}; !result) {
5090 if (!blocks_with_unknown_parent)
continue;
5093 std::deque<uint256> queue;
5094 queue.push_back(hash);
5095 while (!queue.empty()) {
5098 auto range = blocks_with_unknown_parent->equal_range(head);
5099 while (range.first != range.second) {
5100 std::multimap<uint256, FlatFilePos>::iterator it = range.first;
5101 std::shared_ptr<CBlock> pblockrecursive = std::make_shared<CBlock>();
5103 const auto& block_hash{pblockrecursive->GetHash()};
5107 if (
AcceptBlock(pblockrecursive, dummy,
nullptr,
true, &it->second,
nullptr,
true)) {
5109 queue.push_back(block_hash);
5113 blocks_with_unknown_parent->erase(it);
5117 }
catch (
const std::exception& e) {
5129 LogDebug(
BCLog::REINDEX,
"%s: unexpected data at file offset 0x%x - %s. continuing\n", __func__, (nRewind - 1), e.what());
5132 }
catch (
const std::runtime_error& e) {
5135 LogInfo(
"Loaded %i blocks from external file in %dms", nLoaded, Ticks<std::chrono::milliseconds>(SteadyClock::now() - start));
5170 best_hdr_chain.
SetTip(*m_best_header);
5172 std::multimap<const CBlockIndex*, const CBlockIndex*> forward;
5173 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
5175 if (!best_hdr_chain.
Contains(block_index)) {
5177 assert(block_index.pprev);
5178 forward.emplace(block_index.pprev, &block_index);
5192 const CBlockIndex* pindexFirstNeverProcessed =
nullptr;
5193 const CBlockIndex* pindexFirstNotTreeValid =
nullptr;
5194 const CBlockIndex* pindexFirstNotTransactionsValid =
nullptr;
5195 const CBlockIndex* pindexFirstNotChainValid =
nullptr;
5196 const CBlockIndex* pindexFirstNotScriptsValid =
nullptr;
5203 const CBlockIndex *snap_first_missing{}, *snap_first_notx{}, *snap_first_notv{}, *snap_first_nocv{}, *snap_first_nosv{};
5204 auto snap_update_firsts = [&] {
5205 if (pindex == snap_base) {
5206 std::swap(snap_first_missing, pindexFirstMissing);
5207 std::swap(snap_first_notx, pindexFirstNeverProcessed);
5208 std::swap(snap_first_notv, pindexFirstNotTransactionsValid);
5209 std::swap(snap_first_nocv, pindexFirstNotChainValid);
5210 std::swap(snap_first_nosv, pindexFirstNotScriptsValid);
5214 while (pindex !=
nullptr) {
5216 if (pindexFirstInvalid ==
nullptr && pindex->nStatus &
BLOCK_FAILED_VALID) pindexFirstInvalid = pindex;
5217 if (pindexFirstMissing ==
nullptr && !(pindex->nStatus &
BLOCK_HAVE_DATA)) {
5218 pindexFirstMissing = pindex;
5220 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) pindexFirstNeverProcessed = pindex;
5223 if (pindex->
pprev !=
nullptr) {
5224 if (pindexFirstNotTransactionsValid ==
nullptr &&
5226 pindexFirstNotTransactionsValid = pindex;
5229 if (pindexFirstNotChainValid ==
nullptr &&
5231 pindexFirstNotChainValid = pindex;
5234 if (pindexFirstNotScriptsValid ==
nullptr &&
5236 pindexFirstNotScriptsValid = pindex;
5241 if (pindex->
pprev ==
nullptr) {
5244 for (
const auto& c : m_chainstates) {
5245 if (c->m_chain.Genesis() !=
nullptr) {
5246 assert(pindex == c->m_chain.Genesis());
5258 assert(pindexFirstMissing == pindexFirstNeverProcessed);
5264 if (snap_base && snap_base->GetAncestor(pindex->
nHeight) == pindex) {
5274 assert((pindexFirstNotTransactionsValid ==
nullptr || pindex == snap_base) == pindex->
HaveNumChainTxs());
5278 assert(pindexFirstNotTreeValid ==
nullptr);
5282 if (pindexFirstInvalid ==
nullptr) {
5289 if (!pindex->
pprev) {
5304 for (
const auto& c : m_chainstates) {
5305 if (c->m_chain.Tip() ==
nullptr)
continue;
5319 if (!
CBlockIndexWorkComparator()(pindex, c->m_chain.Tip()) && (pindexFirstNeverProcessed ==
nullptr || pindex == snap_base)) {
5323 if (pindexFirstInvalid ==
nullptr) {
5342 if (pindexFirstMissing ==
nullptr || pindex == c->m_chain.Tip() || pindex == c->SnapshotBase()) {
5348 if (!c->TargetBlock() || c->TargetBlock()->GetAncestor(pindex->
nHeight) == pindex) {
5349 assert(c->setBlockIndexCandidates.contains(pindex));
5357 assert(!c->setBlockIndexCandidates.contains(pindex));
5362 bool foundInUnlinked =
false;
5363 for (
auto it = rangeUnlinked.first; it != rangeUnlinked.second; ++it) {
5365 if (it->second == pindex) {
5366 assert(!foundInUnlinked);
5367 foundInUnlinked =
true;
5370 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed !=
nullptr && pindexFirstInvalid ==
nullptr) {
5375 if (pindexFirstMissing ==
nullptr)
assert(!foundInUnlinked);
5376 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed ==
nullptr && pindexFirstMissing !=
nullptr) {
5387 for (
const auto& c : m_chainstates) {
5389 if (pindexFirstInvalid ==
nullptr) {
5390 if (!c->TargetBlock() || c->TargetBlock()->GetAncestor(pindex->
nHeight) == pindex) {
5402 snap_update_firsts();
5403 auto range{forward.equal_range(pindex)};
5404 if (range.first != range.second) {
5406 pindex = range.first->second;
5409 }
else if (best_hdr_chain.
Contains(*pindex)) {
5412 pindex = best_hdr_chain[
nHeight];
5420 snap_update_firsts();
5422 if (pindex == pindexFirstInvalid) pindexFirstInvalid =
nullptr;
5423 if (pindex == pindexFirstMissing) pindexFirstMissing =
nullptr;
5424 if (pindex == pindexFirstNeverProcessed) pindexFirstNeverProcessed =
nullptr;
5425 if (pindex == pindexFirstNotTreeValid) pindexFirstNotTreeValid =
nullptr;
5426 if (pindex == pindexFirstNotTransactionsValid) pindexFirstNotTransactionsValid =
nullptr;
5427 if (pindex == pindexFirstNotChainValid) pindexFirstNotChainValid =
nullptr;
5428 if (pindex == pindexFirstNotScriptsValid) pindexFirstNotScriptsValid =
nullptr;
5432 auto rangePar{forward.equal_range(pindexPar)};
5433 while (rangePar.first->second != pindex) {
5434 assert(rangePar.first != rangePar.second);
5439 if (rangePar.first != rangePar.second) {
5441 pindex = rangePar.first->second;
5443 }
else if (pindexPar == best_hdr_chain[
nHeight - 1]) {
5445 pindex = best_hdr_chain[
nHeight];
5447 assert((pindex ==
nullptr) == (pindexPar == best_hdr_chain.
Tip()));
5459 assert(nNodes == forward.size() + best_hdr_chain.
Height() + 1);
5466 return strprintf(
"Chainstate [%s] @ height %d (%s)",
5471bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size)
5484 LogInfo(
"[%s] resized coinsdb cache to %.1f MiB",
5485 this->
ToString(), coinsdb_size /
double(1_MiB));
5486 LogInfo(
"[%s] resized coinstip cache to %.1f MiB",
5487 this->
ToString(), coinstip_size /
double(1_MiB));
5492 if (coinstip_size > old_coinstip_size) {
5506 if (pindex ==
nullptr) {
5515 const int64_t nNow{TicksSinceEpoch<std::chrono::seconds>(
NodeClock::now())};
5516 const auto block_time{
5517 (
Assume(m_best_header) && std::abs(nNow - pindex->
GetBlockTime()) <= Ticks<std::chrono::seconds>(2h) &&
5531 fTxTotal =
data.tx_count + (nNow -
data.nTime) *
data.dTxRate;
5549 return static_cast<double>(pindex.
m_chain_tx_count) /
static_cast<double>(target_block->m_chain_tx_count);
5555 assert(m_chainstates.empty());
5556 m_chainstates.emplace_back(std::make_unique<Chainstate>(mempool,
m_blockman, *
this));
5557 return *m_chainstates.back();
5569 bool existed = fs::remove(base_blockhash_path);
5571 LogWarning(
"[snapshot] snapshot chainstate dir being removed lacks %s file",
5574 }
catch (
const fs::filesystem_error& e) {
5575 LogWarning(
"[snapshot] failed to remove file %s: %s\n",
5581 LogInfo(
"Removing leveldb dir at %s\n", path_str);
5585 const bool destroyed =
DestroyDB(path_str);
5588 LogError(
"leveldb DestroyDB call failed on %s", path_str);
5615 if (!
GetParams().AssumeutxoForBlockhash(base_blockhash).has_value()) {
5617 std::string heights_formatted =
util::Join(available_heights,
", ", [&](
const auto& i) {
return util::ToString(i); });
5618 return util::Error{
Untranslated(
strprintf(
"assumeutxo block hash in snapshot metadata not recognized (hash: %s). The following snapshot heights are available: %s",
5620 heights_formatted))};
5624 if (!snapshot_start_block) {
5625 return util::Error{
Untranslated(
strprintf(
"The base block header (%s) must appear in the headers chain. Make sure all headers are syncing, and call loadtxoutset again",
5630 if (start_block_invalid) {
5634 if (!m_best_header || m_best_header->GetAncestor(snapshot_start_block->nHeight) != snapshot_start_block) {
5635 return util::Error{
Untranslated(
"A forked headers-chain with more work than the chain with the snapshot base block header exists. Please proceed to sync without AssumeUtxo.")};
5639 if (mempool && mempool->
size() > 0) {
5644 int64_t current_coinsdb_cache_size{0};
5645 int64_t current_coinstip_cache_size{0};
5653 static constexpr double IBD_CACHE_PERC = 0.01;
5654 static constexpr double SNAPSHOT_CACHE_PERC = 0.99;
5672 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
5673 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
5677 return std::make_unique<Chainstate>(
5678 nullptr,
m_blockman, *
this, base_blockhash));
5682 snapshot_chainstate->InitCoinsDB(
5683 static_cast<size_t>(current_coinsdb_cache_size * SNAPSHOT_CACHE_PERC),
5685 snapshot_chainstate->InitCoinsCache(
5686 static_cast<size_t>(current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
5690 this->MaybeRebalanceCaches();
5698 snapshot_chainstate.reset();
5702 "Manually remove it before restarting.\n"), fs::PathToString(*snapshot_datadir)));
5719 return cleanup_bad_snapshot(
Untranslated(
"work does not exceed active chainstate"));
5725 return cleanup_bad_snapshot(
Untranslated(
"could not write base blockhash"));
5729 Chainstate& chainstate{AddChainstate(std::move(snapshot_chainstate))};
5732 chainstate.PopulateBlockIndexCandidates();
5734 LogInfo(
"[snapshot] successfully activated snapshot %s", base_blockhash.
ToString());
5735 LogInfo(
"[snapshot] (%.2f MB)",
5736 chainstate.CoinsTip().DynamicMemoryUsage() / (1000 * 1000));
5738 this->MaybeRebalanceCaches();
5739 return snapshot_start_block;
5746 snapshot_loaded ?
"saving snapshot chainstate" :
"flushing coins cache",
5750 coins_cache.
Flush();
5755 const char*
what() const noexcept
override
5757 return "ComputeUTXOStats interrupted.";
5779 if (!snapshot_start_block) {
5786 int base_height = snapshot_start_block->
nHeight;
5789 if (!maybe_au_data) {
5791 "(%d) - refusing to load snapshot", base_height))};
5806 LogInfo(
"[snapshot] loading %d coins from snapshot %s", coins_left, base_blockhash.
ToString());
5807 int64_t coins_processed{0};
5809 while (coins_left > 0) {
5813 size_t coins_per_txid{0};
5816 if (coins_per_txid > coins_left) {
5820 for (
size_t i = 0; i < coins_per_txid; i++) {
5824 outpoint.
hash = txid;
5826 if (coin.
nHeight > base_height ||
5827 outpoint.
n >= std::numeric_limits<
decltype(outpoint.
n)>::max()
5830 coins_count - coins_left))};
5834 coins_count - coins_left))};
5841 if (coins_processed % 1000000 == 0) {
5842 LogInfo(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)",
5844 static_cast<float>(coins_processed) * 100 /
static_cast<float>(coins_count),
5852 if (coins_processed % 120000 == 0) {
5858 return snapshot_chainstate.GetCoinsCacheSizeState());
5871 }
catch (
const std::ios_base::failure&) {
5884 bool out_of_coins{
false};
5886 std::byte left_over_byte;
5887 coins_file >> left_over_byte;
5888 }
catch (
const std::ios_base::failure&) {
5890 out_of_coins =
true;
5892 if (!out_of_coins) {
5897 LogInfo(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s",
5911 std::optional<CCoinsStats> maybe_stats;
5919 if (!maybe_stats.has_value()) {
5940 constexpr int AFTER_GENESIS_START{1};
5942 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height(); ++i) {
5943 index = snapshot_chainstate.
m_chain[i];
5960 assert(index == snapshot_start_block);
5963 LogInfo(
"[snapshot] validated snapshot (%.2f MB)",
5990 !validated_cs.m_target_blockhash ||
6001 "%s failed to validate the -assumeutxo snapshot state. "
6002 "This indicates a hardware problem, or a bug in the software, or a "
6003 "bad software modification that allowed an invalid snapshot to be "
6004 "loaded. As a result of this, the node will shut down and stop using any "
6005 "state that was built on the snapshot, resetting the chain height "
6006 "from %d to %d. On the next "
6007 "restart, the node will resume syncing from %d "
6008 "without using any snapshot data. "
6009 "Please report this incident to %s, including how you obtained the snapshot. "
6010 "The invalid snapshot chainstate will be left on disk in case it is "
6011 "helpful in diagnosing the issue that caused this error."),
6017 LogError(
"[snapshot] deleting snapshot, reverting to validated chain, and stopping node\n");
6020 validated_cs.SetTargetBlock(
nullptr);
6024 auto rename_result = unvalidated_cs.InvalidateCoinsDBOnDisk();
6025 if (!rename_result) {
6036 if (!maybe_au_data) {
6037 LogWarning(
"[snapshot] assumeutxo data not found for height "
6038 "(%d) - refusing to validate snapshot", validated_cs.
m_chain.
Height());
6039 handle_invalid_snapshot();
6044 std::optional<CCoinsStats> validated_cs_stats;
6045 LogInfo(
"[snapshot] computing UTXO stats for background chainstate to validate "
6046 "snapshot - this could take a few minutes");
6049 CoinStatsHashType::HASH_SERIALIZED,
6050 &validated_coins_db,
6058 if (!validated_cs_stats) {
6059 LogWarning(
"[snapshot] failed to generate stats for validation coins db");
6063 handle_invalid_snapshot();
6074 LogWarning(
"[snapshot] hash mismatch: actual=%s, expected=%s",
6075 validated_cs_stats->hashSerialized.ToString(),
6077 handle_invalid_snapshot();
6081 LogInfo(
"[snapshot] snapshot beginning at %s has been fully validated",
6085 validated_cs.m_target_utxohash =
AssumeutxoHash{validated_cs_stats->hashSerialized};
6086 this->MaybeRebalanceCaches();
6097void ChainstateManager::MaybeRebalanceCaches()
6102 if (!historical_cs && !current_cs.m_from_snapshot_blockhash) {
6106 }
else if (!historical_cs) {
6108 LogInfo(
"[snapshot] allocating all cache to the snapshot chainstate");
6116 historical_cs->ResizeCoinsCaches(
6118 current_cs.ResizeCoinsCaches(
6121 current_cs.ResizeCoinsCaches(
6123 historical_cs->ResizeCoinsCaches(
6129void ChainstateManager::ResetChainstates()
6131 m_chainstates.clear();
6141 if (!opts.check_block_index.has_value()) opts.
check_block_index = opts.chainparams.DefaultConsistencyChecks();
6142 if (!opts.minimum_chain_work.has_value()) opts.minimum_chain_work =
UintToArith256(opts.chainparams.GetConsensus().nMinimumChainWork);
6143 if (!opts.assumed_valid_block.has_value()) opts.assumed_valid_block = opts.chainparams.GetConsensus().defaultAssumeValid;
6144 return std::move(opts);
6149 m_interrupt{interrupt},
6151 m_blockman{interrupt,
std::move(blockman_options)},
6152 m_validation_cache{m_options.script_execution_cache_bytes, m_options.signature_cache_bytes}
6163Chainstate* ChainstateManager::LoadAssumeutxoChainstate()
6171 if (!base_blockhash) {
6174 LogInfo(
"[snapshot] detected active snapshot chainstate (%s) - loading",
6175 fs::PathToString(*path));
6177 auto snapshot_chainstate{std::make_unique<Chainstate>(
nullptr,
m_blockman, *
this, base_blockhash)};
6178 LogInfo(
"[snapshot] switching active chainstate to %s", snapshot_chainstate->ToString());
6179 return &this->AddChainstate(std::move(snapshot_chainstate));
6182Chainstate& ChainstateManager::AddChainstate(std::unique_ptr<Chainstate> chainstate)
6187 assert(!prev_chainstate.m_target_blockhash);
6188 prev_chainstate.m_target_blockhash = chainstate->m_from_snapshot_blockhash;
6189 m_chainstates.push_back(std::move(chainstate));
6191 assert(&curr_chainstate == m_chainstates.back().get());
6195 assert(!prev_chainstate.m_mempool || prev_chainstate.m_mempool->size() == 0);
6196 assert(!curr_chainstate.m_mempool);
6197 std::swap(curr_chainstate.m_mempool, prev_chainstate.m_mempool);
6198 return curr_chainstate;
6203 return (block_index.
nHeight==91842 && block_index.
GetBlockHash() ==
uint256{
"00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec"}) ||
6204 (block_index.
nHeight==91880 && block_index.
GetBlockHash() ==
uint256{
"00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721"});
6209 return (block_height==91722 && block_hash ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
6210 (block_height==91812 && block_hash ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"});
6222 const fs::path invalid_path{db_path +
"_INVALID"};
6225 LogInfo(
"[snapshot] renaming snapshot datadir %s to %s", db_path_str, invalid_path_str);
6231 fs::rename(db_path, invalid_path);
6232 }
catch (
const fs::filesystem_error& e) {
6233 LogError(
"While invalidating the coins db: Error renaming file '%s' -> '%s': %s",
6234 db_path_str, invalid_path_str, e.what());
6236 "Rename of '%s' -> '%s' failed. "
6237 "You should resolve this by manually moving or deleting the invalid "
6238 "snapshot directory %s, otherwise you will encounter the same error again "
6239 "on the next startup."),
6240 db_path_str, invalid_path_str, db_path_str)};
6245bool ChainstateManager::DeleteChainstate(
Chainstate& chainstate)
6251 LogError(
"Deletion of %s failed. Please remove it manually to continue reindexing.",
6252 fs::PathToString(db_path));
6257 assert(prev_chainstate->m_mempool->size() == 0);
6258 assert(!curr_chainstate.m_mempool);
6259 std::swap(curr_chainstate.m_mempool, prev_chainstate->m_mempool);
6268void ChainstateManager::RecalculateBestHeader()
6272 for (
auto& entry :
m_blockman.m_block_index) {
6273 if (!(entry.second.nStatus &
BLOCK_FAILED_VALID) && m_best_header->nChainWork < entry.second.nChainWork) {
6274 m_best_header = &entry.second;
6279std::optional<int> ChainstateManager::BlocksAheadOfTip()
const
6285 if (best_header && tip && best_header->nChainWork > tip->
nChainWork &&
6286 best_header->GetAncestor(tip->
nHeight) == tip) {
6289 return std::nullopt;
6292bool ChainstateManager::ValidatedSnapshotCleanup(
Chainstate& validated_cs,
Chainstate& unvalidated_cs)
6300 const fs::path validated_path{validated_cs.
StoragePath()};
6301 const fs::path assumed_valid_path{unvalidated_cs.
StoragePath()};
6302 const fs::path delete_path{validated_path +
"_todelete"};
6310 this->ResetChainstates();
6311 assert(this->m_chainstates.size() == 0);
6313 LogInfo(
"[snapshot] deleting background chainstate directory (now unnecessary) (%s)",
6314 fs::PathToString(validated_path));
6316 auto rename_failed_abort = [
this](
6319 const fs::filesystem_error& err) {
6320 LogError(
"[snapshot] Error renaming path (%s) -> (%s): %s\n",
6321 fs::PathToString(p_old), fs::PathToString(p_new), err.what());
6323 "Rename of '%s' -> '%s' failed. "
6324 "Cannot clean up the background chainstate leveldb directory."),
6325 fs::PathToString(p_old), fs::PathToString(p_new)));
6329 fs::rename(validated_path, delete_path);
6330 }
catch (
const fs::filesystem_error& e) {
6331 rename_failed_abort(validated_path, delete_path, e);
6335 LogInfo(
"[snapshot] moving snapshot chainstate (%s) to "
6336 "default chainstate directory (%s)",
6337 fs::PathToString(assumed_valid_path), fs::PathToString(validated_path));
6340 fs::rename(assumed_valid_path, validated_path);
6341 }
catch (
const fs::filesystem_error& e) {
6342 rename_failed_abort(assumed_valid_path, validated_path, e);
6349 LogWarning(
"Deletion of %s failed. Please remove it manually, as the "
6350 "directory is now unnecessary.",
6351 fs::PathToString(delete_path));
6353 LogInfo(
"[snapshot] deleted background chainstate directory (%s)",
6354 fs::PathToString(validated_path));
6359std::pair<int, int> Chainstate::GetPruneRange(
int last_height_can_prune)
const
6370 prune_start =
Assert(SnapshotBase())->nHeight + 1;
6373 int max_prune = std::max<int>(
6382 int prune_end = std::min(last_height_can_prune, max_prune);
6384 return {prune_start, prune_end};
6387std::optional<std::pair<const CBlockIndex*, const CBlockIndex*>> ChainstateManager::GetHistoricalBlockRange()
const
6390 if (!chainstate)
return {};
6391 return std::make_pair(chainstate->
m_chain.
Tip(), chainstate->TargetBlock());
6400 std::vector<Chainstate*> chainstates;
6403 chainstates.reserve(m_chainstates.size());
6404 for (
const auto& chainstate : m_chainstates) {
6405 if (chainstate && chainstate->m_assumeutxo !=
Assumeutxo::INVALID && !chainstate->m_target_utxohash) {
6406 chainstates.push_back(chainstate.get());
6412 if (!chainstate->ActivateBestChain(state,
nullptr)) {
bool MoneyRange(const CAmount &nValue)
int64_t CAmount
Amount in satoshis (Can be negative)
static constexpr CAmount COIN
The amount of satoshis in one BTC.
arith_uint256 UintToArith256(const uint256 &a)
void InvalidateBlock(ChainstateManager &chainman, const uint256 block_hash)
CBlockLocator GetLocator(const CBlockIndex *index)
Get a locator for a block index entry.
int64_t GetBlockProofEquivalentTime(const CBlockIndex &to, const CBlockIndex &from, const CBlockIndex &tip, const Consensus::Params ¶ms)
Return the time it would take to redo the work difference between from and to, assuming the current h...
const CBlockIndex * LastCommonAncestor(const CBlockIndex *pa, const CBlockIndex *pb)
Find the last common ancestor two blocks have.
@ BLOCK_VALID_CHAIN
Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends,...
@ BLOCK_VALID_MASK
All validity bits.
@ BLOCK_VALID_TRANSACTIONS
Only first tx is coinbase, 2 <= coinbase input script length <= 100, transactions valid,...
@ BLOCK_VALID_SCRIPTS
Scripts & signatures ok.
@ BLOCK_VALID_TREE
All parent headers found, difficulty matches, timestamp >= median previous.
@ BLOCK_HAVE_UNDO
undo data available in rev*.dat
@ BLOCK_HAVE_DATA
full block available in blk*.dat
@ BLOCK_FAILED_VALID
stage after last reached validness failed
@ BLOCK_OPT_WITNESS
block data in blk*.dat was received with a witness-enforcing client
static constexpr int32_t SEQ_ID_BEST_CHAIN_FROM_DISK
Init values for CBlockIndex nSequenceId when loaded from disk.
arith_uint256 GetBlockProof(const CBlockIndex &block)
Compute how much work a block index entry corresponds to.
static constexpr int32_t SEQ_ID_INIT_FROM_DISK
#define NONFATAL_UNREACHABLE()
NONFATAL_UNREACHABLE() is a macro that is used to mark unreachable code.
#define Assert(val)
Identity function.
#define STR_INTERNAL_BUG(msg)
#define Assume(val)
Assume is the identity function.
Non-refcounted RAII wrapper for FILE*.
std::string ToString() const
Wrapper around an AutoFile& that implements a ring buffer to deserialize from.
bool m_checked_merkle_root
std::vector< CTransactionRef > vtx
bool m_checked_witness_commitment
The block chain is a tree shaped structure starting with the genesis block at the root,...
bool IsValid(enum BlockStatus nUpTo) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
CBlockIndex * pprev
pointer to the index of the predecessor of this block
uint64_t m_chain_tx_count
(memory only) Number of transactions in the chain up to and including this block.
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
bool HaveNumChainTxs() const
Check whether this block and all previous blocks back to the genesis block or an assumeutxo snapshot ...
int32_t nSequenceId
(memory only) Sequential id assigned to distinguish order in which blocks are received.
uint256 GetBlockHash() const
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
FlatFilePos GetUndoPos() const EXCLUSIVE_LOCKS_REQUIRED(
bool RaiseValidity(enum BlockStatus nUpTo) EXCLUSIVE_LOCKS_REQUIRED(
Raise the validity level of this block index entry.
CBlockIndex * pskip
pointer to the index of some further predecessor of this block
unsigned int nTx
Number of transactions in this block.
int32_t nVersion
block header
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
int nHeight
height of the entry in the chain. The genesis block has height 0
const uint256 * phashBlock
pointer to the hash of the block, if any. Memory is owned by this CBlockIndex
Undo information for a CBlock.
std::vector< CTxUndo > vtxundo
An in-memory indexed chain of blocks.
bool Contains(const CBlockIndex &index) const
Efficiently check whether a block is present in this chain.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
const CBlockIndex * FindFork(const CBlockIndex &index) const
Find the last common block between this chain and a block index entry.
void SetTip(CBlockIndex &block)
Set/initialize a chain with a given tip.
CBlockIndex * Next(const CBlockIndex &index) const
Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip...
CBlockIndex * Genesis() const
Returns the index entry for the genesis block of this chain, or nullptr if none.
int Height() const
Return the maximal height in the chain.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
std::vector< int > GetAvailableSnapshotHeights() const
const ChainTxData & TxData() const
std::optional< AssumeutxoData > AssumeutxoForHeight(int height) const
CCoinsView that adds a memory cache for transactions to another CCoinsView.
void Sync()
Push the modifications applied to this cache to its base while retaining the contents of this cache (...
bool SpendCoin(const COutPoint &outpoint, Coin *moveto=nullptr)
Spend a coin.
ResetGuard CreateResetGuard() noexcept
Create a scoped guard that will call Reset() on this cache when it goes out of scope.
void Uncache(const COutPoint &outpoint)
Removes the UTXO with the given outpoint from the cache, if it is not modified.
void AddCoin(const COutPoint &outpoint, Coin &&coin, bool possible_overwrite)
Add a coin.
void Flush(bool reallocate_cache=true)
Push the modifications applied to this cache to its base and wipe local state.
void SetBestBlock(const uint256 &block_hash)
unsigned int GetCacheSize() const
Size of the cache (in number of transaction outputs)
uint256 GetBestBlock() const override
Retrieve the block hash whose state this CCoinsView currently represents.
bool HaveCoinInCache(const COutPoint &outpoint) const
Check if we have the given utxo already loaded in this cache.
size_t DynamicMemoryUsage() const
Calculate the size of the cache (in bytes)
void EmplaceCoinInternalDANGER(COutPoint &&outpoint, Coin &&coin)
Emplace a coin into cacheCoins without performing any checks, marking the emplaced coin as dirty.
bool HaveCoin(const COutPoint &outpoint) const override
Just check whether a given outpoint is unspent.
const Coin & AccessCoin(const COutPoint &output) const
Return a reference to Coin in the cache, or coinEmpty if not found.
CCoinsView backed by the coin database (chainstate/)
std::shared_future< void > CompactFullAsync() EXCLUSIVE_LOCKS_REQUIRED(cs_main
Perform a full compaction of the underlying LevelDB on a one-shot background thread.
void ResizeCache(size_t new_cache_size) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Dynamically alter the underlying leveldb cache size.
Pure abstract view on the open txout dataset.
virtual std::optional< Coin > GetCoin(const COutPoint &outpoint) const =0
Retrieve the Coin (unspent transaction output) for a given outpoint.
CCoinsView that brings transactions from a mempool into view.
Fee rate in satoshis per virtualbyte: CAmount / vB the feerate is represented internally as FeeFrac.
A hasher class for Bitcoin's 256-bit hash (double SHA-256).
void Finalize(std::span< unsigned char > output)
CHash256 & Write(std::span< const unsigned char > input)
An outpoint - a combination of a transaction hash and an index n into its vout.
A hasher class for SHA-256.
void Finalize(unsigned char hash[OUTPUT_SIZE])
CSHA256 & Write(const unsigned char *data, size_t len)
Closure representing one script verification Note that this stores references to the spending transac...
SignatureCache * m_signature_cache
PrecomputedTransactionData * txdata
script_verify_flags m_flags
std::optional< std::pair< ScriptError, std::string > > operator()()
const CTransaction * ptxTo
Serialized script, used inside transaction inputs and outputs.
The basic transaction that is broadcasted on the network and contained in blocks.
const std::vector< CTxOut > vout
const Wtxid & GetWitnessHash() const LIFETIMEBOUND
const Txid & GetHash() const LIFETIMEBOUND
const std::vector< CTxIn > vin
An input of a transaction.
CTxMemPool::txiter TxHandle
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
void check(const CCoinsViewCache &active_coins_tip, int64_t spendheight) const EXCLUSIVE_LOCKS_REQUIRED(void removeRecursive(const CTransaction &tx, MemPoolRemovalReason reason) EXCLUSIVE_LOCKS_REQUIRED(cs)
If sanity-checking is turned on, check makes sure the pool is consistent (does not contain two transa...
void UpdateTransactionsFromBlock(const std::vector< Txid > &vHashesToUpdate) EXCLUSIVE_LOCKS_REQUIRED(cs
UpdateTransactionsFromBlock is called when adding transactions from a disconnected block back to the ...
void AddTransactionsUpdated(unsigned int n)
CTransactionRef get(const Txid &hash) const
size_t DynamicMemoryUsage() const
void removeForReorg(CChain &chain, std::function< bool(txiter)> filter_final_and_mature) EXCLUSIVE_LOCKS_REQUIRED(cs
After reorg, filter the entries that would no longer be valid in the next block, and update the entri...
bool exists(const Txid &txid) const
std::set< txiter, CompareIteratorByHash > setEntries
void removeForBlock(const std::vector< CTransactionRef > &vtx, unsigned int nBlockHeight) EXCLUSIVE_LOCKS_REQUIRED(cs)
indexed_transaction_set::nth_index< 0 >::type::const_iterator txiter
unsigned long size() const
An output of a transaction.
Undo information for a CTransaction.
std::vector< Coin > vprevout
VerifyDBResult VerifyDB(Chainstate &chainstate, const Consensus::Params &consensus_params, CCoinsView &coinsview, int nCheckLevel, int nCheckDepth) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
kernel::Notifications & m_notifications
CVerifyDB(kernel::Notifications ¬ifications)
Chainstate stores and provides an API to update our local knowledge of the current best chain.
void InitCoinsCache(size_t cache_size_bytes) EXCLUSIVE_LOCKS_REQUIRED(bool CanFlushToDisk() const EXCLUSIVE_LOCKS_REQUIRED(
Initialize the in-memory coins cache (to be done after the health of the on-disk database is verified...
Mutex m_chainstate_mutex
The ChainState Mutex A lock that must be held when modifying this ChainState - held in ActivateBestCh...
CChain m_chain
The current chain of blockheaders we consult and build on.
CTxMemPool * GetMempool()
bool RollforwardBlock(const CBlockIndex *pindex, CCoinsViewCache &inputs) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Apply the effects of a block on the utxo cache, ignoring that it may already have been applied.
size_t m_coinstip_cache_size_bytes
The cache size of the in-memory coins view.
void UpdateTip(const CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(NodeClock::time_poin m_next_write)
Check warning conditions and do some notifications on new chain tip set.
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
bool LoadChainTip() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Update the chain tip based on database information, i.e.
size_t m_coinsdb_cache_size_bytes
The cache size of the on-disk coins view.
bool PreciousBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(bool InvalidateBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(void SetBlockFailureFlags(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(voi ResetBlockFailureFlags)(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as precious and reorganize.
void InvalidBlockFound(CBlockIndex *pindex, const BlockValidationState &state) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ConnectTip(BlockValidationState &state, CBlockIndex *pindexNew, std::shared_ptr< const CBlock > block_to_connect, std::vector< ConnectedBlock > &connected_blocks, DisconnectedBlockTransactions &disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Connect a new block to m_chain.
void CheckForkWarningConditions() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
const CBlockIndex *SnapshotBase() const EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *TargetBlock() const EXCLUSIVE_LOCKS_REQUIRED(void SetTargetBlock(CBlockIndex *block) EXCLUSIVE_LOCKS_REQUIRED(void SetTargetBlockHash(uint256 block_hash) EXCLUSIVE_LOCKS_REQUIRED(boo ReachedTarget)() const EXCLUSIVE_LOCKS_REQUIRED(
The base of the snapshot this chainstate was created from.
bool LoadGenesisBlock()
Ensures we have a genesis block in the block tree, possibly writing one to disk.
kernel::ChainstateRole GetRole() const EXCLUSIVE_LOCKS_REQUIRED(void InitCoinsDB(size_t cache_size_bytes, bool in_memory, bool should_wipe)
Return the current role of the chainstate.
const std::optional< uint256 > m_from_snapshot_blockhash
The blockhash which is the base of the snapshot this chainstate was created from.
bool ActivateBestChain(BlockValidationState &state, std::shared_ptr< const CBlock > pblock=nullptr) LOCKS_EXCLUDED(DisconnectResult DisconnectBlock(const CBlock &block, const CBlockIndex *pindex, CCoinsViewCache &view) EXCLUSIVE_LOCKS_REQUIRED(boo ConnectBlock)(const CBlock &block, BlockValidationState &state, CBlockIndex *pindex, CCoinsViewCache &view, bool fJustCheck=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Find the best known block, and make it the tip of the block chain.
bool ActivateBestChainStep(BlockValidationState &state, CBlockIndex &index_most_work, const std::shared_ptr< const CBlock > &pblock, bool &fInvalidFound, std::vector< ConnectedBlock > &connected_blocks) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Return the [start, end] (inclusive) of block heights we can prune.
CTxMemPool * m_mempool
Optional mempool that is kept in sync with the chain.
CCoinsViewDB & CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(
bool DisconnectTip(BlockValidationState &state, DisconnectedBlockTransactions *disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Disconnect m_chain's tip.
CBlockIndex * FindMostWorkChain() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Return the tip of the chain with the most work in it, that isn't known to be invalid (it's however fa...
std::set< CBlockIndex *, node::CBlockIndexWorkComparator > setBlockIndexCandidates
The set of all CBlockIndex entries that have as much work as our current tip or more,...
ChainstateManager & m_chainman
The chainstate manager that owns this chainstate.
std::unique_ptr< CoinsViews > m_coins_views
Manages the UTXO set, which is a reflection of the contents of m_chain.
bool ReplayBlocks()
Replay blocks that aren't fully applied to the database.
void PruneBlockIndexCandidates()
Delete all entries in setBlockIndexCandidates that are worse than the current tip.
void TryAddBlockIndexCandidate(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Add a block to the candidate set if it has as much work as the current tip.
void PruneAndFlush()
Prune blockfiles from the disk if necessary and then flush chainstate changes if we pruned.
bool ResizeCoinsCaches(size_t coinstip_size, size_t coinsdb_size) EXCLUSIVE_LOCKS_REQUIRED(bool FlushStateToDisk(BlockValidationState &state, FlushStateMode mode, int nManualPruneHeight=0)
Resize the CoinsViews caches dynamically and flush state to disk.
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances.
void ForceFlushStateToDisk(bool wipe_cache=true)
Flush all changes to disk.
void MaybeUpdateMempoolForReorg(DisconnectedBlockTransactions &disconnectpool, bool fAddToMempool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Make mempool consistent after a reorg, by re-adding or recursively erasing disconnected block transac...
void ClearBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(void PopulateBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex * FindForkInGlobalIndex(const CBlockLocator &locator) const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Populate the candidate set by calling TryAddBlockIndexCandidate on all valid block indices.
void InvalidChainFound(CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Chainstate(CTxMemPool *mempool, node::BlockManager &blockman, ChainstateManager &chainman, std::optional< uint256 > from_snapshot_blockhash=std::nullopt)
fs::path StoragePath() const
Return path to chainstate leveldb directory.
bool NeedsRedownload() const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Whether the chain state needs to be redownloaded due to lack of witness data.
CoinsCacheSizeState GetCoinsCacheSizeState() EXCLUSIVE_LOCKS_REQUIRED(CoinsCacheSizeState GetCoinsCacheSizeState(size_t max_coins_cache_size_bytes, size_t max_mempool_size_bytes) EXCLUSIVE_LOCKS_REQUIRED(std::string ToString() EXCLUSIVE_LOCKS_REQUIRED(RecursiveMutex * MempoolMutex() const LOCK_RETURNED(m_mempool -> cs)
Dictates whether we need to flush the cache to disk or not.
Interface for managing multiple Chainstate objects, where each chainstate is associated with chainsta...
util::Result< void > PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
Chainstate * HistoricalChainstate() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Return historical chainstate targeting a specific block, if any.
const uint256 & AssumedValidBlock() const
ValidationCache m_validation_cache
double GetBackgroundVerificationProgress(const CBlockIndex &pindex) const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Guess background verification progress in case assume-utxo was used (as a fraction between 0....
double GuessVerificationProgress(const CBlockIndex *pindex) const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Guess verification progress (as a fraction between 0.0=genesis and 1.0=current tip).
bool IsInitialBlockDownload() const noexcept
Check whether we are doing an initial block download (synchronizing from disk or network)
size_t m_total_coinstip_cache
The total number of bytes available for us to use across all in-memory coins caches.
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.
std::unique_ptr< Chainstate > RemoveChainstate(Chainstate &chainstate) EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Remove a chainstate.
kernel::Notifications & GetNotifications() const
void ReceivedBlockTransactions(const CBlock &block, CBlockIndex *pindexNew, const FlatFilePos &pos) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as having its data received and checked (up to BLOCK_VALID_TRANSACTIONS).
bool ShouldCheckBlockIndex() const
RecursiveMutex & GetMutex() const LOCK_RETURNED(
Alias for cs_main.
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Chainstate & ActiveChainstate() const
Alternatives to CurrentChainstate() used by older code to query latest chainstate information without...
SnapshotCompletionResult MaybeValidateSnapshot(Chainstate &validated_cs, Chainstate &unvalidated_cs) EXCLUSIVE_LOCKS_REQUIRED(Chainstate & CurrentChainstate() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Try to validate an assumeutxo snapshot by using a validated historical chainstate targeted at the sna...
bool ProcessNewBlock(const std::shared_ptr< const CBlock > &block, bool force_processing, bool min_pow_checked, bool *new_block) LOCKS_EXCLUDED(cs_main)
Process an incoming block.
size_t m_total_coinsdb_cache
The total number of bytes available for us to use across all leveldb coins databases.
void CheckBlockIndex() const
Make various assertions about the state of the block index.
const util::SignalInterrupt & m_interrupt
void LoadExternalBlockFile(AutoFile &file_in, FlatFilePos *dbp=nullptr, std::multimap< uint256, FlatFilePos > *blocks_with_unknown_parent=nullptr)
Import blocks from an external file.
int ActiveHeight() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
VersionBitsCache m_versionbitscache
Track versionbit status.
std::function< void()> snapshot_download_completed
Function to restart active indexes; set dynamically to avoid a circular dependency on base/index....
const CChainParams & GetParams() const
void GenerateCoinbaseCommitment(CBlock &block, const CBlockIndex *pindexPrev) const
Produce the necessary coinbase commitment for a block (modifies the hash, don't call for mined blocks...
bool ProcessNewBlockHeaders(std::span< const CBlockHeader > headers, bool min_pow_checked, BlockValidationState &state, const CBlockIndex **ppindex=nullptr) LOCKS_EXCLUDED(cs_main)
Process incoming block headers.
const Consensus::Params & GetConsensus() const
ChainstateManager(const util::SignalInterrupt &interrupt, Options options, node::BlockManager::Options blockman_options)
const arith_uint256 & MinimumChainWork() const
void UpdateIBDStatus() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Update and possibly latch the IBD status.
bool LoadBlockIndex() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Load the block tree and coins database from disk, initializing state if we're running with -reindex.
Chainstate &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(util::Result< CBlockIndex * ActivateSnapshot)(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Instantiate a new chainstate.
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
bool AcceptBlockHeader(const CBlockHeader &block, BlockValidationState &state, CBlockIndex **ppindex, bool min_pow_checked) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
If a block header hasn't already been seen, call CheckBlockHeader on it, ensure that it doesn't desce...
arith_uint256 nLastPreciousChainwork
chainwork for the last block that preciousblock has been applied to.
void ReportHeadersPresync(int64_t height, int64_t timestamp)
This is used by net_processing to report pre-synchronization progress of headers, as headers are not ...
std::atomic_bool m_cached_is_ibd
Whether initial block download (IBD) is ongoing.
bool NotifyHeaderTip() LOCKS_EXCLUDED(GetMutex())
void MaybeRebalanceCaches() EXCLUSIVE_LOCKS_REQUIRED(void UpdateUncommittedBlockStructures(CBlock &block, const CBlockIndex *pindexPrev) const
Check to see if caches are out of balance and if so, call ResizeCoinsCaches() as needed.
Chainstate *LoadAssumeutxoChainstate() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &AddChainstate(std::unique_ptr< Chainstate > chainstate) EXCLUSIVE_LOCKS_REQUIRED(void ResetChainstates() EXCLUSIVE_LOCKS_REQUIRED(bool DeleteChainstate(Chainstate &chainstate) EXCLUSIVE_LOCKS_REQUIRED(bool ValidatedSnapshotCleanup(Chainstate &validated_cs, Chainstate &unvalidated_cs) EXCLUSIVE_LOCKS_REQUIRED(std::optional< std::pair< const CBlockIndex *, const CBlockIndex * > > GetHistoricalBlockRange() const EXCLUSIVE_LOCKS_REQUIRED(util::Result< void > ActivateBestChains() LOCKS_EXCLUDED(void RecalculateBestHeader() EXCLUSIVE_LOCKS_REQUIRED(std::optional< int > BlocksAheadOfTip() const LOCKS_EXCLUDED(CCheckQueue< CScriptCheck > & GetCheckQueue()
When starting up, search the datadir for a chainstate based on a UTXO snapshot that is in the process...
int32_t nBlockReverseSequenceId
Decreasing counter (used by subsequent preciousblock calls).
node::BlockManager m_blockman
A single BlockManager instance is shared across each constructed chainstate to avoid duplicating bloc...
bool AcceptBlock(const std::shared_ptr< const CBlock > &pblock, BlockValidationState &state, CBlockIndex **ppindex, bool fRequested, const FlatFilePos *dbp, bool *fNewBlock, bool min_pow_checked) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Sufficiently validate a block for disk storage (and store on disk).
CTxOut out
unspent transaction output
bool IsSpent() const
Either this coin never existed (see e.g.
bool fCoinBase
whether containing transaction was a coinbase
uint32_t nHeight
at which height this containing transaction was included in the active block chain
static CoinsViewEmpty & Get()
CoinsViews(DBParams db_params, CoinsViewOptions options)
This constructor initializes CCoinsViewDB and CCoinsViewErrorCatcher instances, but it does not creat...
std::pair< uint32_t, size_t > setup_bytes(size_t bytes)
setup_bytes is a convenience function which accounts for internal memory usage when deciding how many...
void insert(Element e)
insert loops at most depth_limit times trying to insert a hash at various locations in the table via ...
bool contains(const Element &e, const bool erase) const
contains iterates through the hash locations for a given element and checks to see if it is present.
DisconnectedBlockTransactions.
std::list< CTransactionRef > take()
Clear all data structures and return the list of transactions.
void removeForBlock(const std::vector< CTransactionRef > &vtx)
Remove any entries that are in this block.
std::vector< CTransactionRef > AddTransactionsFromBlock(const std::vector< CTransactionRef > &vtx)
Add transactions from the block, iterating through vtx in reverse order.
Tp rand_uniform_delay(const Tp &time, typename Tp::duration range) noexcept
Return the time point advanced by a uniform random duration.
I randrange(I range) noexcept
Generate a random integer in the range [0..range), with range > 0.
Convenience class for initializing and passing the script execution cache and signature cache.
ValidationCache(size_t script_execution_cache_bytes, size_t signature_cache_bytes)
CuckooCache::cache< uint256, SignatureCacheHasher > m_script_execution_cache
CSHA256 ScriptExecutionCacheHasher() const
Return a copy of the pre-initialized hasher.
CSHA256 m_script_execution_cache_hasher
Pre-initialized hasher to avoid having to recreate it for every hash calculation.
SignatureCache m_signature_cache
void BlockConnected(const kernel::ChainstateRole &, std::shared_ptr< const CBlock >, const CBlockIndex *pindex)
void BlockChecked(const std::shared_ptr< const CBlock > &, const BlockValidationState &)
void ChainStateFlushed(const kernel::ChainstateRole &, const CBlockLocator &)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void ActiveTipChange(const CBlockIndex &, bool)
void BlockDisconnected(std::shared_ptr< const CBlock >, const CBlockIndex *pindex)
std::string GetRejectReason() const
std::string GetDebugMessage() const
bool Error(const std::string &reject_reason)
bool Invalid(Result result, const std::string &reject_reason="", const std::string &debug_message="")
std::string ToString() const
std::vector< std::pair< int, bool > > CheckUnknownActivations(const CBlockIndex *pindex, const CChainParams &chainparams) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Check for unknown activations Returns a vector containing the bit number used for signalling and a bo...
void Clear() EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
256-bit unsigned big integer.
constexpr bool IsNull() const
std::string ToString() const
constexpr unsigned char * begin()
A base class defining functions for notifying about certain kernel events.
virtual void headerTip(SynchronizationState state, int64_t height, int64_t timestamp, bool presync)
virtual void fatalError(const bilingual_str &message)
The fatal error notification is sent to notify the user when an error occurs in kernel code that can'...
virtual void warningSet(Warning id, const bilingual_str &message)
virtual void progress(const bilingual_str &title, int progress_percent, bool resume_possible)
virtual InterruptResult blockTip(SynchronizationState state, const CBlockIndex &index, double verification_progress)
virtual void warningUnset(Warning id)
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
const kernel::BlockManagerOpts m_opts
void FindFilesToPrune(std::set< int > &setFilesToPrune, int last_prune, const Chainstate &chain, ChainstateManager &chainman)
Prune block and undo files (blk???.dat and rev???.dat) so that the disk space used is less than a use...
CBlockIndex * LookupBlockIndex(const uint256 &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ReadBlockUndo(CBlockUndo &blockundo, const CBlockIndex &index) const
CBlockFileInfo *GetBlockFileInfo(size_t n) EXCLUSIVE_LOCKS_REQUIRED(bool WriteBlockUndo(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos WriteBlock(const CBlock &block, int nHeight) EXCLUSIVE_LOCKS_REQUIRED(void UpdateBlockInfo(const CBlock &block, unsigned int nHeight, const FlatFilePos &pos) EXCLUSIVE_LOCKS_REQUIRED(bool IsPruneMode() const
Get block file info entry for one block file.
std::atomic_bool m_blockfiles_indexed
Whether all blockfiles have been added to the block tree database.
std::vector< CBlockIndex * > GetAllBlockIndices() EXCLUSIVE_LOCKS_REQUIRED(std::multimap< CBlockIndex *, CBlockIndex * > m_blocks_unlinked
All pairs A->B, where A (or one of its ancestors) misses transactions, but B has transactions.
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
bool LoadingBlocks() const
void UnlinkPrunedFiles(const std::set< int > &setFilesToPrune) const
Actually unlink the specified files.
void WriteBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(bool LoadBlockIndexDB(const std::optional< uint256 > &snapshot_blockhash) EXCLUSIVE_LOCKS_REQUIRED(void ScanAndUnlinkAlreadyPrunedFiles() EXCLUSIVE_LOCKS_REQUIRED(CBlockIndex * AddToBlockIndex(const CBlockHeader &block, CBlockIndex *&best_header) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove any pruned block & undo files that are still on disk.
void AddUnlinkedBlock(CBlockIndex *block) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ReadBlock(CBlock &block, const FlatFilePos &pos, const std::optional< uint256 > &expected_hash) const
Functions for disk access for blocks.
bool m_check_for_pruning
Global flag to indicate we should check to see if there are block/undo files that should be deleted.
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, const Chainstate &chain)
std::optional< int > m_snapshot_height
The height of the base block of an assumeutxo snapshot, if one is in use.
uint64_t CalculateCurrentUsage() EXCLUSIVE_LOCKS_REQUIRED(bool CheckBlockDataAvailability(const CBlockIndex &upper_block, const CBlockIndex &lower_block, BlockStatus block_status=BLOCK_HAVE_DATA) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex &GetFirstBlock(const CBlockIndex &upper_block LIFETIMEBOUND, uint32_t status_mask, const CBlockIndex *lower_block LIFETIMEBOUND=nullptr) const EXCLUSIVE_LOCKS_REQUIRED(boo m_have_pruned)
Calculate the amount of disk space the block & undo files currently use.
std::string ToString() const
constexpr const std::byte * begin() const
const uint256 & ToUint256() const LIFETIMEBOUND
std::string GetHex() const
Helper class that manages an interrupt flag, and allows a thread or signal to interrupt another threa...
std::string FormatFullVersion()
const Coin & AccessByTxid(const CCoinsViewCache &view, const Txid &txid)
Utility function to find any unspent output with a given txid.
void AddCoins(CCoinsViewCache &cache, const CTransaction &tx, int nHeight, bool check_for_overwrite)
Utility function to add all of a transaction's outputs to a cache.
uint256 BlockMerkleRoot(const CBlock &block, bool *mutated)
uint256 BlockWitnessMerkleRoot(const CBlock &block)
static constexpr int NO_WITNESS_COMMITMENT
Index marker for when no witness commitment is present in a coinbase transaction.
static constexpr size_t MINIMUM_WITNESS_COMMITMENT
Minimum size of a witness commitment structure.
static int64_t GetBlockWeight(const CBlock &block)
@ BLOCK_HEADER_LOW_WORK
the block header may be on a too-little-work chain
@ BLOCK_INVALID_HEADER
invalid proof of work or time too old
@ BLOCK_CACHED_INVALID
this block was cached as being invalid and we didn't store the reason why
@ BLOCK_CONSENSUS
invalid by consensus rules (excluding any below reasons)
@ BLOCK_MISSING_PREV
We don't have the previous block the checked one is built on.
@ BLOCK_INVALID_PREV
A block this one builds on is invalid.
@ BLOCK_MUTATED
the block's data didn't match the data committed to by the PoW
@ BLOCK_TIME_FUTURE
block timestamp was > 2 hours in the future (or our clock is bad)
int GetWitnessCommitmentIndex(const CBlock &block)
Compute at which vout of the block's coinbase transaction the witness commitment occurs,...
@ TX_MISSING_INPUTS
transaction was missing some of its inputs
@ TX_MEMPOOL_POLICY
violated mempool's fee/size/descendant/RBF/etc limits
@ TX_PREMATURE_SPEND
transaction spends a coinbase too early, or violates locktime/sequence locks
@ TX_WITNESS_STRIPPED
Transaction is missing a witness.
@ TX_CONFLICT
Tx already in mempool or conflicts with a tx in the chain (if it conflicts with another tx in mempool...
@ TX_NOT_STANDARD
otherwise didn't meet our local policy rules
@ TX_WITNESS_MUTATED
Transaction might have a witness prior to SegWit activation, or witness may have been malleated (whic...
@ TX_NO_MEMPOOL
this node does not have a mempool so can't validate the transaction
@ TX_CONSENSUS
invalid by consensus rules
@ TX_RECONSIDERABLE
fails some policy, but might be acceptable if submitted in a (different) package
static constexpr unsigned int LOCKTIME_VERIFY_SEQUENCE
Flags for nSequence and nLockTime locks.
static constexpr int64_t MAX_TIMEWARP
Maximum number of seconds that the timestamp of the first block of a difficulty adjustment period is ...
static const unsigned int MAX_BLOCK_WEIGHT
The maximum allowed weight for a block, see BIP 141 (network rule)
static const unsigned int MAX_BLOCK_SERIALIZED_SIZE
The maximum allowed size for a serialized block, in bytes (only for buffer size limits)
static const int64_t MAX_BLOCK_SIGOPS_COST
The maximum allowed number of signature check operations in a block (network rule)
static const int COINBASE_MATURITY
Coinbase transaction outputs can only be spent after this number of new blocks (network rule)
static const int WITNESS_SCALE_FACTOR
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
bool DestroyDB(const std::string &path_str)
bool DeploymentActiveAfter(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, VersionBitsCache &versionbitscache)
Determine if a deployment is active for the next block.
bool DeploymentActiveAt(const CBlockIndex &index, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, VersionBitsCache &versionbitscache)
Determine if a deployment is active for this block.
static const unsigned int MAX_DISCONNECTED_TX_POOL_BYTES
Maximum bytes for transactions to store for processing during reorg.
bool CheckEphemeralSpends(const Package &package, CFeeRate dust_relay_rate, const CTxMemPool &tx_pool, TxValidationState &out_child_state, Wtxid &out_child_wtxid)
Called for each transaction(package) if any dust is in the package.
bool PreCheckEphemeralTx(const CTransaction &tx, CFeeRate dust_relay_rate, CAmount base_fee, CAmount mod_fee, TxValidationState &state)
These utility functions ensure that ephemeral dust is safely created and spent without unduly risking...
static bool exists(const path &p)
static std::string PathToString(const path &path)
Convert path object to a byte string.
bool CheckDiskSpace(const fs::path &dir, uint64_t additional_bytes)
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, const CScriptWitness *witness, script_verify_flags flags, const BaseSignatureChecker &checker, ScriptError *serror)
@ SCRIPT_VERIFY_NULLDUMMY
@ SCRIPT_VERIFY_CHECKSEQUENCEVERIFY
@ SCRIPT_VERIFY_CHECKLOCKTIMEVERIFY
is a home for simple enum and struct type definitions that can be used internally by functions in the...
#define LogDebug(category,...)
@ REORG
Removed for reorganization.
std::array< uint8_t, 4 > MessageStartChars
BlockValidationState m_state
bool CheckTxInputs(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, int nSpendHeight, CAmount &txfee)
Check whether all inputs of this transaction are valid (no double spends and amounts) This does not m...
std::function< FILE *(const fs::path &, const char *)> FopenFn
bool IsInterrupted(const T &result)
@ UNKNOWN_NEW_RULES_ACTIVATED
@ LARGE_WORK_INVALID_CHAIN
static std::optional< CCoinsStats > ComputeUTXOStats(T hash_obj, CCoinsView *view, node::BlockManager &blockman, const std::function< void()> &interruption_point)
Calculate statistics about the unspent transaction output set.
const fs::path SNAPSHOT_BLOCKHASH_FILENAME
The file in the snapshot chainstate dir which stores the base blockhash.
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate)
std::optional< fs::path > FindAssumeutxoChainstateDir(const fs::path &data_dir)
Return a path to the snapshot-based chainstate dir, if one exists.
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate) EXCLUSIVE_LOCKS_REQUIRED(std::optional< uint256 > ReadSnapshotBaseBlockhash(fs::path chaindir) EXCLUSIVE_LOCKS_REQUIRED(constexpr std::string_view SNAPSHOT_CHAINSTATE_SUFFIX
Write out the blockhash of the snapshot base block that was used to construct this chainstate.
std::unordered_map< uint256, CBlockIndex, BlockHasher > BlockMap
std::optional< uint256 > ReadSnapshotBaseBlockhash(fs::path chaindir)
constexpr NoRateLimitTag NO_RATE_LIMIT
bilingual_str ErrorString(const Result< T > &result)
std::string ToString(const T &t)
Locale-independent version of std::to_string.
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
static feebumper::Result CheckFeeRate(const CWallet &wallet, const CMutableTransaction &mtx, const CFeeRate &newFeerate, const int64_t maxTxSize, CAmount old_fee, std::vector< bilingual_str > &errors)
Check if the user provided a valid feeRate.
std::shared_ptr< Chain::Notifications > m_notifications
bool IsChildWithParents(const Package &package)
Context-free check that a package is exactly one child and its parents; not all parents need to be pr...
bool IsWellFormedPackage(const Package &txns, PackageValidationState &state)
Context-free package policy checks:
uint256 GetPackageHash(const std::vector< CTransactionRef > &transactions)
Get the hash of the concatenated wtxids of transactions, with wtxids treated as a little-endian numbe...
std::vector< CTransactionRef > Package
A package is an ordered list of transactions.
@ PCKG_POLICY
The package itself is invalid (e.g. too many transactions).
@ PCKG_MEMPOOL_ERROR
Mempool logic error.
@ PCKG_TX
At least one tx is invalid.
std::optional< std::pair< DiagramCheckError, std::string > > ImprovesFeerateDiagram(CTxMemPool::ChangeSet &changeset)
The replacement transaction must improve the feerate diagram of the mempool.
std::optional< std::string > PaysForRBF(CAmount original_fees, CAmount replacement_fees, size_t replacement_vsize, CFeeRate relay_fee, const Txid &txid)
The replacement transaction must pay more fees than the original transactions.
std::optional< std::string > EntriesAndTxidsDisjoint(const CTxMemPool::setEntries &ancestors, const std::set< Txid > &direct_conflicts, const Txid &txid)
Check the intersection between two sets of transactions (a set of mempool entries and a set of txids)...
std::optional< std::string > GetEntriesForConflicts(const CTransaction &tx, CTxMemPool &pool, const CTxMemPool::setEntries &iters_conflicting, CTxMemPool::setEntries &all_conflicts)
Get all descendants of iters_conflicting.
@ FAILURE
New diagram wasn't strictly superior
TxValidationState ValidateInputsStandardness(const CTransaction &tx, const CCoinsViewCache &mapInputs)
Check transaction inputs.
bool SpendsNonAnchorWitnessProg(const CTransaction &tx, const CCoinsViewCache &prevouts)
Check whether this transaction spends any witness program but P2A, including not-yet-defined ones.
bool IsWitnessStandard(const CTransaction &tx, const CCoinsViewCache &mapInputs)
Check if the transaction is over standard P2WSH resources limit: 3600bytes witnessScript size,...
bool IsStandardTx(const CTransaction &tx, const std::optional< unsigned > &max_datacarrier_bytes, bool permit_bare_multisig, const CFeeRate &dust_relay_fee, std::string &reason)
Check for standard transaction types.
static constexpr script_verify_flags STANDARD_SCRIPT_VERIFY_FLAGS
Standard script verification flags that standard transactions will comply with.
static constexpr unsigned int STANDARD_LOCKTIME_VERIFY_FLAGS
Used as the flags parameter to sequence and nLocktime checks in non-consensus code.
static constexpr unsigned int MAX_STANDARD_TX_SIGOPS_COST
The maximum number of sigops we're willing to relay/mine in a single tx.
static constexpr unsigned int MIN_STANDARD_TX_NONWITNESS_SIZE
The minimum non-witness size for transactions we're willing to relay/mine: one larger than 64
static constexpr script_verify_flags STANDARD_NOT_MANDATORY_VERIFY_FLAGS
For convenience, standard but not mandatory verify flags.
unsigned int GetNextWorkRequired(const CBlockIndex *pindexLast, const CBlockHeader *pblock, const Consensus::Params ¶ms)
bool CheckProofOfWork(uint256 hash, unsigned int nBits, const Consensus::Params ¶ms)
Check whether a block hash satisfies the proof-of-work requirement specified by nBits.
static constexpr TransactionSerParams TX_NO_WITNESS
static constexpr TransactionSerParams TX_WITH_WITNESS
static CTransactionRef MakeTransactionRef(Tx &&txIn)
std::shared_ptr< const CTransaction > CTransactionRef
uint256 GetRandHash() noexcept
Generate a random uint256.
std::string ScriptErrorString(const ScriptError serror)
enum ScriptError_t ScriptError
@ SCRIPT_ERR_UNKNOWN_ERROR
uint64_t ReadCompactSize(Stream &is, bool range_check=true)
Decode a CompactSize-encoded variable-length integer.
uint64_t GetSerializeSize(const T &t)
bool CheckSignetBlockSolution(const CBlock &block, const Consensus::Params &consensusParams)
Extract signature and check whether a block has a valid solution.
unsigned char * UCharCast(char *c)
Holds configuration for use during UTXO snapshot load and validation.
AssumeutxoHash hash_serialized
The expected hash of the deserialized UTXO set.
uint64_t m_chain_tx_count
Used to populate the m_chain_tx_count value, which is used during BlockManager::LoadBlockIndex().
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
std::vector< uint256 > vHave
A mutable version of CTransaction.
std::vector< CTxOut > vout
Holds various statistics on transactions within a chain.
User-controlled performance and debug options.
std::shared_ptr< const CBlock > pblock
const CBlockIndex * pindex
Parameters that influence chain consensus.
bool enforce_BIP94
Enforce BIP94 timewarp attack mitigation.
int64_t DifficultyAdjustmentInterval() const
bool signet_blocks
If true, witness commitments contain a payload equal to a Bitcoin Script solution to the signet chall...
int nSubsidyHalvingInterval
std::map< uint256, script_verify_flags > script_flag_exceptions
Hashes of blocks that.
int64_t nPowTargetSpacing
std::chrono::seconds PowTargetSpacing() const
Application-specific storage settings.
fs::path path
Location in the filesystem where leveldb data will be stored.
Data structure storing a fee and size.
Validation result for a transaction evaluated by MemPoolAccept (single or package).
const ResultType m_result_type
Result type.
const TxValidationState m_state
Contains information about why the transaction failed.
@ INVALID
‍Fully validated, valid.
static MempoolAcceptResult Failure(TxValidationState state)
static MempoolAcceptResult FeeFailure(TxValidationState state, CFeeRate effective_feerate, const std::vector< Wtxid > &wtxids_fee_calculations)
static MempoolAcceptResult MempoolTxDifferentWitness(const Wtxid &other_wtxid)
static MempoolAcceptResult MempoolTx(int64_t vsize, CAmount fees)
static MempoolAcceptResult Success(std::list< CTransactionRef > &&replaced_txns, int64_t vsize, CAmount fees, CFeeRate effective_feerate, const std::vector< Wtxid > &wtxids_fee_calculations)
static time_point now() noexcept
Return current system time or mocked time, if set.
static time_point now() noexcept
Return current system time or mocked time, if set.
Validation result for package mempool acceptance.
void Init(const T &tx, std::vector< CTxOut > &&spent_outputs, bool force=false)
Initialize this PrecomputedTransactionData with transaction data.
bool m_spent_outputs_ready
Whether m_spent_outputs is initialized.
std::vector< CTxOut > m_spent_outputs
const char * what() const noexcept override
An options struct for BlockManager, more ergonomically referred to as BlockManager::Options due to th...
const fs::path blocks_dir
An options struct for ChainstateManager, more ergonomically referred to as ChainstateManager::Options...
ValidationSignals * signals
std::optional< int32_t > check_block_index
std::chrono::seconds max_tip_age
If the tip is older than this, the node is considered to be in initial block download.
const CChainParams & chainparams
CoinsViewOptions coins_view
Information about chainstate that notifications are sent from.
bool validated
Whether this is a notification from a chainstate that's been fully validated starting from the genesi...
#define AssertLockNotHeld(cs)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define EXCLUSIVE_LOCKS_REQUIRED(...)
#define LOCKS_EXCLUDED(...)
#define LOG_TIME_MILLIS_WITH_CATEGORY(end_msg, log_category)
#define LOG_TIME_MILLIS_WITH_CATEGORY_MSG_ONCE(end_msg, log_category)
#define TRACEPOINT(context,...)
consteval auto _(util::TranslatedLiteral str)
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
std::optional< std::pair< std::string, CTransactionRef > > SingleTRUCChecks(const CTxMemPool &pool, const CTransactionRef &ptx, const std::vector< CTxMemPoolEntry::CTxMemPoolEntryRef > &mempool_parents, const std::set< Txid > &direct_conflicts, int64_t vsize)
Must be called for every transaction, even if not TRUC.
std::optional< std::string > PackageTRUCChecks(const CTxMemPool &pool, const CTransactionRef &ptx, int64_t vsize, const Package &package, const std::vector< CTxMemPoolEntry::CTxMemPoolEntryRef > &mempool_parents)
Must be called for every transaction that is submitted within a package, even if not TRUC.
bool CheckTransaction(const CTransaction &tx, TxValidationState &state)
bool EvaluateSequenceLocks(const CBlockIndex &block, std::pair< int, int64_t > lockPair)
std::pair< int, int64_t > CalculateSequenceLocks(const CTransaction &tx, int flags, std::vector< int > &prevHeights, const CBlockIndex &block)
Calculates the block height and previous block's median time past at which the transaction will be co...
int64_t GetTransactionSigOpCost(const CTransaction &tx, const CCoinsViewCache &inputs, script_verify_flags flags)
Compute total signature operation cost of a transaction.
unsigned int GetLegacySigOpCount(const CTransaction &tx)
Auxiliary functions for transaction validation (ideally should not be exposed)
bool SequenceLocks(const CTransaction &tx, int flags, std::vector< int > &prevHeights, const CBlockIndex &block)
Check if transaction is final per BIP 68 sequence numbers and can be included in a block.
bool IsFinalTx(const CTransaction &tx, int nBlockHeight, int64_t nBlockTime)
Check if transaction is final and can be included in a block with the specified height and time.
bool TestLockPointValidity(CChain &active_chain, const LockPoints &lp)
Test whether the LockPoints height and time are still valid on the current chain.
static const uint32_t MEMPOOL_HEIGHT
Fake height value used in Coin to signify they are only in the memory pool (since 0....
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
std::string FormatISO8601DateTime(int64_t nTime)
ISO 8601 formatting is preferred.
constexpr int64_t count_seconds(std::chrono::seconds t)
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept, const std::optional< CFeeRate > &client_maxfeerate)
Validate (and maybe submit) a package to the mempool.
static void LimitMempoolSize(CTxMemPool &pool, CCoinsViewCache &coins_cache) EXCLUSIVE_LOCKS_REQUIRED(
bool IsBlockMutated(const CBlock &block, bool check_witness_root)
Check if a block has been mutated (with respect to its merkle root and witness commitments).
script_verify_flags GetBlockScriptFlags(const CBlockIndex &block_index, const ChainstateManager &chainman)
std::optional< LockPoints > CalculateLockPointsAtTip(CBlockIndex *tip, const CCoinsView &coins_view, const CTransaction &tx)
bool CheckInputScripts(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, script_verify_flags flags, bool cacheSigStore, bool cacheFullScriptStore, PrecomputedTransactionData &txdata, ValidationCache &validation_cache, std::vector< CScriptCheck > *pvChecks=nullptr) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Check whether all of this transaction's input scripts succeed.
bool CheckFinalTxAtTip(const CBlockIndex &active_chain_tip, const CTransaction &tx)
CAmount GetBlockSubsidy(int nHeight, const Consensus::Params &consensusParams)
MempoolAcceptResult AcceptToMemoryPool(Chainstate &active_chainstate, const CTransactionRef &tx, int64_t accept_time, bool bypass_limits, bool test_accept)
Try to add a transaction to the mempool.
bool HasValidProofOfWork(std::span< const CBlockHeader > headers, const Consensus::Params &consensusParams)
Check that the proof of work on each blockheader matches the value in nBits.
int ApplyTxInUndo(Coin &&undo, CCoinsViewCache &view, const COutPoint &out)
Restore the UTXO in a Coin at a given COutPoint.
static bool ContextualCheckBlock(const CBlock &block, BlockValidationState &state, const ChainstateManager &chainman, const CBlockIndex *pindexPrev)
NOTE: This function is not currently invoked by ConnectBlock(), so we should consider upgrade issues ...
bool FatalError(Notifications ¬ifications, BlockValidationState &state, const bilingual_str &message)
bool CheckSequenceLocksAtTip(CBlockIndex *tip, const LockPoints &lock_points)
Check if transaction will be BIP68 final in the next block to be created on top of tip.
static bool ContextualCheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, const ChainstateManager &chainman, const CBlockIndex *pindexPrev) EXCLUSIVE_LOCKS_REQUIRED(
Context-dependent validity checks.
static ChainstateManager::Options && Flatten(ChainstateManager::Options &&opts)
Apply default chain params to nullopt members.
static void UpdateTipLog(const ChainstateManager &chainman, const CCoinsViewCache &coins_tip, const CBlockIndex *tip, const std::string &func_name, const std::string &prefix, const std::string &warning_messages, const bool background_validation) EXCLUSIVE_LOCKS_REQUIRED(
static bool CheckInputsFromMempoolAndCache(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &view, const CTxMemPool &pool, script_verify_flags flags, PrecomputedTransactionData &txdata, CCoinsViewCache &coins_tip, ValidationCache &validation_cache) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Checks to avoid mempool polluting consensus critical paths since cached signature and script validity...
static constexpr auto DATABASE_WRITE_INTERVAL_MAX
static bool CheckWitnessMalleation(const CBlock &block, bool expect_witness_commitment, BlockValidationState &state)
CheckWitnessMalleation performs checks for block malleation with regard to its witnesses.
void UpdateCoins(const CTransaction &tx, CCoinsViewCache &inputs, CTxUndo &txundo, int nHeight)
static bool DeleteCoinsDBFromDisk(const fs::path db_path, bool is_snapshot) EXCLUSIVE_LOCKS_REQUIRED(
static bool CheckMerkleRoot(const CBlock &block, BlockValidationState &state)
static constexpr int PRUNE_LOCK_BUFFER
The number of blocks to keep below the deepest prune lock.
arith_uint256 CalculateClaimedHeadersWork(std::span< const CBlockHeader > headers)
Return the sum of the claimed work on a given set of headers.
const std::vector< std::string > CHECKLEVEL_DOC
Documentation for argument 'checklevel'.
bool CheckBlock(const CBlock &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW, bool fCheckMerkleRoot)
Functions for validating blocks and updating the block tree.
static constexpr std::chrono::hours MAX_FEE_ESTIMATION_TIP_AGE
Maximum age of our tip for us to be considered current for fee estimation.
void PruneBlockFilesManual(Chainstate &active_chainstate, int nManualPruneHeight)
Prune block files up to a given height.
static void FlushSnapshotToDisk(CCoinsViewCache &coins_cache, bool snapshot_loaded)
static bool IsCurrentForFeeEstimation(Chainstate &active_chainstate) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
static constexpr auto DATABASE_WRITE_INTERVAL_MIN
Time window to wait between writing blocks/block index and chainstate to disk.
BlockValidationState TestBlockValidity(Chainstate &chainstate, const CBlock &block, const bool check_pow, const bool check_merkle_root)
Verify a block, including transactions.
static bool CheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW=true)
bool IsBIP30Repeat(const CBlockIndex &block_index)
Identifies blocks that overwrote an existing coinbase output in the UTXO set (see BIP30)
static void SnapshotUTXOHashBreakpoint(const util::SignalInterrupt &interrupt)
static bool ShouldCompactChainstate(bool in_ibd)
static SynchronizationState GetSynchronizationState(bool init, bool blockfiles_indexed)
bool IsBIP30Unspendable(const uint256 &block_hash, int block_height)
Identifies blocks which coinbase output was subsequently overwritten in the UTXO set (see BIP30)
TRACEPOINT_SEMAPHORE(validation, block_connected)
static void LimitValidationInterfaceQueue(ValidationSignals &signals) LOCKS_EXCLUDED(cs_main)
static constexpr int MAX_SCRIPTCHECK_THREADS
Maximum number of dedicated script-checking threads allowed.
static const unsigned int MIN_BLOCKS_TO_KEEP
Block files containing a block-height within MIN_BLOCKS_TO_KEEP of ActiveChain().Tip() will not be pr...
Assumeutxo
Chainstate assumeutxo validity.
@ VALIDATED
Every block in the chain has been validated.
@ UNVALIDATED
Blocks after an assumeutxo snapshot have been validated but the snapshot itself has not been validate...
@ INVALID
The assumeutxo snapshot failed validation.
SynchronizationState
Current sync state passed to tip changed callbacks.
constexpr std::array FlushStateModeNames
constexpr int64_t LargeCoinsCacheThreshold(int64_t total_space) noexcept
@ LARGE
The cache is at >= 90% capacity.
@ CRITICAL
The coins cache is in immediate need of a flush.