100 "level 0 reads the blocks from disk",
101 "level 1 verifies block validity",
102 "level 2 verifies undo data",
103 "level 3 checks disconnection of tip blocks",
104 "level 4 tries to reconnect the blocks",
105 "each level includes the checks of the previous levels",
122 : excessiveBlockSize(config.GetMaxBlockSize()), checkPoW(true),
123 checkMerkleRoot(true) {}
160std::optional<std::vector<int>> CalculatePrevHeights(
const CBlockIndex &tip,
162 const CTransaction &tx) {
163 std::vector<int> prev_heights;
164 prev_heights.resize(tx.vin.size());
165 for (
size_t i = 0; i < tx.vin.size(); ++i) {
166 const CTxIn &txin = tx.vin[i];
168 if (!coins.
GetCoin(txin.prevout, coin)) {
169 LogPrintf(
"ERROR: %s: Missing input %d in transaction \'%s\'\n",
170 __func__, i, tx.GetId().GetHex());
175 prev_heights[i] = tip.
nHeight + 1;
186 const CTransaction &tx) {
189 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
190 if (!prev_heights.has_value()) {
195 next_tip.
pprev = tip;
228 int64_t nMedianTimePast) {
229 return nMedianTimePast >=
gArgs.
GetIntArg(
"-replayprotectionactivationtime",
235 if (pindexPrev ==
nullptr) {
256 for (
const CTxIn &txin : tx.vin) {
257 const Coin &coin = view.AccessCoin(txin.prevout);
272 assert(txFrom->GetId() == txin.prevout.GetTxId());
273 assert(txFrom->vout.size() > txin.prevout.GetN());
276 const Coin &coinFromUTXOSet = coins_tip.AccessCoin(txin.prevout);
285 true, txdata, nSigChecksOut);
293 : m_pool(mempool), m_view(&m_dummy),
294 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
295 m_active_chainstate(active_chainstate) {}
301 const int64_t m_accept_time;
302 const bool m_bypass_limits;
310 std::vector<COutPoint> &m_coins_to_uncache;
311 const bool m_test_accept;
312 const unsigned int m_heightOverride;
318 const bool m_package_submission;
324 const bool m_package_feerates;
327 static ATMPArgs SingleAccept(
const Config &config, int64_t accept_time,
329 std::vector<COutPoint> &coins_to_uncache,
331 unsigned int heightOverride) {
349 PackageTestAccept(
const Config &config, int64_t accept_time,
350 std::vector<COutPoint> &coins_to_uncache) {
366 PackageChildWithParents(
const Config &config, int64_t accept_time,
367 std::vector<COutPoint> &coins_to_uncache) {
381 static ATMPArgs SingleInPackageAccept(
const ATMPArgs &package_args) {
383 package_args.m_config,
384 package_args.m_accept_time,
386 package_args.m_coins_to_uncache,
387 package_args.m_test_accept,
388 package_args.m_heightOverride,
400 ATMPArgs(
const Config &config, int64_t accept_time,
bool bypass_limits,
401 std::vector<COutPoint> &coins_to_uncache,
bool test_accept,
402 unsigned int height_override,
bool package_submission,
403 bool package_feerates)
404 : m_config{config}, m_accept_time{accept_time},
405 m_bypass_limits{bypass_limits},
406 m_coins_to_uncache{coins_to_uncache}, m_test_accept{test_accept},
407 m_heightOverride{height_override},
408 m_package_submission{package_submission},
409 m_package_feerates(package_feerates) {}
424 AcceptMultipleTransactions(
const std::vector<CTransactionRef> &txns,
441 AcceptSubPackage(
const std::vector<CTransactionRef> &subpackage,
459 const uint32_t next_block_script_verify_flags)
461 m_next_block_script_verify_flags(next_block_script_verify_flags) {
468 std::unique_ptr<CTxMemPoolEntry> m_entry;
506 const uint32_t m_next_block_script_verify_flags;
507 int m_sig_checks_standard;
515 bool PreChecks(ATMPArgs &args, Workspace &ws)
522 bool ConsensusScriptChecks(
const ATMPArgs &args, Workspace &ws)
528 bool Finalize(
const ATMPArgs &args, Workspace &ws)
536 bool SubmitPackage(
const ATMPArgs &args, std::vector<Workspace> &workspaces,
538 std::map<TxId, MempoolAcceptResult> &results)
542 bool CheckFeeRate(
size_t package_size,
size_t package_vsize,
548 const Amount mempoolRejectFee =
549 m_pool.GetMinFee().GetFee(package_vsize);
552 package_fee < mempoolRejectFee) {
553 return state.Invalid(
555 "mempool min fee not met",
556 strprintf(
"%d < %d", package_fee, mempoolRejectFee));
561 if (package_fee < m_pool.m_min_relay_feerate.GetFee(package_size)) {
562 return state.Invalid(
564 "min relay fee not met",
566 m_pool.m_min_relay_feerate.GetFee(package_size)));
581bool MemPoolAccept::PreChecks(ATMPArgs &args, Workspace &ws) {
585 const CTransaction &tx = *ws.m_ptx;
586 const TxId &txid = ws.m_ptx->GetId();
589 const int64_t nAcceptTime = args.m_accept_time;
590 const bool bypass_limits = args.m_bypass_limits;
591 std::vector<COutPoint> &coins_to_uncache = args.m_coins_to_uncache;
592 const unsigned int heightOverride = args.m_heightOverride;
604 if (m_pool.m_require_standard &&
606 m_pool.m_permit_bare_multisig,
607 m_pool.m_dust_relay_feerate, reason)) {
616 *
Assert(m_active_chainstate.m_chain.Tip()),
617 args.m_config.GetChainParams().GetConsensus(), tx, ctxState)) {
626 if (m_pool.exists(txid)) {
628 "txn-already-in-mempool");
632 for (
const CTxIn &txin : tx.vin) {
633 if (
const auto ptxConflicting = m_pool.GetConflictTx(txin.prevout)) {
634 if (m_pool.isAvalancheFinalized(ptxConflicting->GetId())) {
636 "finalized-tx-conflict");
641 "txn-mempool-conflict");
645 m_view.SetBackend(m_viewmempool);
649 for (
const CTxIn &txin : tx.vin) {
651 coins_to_uncache.push_back(txin.prevout);
658 if (!m_view.HaveCoin(txin.prevout)) {
660 for (
size_t out = 0; out < tx.vout.size(); out++) {
665 "txn-already-known");
672 "bad-txns-inputs-missingorspent");
677 if (!m_view.HaveInputs(tx)) {
679 "bad-txns-inputs-spent");
683 m_view.GetBestBlock();
688 m_view.SetBackend(m_dummy);
690 assert(m_active_chainstate.m_blockman.LookupBlockIndex(
691 m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
700 m_active_chainstate.m_chain.Tip(), m_view, tx)};
701 if (!lock_points.has_value() ||
711 m_active_chainstate.m_chain.Height() + 1,
718 if (m_pool.m_require_standard &&
721 "bad-txns-nonstandard-inputs");
725 ws.m_modified_fees = ws.m_base_fees;
726 m_pool.ApplyDelta(txid, ws.m_modified_fees);
728 unsigned int nSize = tx.GetTotalSize();
731 const uint32_t scriptVerifyFlags =
735 ws.m_precomputed_txdata, ws.m_sig_checks_standard)) {
740 ws.m_entry = std::make_unique<CTxMemPoolEntry>(
741 ptx, ws.m_base_fees, nAcceptTime,
742 heightOverride ? heightOverride : m_active_chainstate.m_chain.Height(),
743 ws.m_sig_checks_standard, lock_points.value());
745 ws.m_vsize = ws.m_entry->GetTxVirtualSize();
753 if (!bypass_limits &&
755 m_pool.m_min_relay_feerate.GetFee(ws.m_ptx->GetTotalSize())) {
762 m_pool.m_min_relay_feerate.GetFee(nSize)));
767 if (!bypass_limits && !args.m_package_feerates &&
768 !CheckFeeRate(nSize, ws.m_vsize, ws.m_modified_fees, state)) {
775bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs &args, Workspace &ws) {
778 const CTransaction &tx = *ws.m_ptx;
779 const TxId &txid = tx.GetId();
793 int nSigChecksConsensus;
795 tx, state, m_view, m_pool, ws.m_next_block_script_verify_flags,
796 ws.m_precomputed_txdata, nSigChecksConsensus,
797 m_active_chainstate.CoinsTip())) {
801 LogPrintf(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against "
802 "latest-block but not STANDARD flags %s, %s\n",
807 if (ws.m_sig_checks_standard != nSigChecksConsensus) {
812 "%s: BUG! PLEASE REPORT THIS! SigChecks count differed between "
813 "standard and consensus flags in %s",
822 std::vector<Coin> spent_coins;
823 spent_coins.reserve(ptx->vin.size());
824 for (
const CTxIn &input : ptx->vin) {
826 const bool coinFound = coins_view.
GetCoin(input.prevout, coin);
828 spent_coins.push_back(std::move(coin));
833bool MemPoolAccept::Finalize(
const ATMPArgs &args, Workspace &ws) {
836 const TxId &txid = ws.m_ptx->GetId();
838 const bool bypass_limits = args.m_bypass_limits;
843 m_pool.addUnchecked(entry);
847 std::make_shared<
const std::vector<Coin>>(
848 getSpentCoins(ws.m_ptx, m_view)),
849 m_pool.GetAndIncrementSequence());
856 if (!args.m_package_submission && !bypass_limits) {
857 m_pool.LimitSize(m_active_chainstate.CoinsTip());
858 if (!m_pool.exists(txid)) {
868bool MemPoolAccept::SubmitPackage(
869 const ATMPArgs &args, std::vector<Workspace> &workspaces,
871 std::map<TxId, MempoolAcceptResult> &results) {
876 workspaces.cbegin(), workspaces.cend(),
877 [
this](
const auto &ws) { return !m_pool.exists(ws.m_ptx->GetId()); }));
879 bool all_submitted =
true;
886 for (Workspace &ws : workspaces) {
887 if (!ConsensusScriptChecks(args, ws)) {
888 results.emplace(ws.m_ptx->GetId(),
891 all_submitted =
false;
894 strprintf(
"BUG! PolicyScriptChecks succeeded but "
895 "ConsensusScriptChecks failed: %s",
896 ws.m_ptx->GetId().ToString()));
906 if (!Finalize(args, ws)) {
907 results.emplace(ws.m_ptx->GetId(),
910 all_submitted =
false;
912 strprintf(
"BUG! Adding to mempool failed: %s",
913 ws.m_ptx->GetId().ToString()));
919 m_pool.LimitSize(m_active_chainstate.CoinsTip());
921 std::vector<TxId> all_package_txids;
922 all_package_txids.reserve(workspaces.size());
923 std::transform(workspaces.cbegin(), workspaces.cend(),
924 std::back_inserter(all_package_txids),
925 [](
const auto &ws) { return ws.m_ptx->GetId(); });
929 for (Workspace &ws : workspaces) {
930 const auto effective_feerate =
931 args.m_package_feerates
932 ? ws.m_package_feerate
934 static_cast<uint32_t
>(ws.m_vsize)};
935 const auto effective_feerate_txids =
936 args.m_package_feerates ? all_package_txids
937 : std::vector<TxId>({ws.m_ptx->GetId()});
938 results.emplace(ws.m_ptx->GetId(),
941 effective_feerate_txids));
943 return all_submitted;
954 const CBlockIndex *tip = m_active_chainstate.m_chain.Tip();
959 const std::vector<TxId> single_txid{ws.m_ptx->GetId()};
964 if (!PreChecks(args, ws)) {
965 if (ws.m_state.GetResult() ==
970 ws.m_state,
CFeeRate(ws.m_modified_fees, ws.m_vsize),
976 if (!ConsensusScriptChecks(args, ws)) {
980 const TxId txid = ptx->GetId();
990 if (
auto it = m_pool.mapNextTx.lower_bound(COutPoint{txid, 0});
991 it != m_pool.mapNextTx.end() && it->first->GetTxId() == txid) {
992 LogPrintf(
"%s: BUG! PLEASE REPORT THIS! Attempt to add txid %s, but "
993 "its outputs are already spent in the "
997 "txn-child-before-parent");
1001 const CFeeRate effective_feerate{ws.m_modified_fees,
1002 static_cast<uint32_t
>(ws.m_vsize)};
1004 if (args.m_test_accept) {
1006 effective_feerate, single_txid);
1009 if (!Finalize(args, ws)) {
1013 Assume(ws.m_state.GetResult() ==
1016 ws.m_state,
CFeeRate(ws.m_modified_fees, ws.m_vsize), single_txid);
1020 effective_feerate, single_txid);
1024 const std::vector<CTransactionRef> &txns, ATMPArgs &args) {
1034 std::vector<Workspace> workspaces{};
1035 workspaces.reserve(txns.size());
1037 txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1038 [
this](
const auto &tx) {
1040 tx, GetNextBlockScriptFlags(m_active_chainstate.m_chain.Tip(),
1041 m_active_chainstate.m_chainman));
1043 std::map<TxId, MempoolAcceptResult> results;
1049 std::vector<TxId> valid_txids;
1050 for (Workspace &ws : workspaces) {
1051 if (!PreChecks(args, ws)) {
1053 "transaction failed");
1056 results.emplace(ws.m_ptx->GetId(),
1059 std::move(results));
1063 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1064 valid_txids.push_back(ws.m_ptx->GetId());
1079 const auto m_total_size = std::accumulate(
1080 workspaces.cbegin(), workspaces.cend(), int64_t{0},
1081 [](int64_t
sum,
auto &ws) { return sum + ws.m_ptx->GetTotalSize(); });
1082 const auto m_total_vsize =
1083 std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1084 [](int64_t
sum,
auto &ws) { return sum + ws.m_vsize; });
1085 const auto m_total_modified_fees = std::accumulate(
1086 workspaces.cbegin(), workspaces.cend(),
Amount::zero(),
1087 [](
Amount sum,
auto &ws) { return sum + ws.m_modified_fees; });
1088 const CFeeRate package_feerate(m_total_modified_fees, m_total_vsize);
1089 std::vector<TxId> all_package_txids;
1090 all_package_txids.reserve(workspaces.size());
1091 std::transform(workspaces.cbegin(), workspaces.cend(),
1092 std::back_inserter(all_package_txids),
1093 [](
const auto &ws) { return ws.m_ptx->GetId(); });
1095 if (args.m_package_feerates &&
1096 !CheckFeeRate(m_total_size, m_total_vsize, m_total_modified_fees,
1097 placeholder_state)) {
1099 "transaction failed");
1101 package_state, {{workspaces.back().m_ptx->GetId(),
1104 CFeeRate(m_total_modified_fees, m_total_vsize),
1105 all_package_txids)}});
1108 for (Workspace &ws : workspaces) {
1109 ws.m_package_feerate = package_feerate;
1110 const TxId &ws_txid = ws.m_ptx->GetId();
1111 if (args.m_test_accept &&
1112 std::find(valid_txids.begin(), valid_txids.end(), ws_txid) !=
1113 valid_txids.end()) {
1114 const auto effective_feerate =
1115 args.m_package_feerates
1116 ? ws.m_package_feerate
1118 static_cast<uint32_t
>(ws.m_vsize)};
1119 const auto effective_feerate_txids =
1120 args.m_package_feerates ? all_package_txids
1121 : std::vector<TxId>{ws.m_ptx->GetId()};
1125 results.emplace(ws_txid,
1127 ws.m_vsize, ws.m_base_fees, effective_feerate,
1128 effective_feerate_txids));
1132 if (args.m_test_accept) {
1136 if (!SubmitPackage(args, workspaces, package_state, results)) {
1145MemPoolAccept::AcceptSubPackage(
const std::vector<CTransactionRef> &subpackage,
1151 if (subpackage.size() > 1) {
1152 return AcceptMultipleTransactions(subpackage, args);
1154 const auto &tx = subpackage.front();
1155 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(args);
1156 const auto single_res = AcceptSingleTransaction(tx, single_args);
1158 if (single_res.m_result_type !=
1161 "transaction failed");
1164 {{tx->GetId(), single_res}});
1194 for (
const auto &outpoint : m_viewmempool.GetNonBaseCoins()) {
1198 m_view.Uncache(outpoint);
1201 m_viewmempool.Reset();
1217 if (!
CheckPackage(package, package_state_quit_early)) {
1226 "package-not-child-with-parents");
1231 assert(package.size() > 1);
1234 const auto &child = package.back();
1235 std::unordered_set<TxId, SaltedTxIdHasher> unconfirmed_parent_txids;
1237 package.cbegin(), package.cend() - 1,
1238 std::inserter(unconfirmed_parent_txids, unconfirmed_parent_txids.end()),
1239 [](
const auto &tx) { return tx->GetId(); });
1248 const CCoinsViewCache &coins_tip_cache = m_active_chainstate.CoinsTip();
1249 for (
const auto &input : child->vin) {
1251 args.m_coins_to_uncache.push_back(input.prevout);
1258 m_view.
SetBackend(m_active_chainstate.CoinsTip());
1259 const auto package_or_confirmed = [
this, &unconfirmed_parent_txids](
1260 const auto &input) {
1261 return unconfirmed_parent_txids.count(input.prevout.GetTxId()) > 0 ||
1262 m_view.HaveCoin(input.prevout);
1264 if (!std::all_of(child->vin.cbegin(), child->vin.cend(),
1265 package_or_confirmed)) {
1266 package_state_quit_early.
Invalid(
1268 "package-not-child-with-unconfirmed-parents");
1274 m_view.SetBackend(m_dummy);
1280 std::map<TxId, MempoolAcceptResult> results_final;
1285 std::map<TxId, MempoolAcceptResult> individual_results_nonfinal;
1286 bool quit_early{
false};
1287 std::vector<CTransactionRef> txns_package_eval;
1288 for (
const auto &tx : package) {
1289 const auto &txid = tx->GetId();
1292 if (m_pool.exists(txid)) {
1306 auto iter = m_pool.GetIter(txid);
1307 assert(iter != std::nullopt);
1309 (*iter.value())->GetTxSize(),
1310 (*iter.value())->GetFee()));
1314 const auto single_package_res = AcceptSubPackage({tx}, args);
1315 const auto &single_res = single_package_res.m_tx_results.at(txid);
1316 if (single_res.m_result_type ==
1321 assert(m_pool.exists(txid));
1322 results_final.emplace(txid, single_res);
1323 }
else if (single_res.m_state.GetResult() !=
1325 single_res.m_state.GetResult() !=
1340 package_state_quit_early.
Invalid(
1342 individual_results_nonfinal.emplace(txid, single_res);
1344 individual_results_nonfinal.emplace(txid, single_res);
1345 txns_package_eval.push_back(tx);
1350 auto multi_submission_result =
1351 quit_early || txns_package_eval.empty()
1353 : AcceptSubPackage(txns_package_eval, args);
1355 multi_submission_result.m_state;
1360 m_pool.LimitSize(m_active_chainstate.CoinsTip());
1362 for (
const auto &tx : package) {
1363 const auto &txid = tx->GetId();
1364 if (multi_submission_result.m_tx_results.count(txid) > 0) {
1367 Assume(results_final.count(txid) == 0);
1371 const auto &txresult =
1372 multi_submission_result.m_tx_results.at(txid);
1373 if (txresult.m_result_type ==
1375 !m_pool.exists(txid)) {
1377 "transaction failed");
1381 results_final.emplace(
1384 results_final.emplace(txid, txresult);
1386 }
else if (
const auto final_it{results_final.find(txid)};
1387 final_it != results_final.end()) {
1390 Assume(final_it->second.m_result_type !=
1392 Assume(individual_results_nonfinal.count(txid) == 0);
1393 if (!m_pool.exists(tx->GetId())) {
1395 "transaction failed");
1400 results_final.erase(txid);
1401 results_final.emplace(
1404 }
else if (
const auto non_final_it{
1405 individual_results_nonfinal.find(txid)};
1406 non_final_it != individual_results_nonfinal.end()) {
1407 Assume(non_final_it->second.m_result_type ==
1410 results_final.emplace(txid, non_final_it->second);
1413 Assume(results_final.size() == package.size());
1415 std::move(results_final));
1421 int64_t accept_time,
bool bypass_limits,
1423 unsigned int heightOverride) {
1428 std::vector<COutPoint> coins_to_uncache;
1429 auto args = MemPoolAccept::ATMPArgs::SingleAccept(
1431 coins_to_uncache, test_accept, heightOverride);
1433 .AcceptSingleTransaction(tx, args);
1441 for (
const COutPoint &outpoint : coins_to_uncache) {
1458 assert(!package.empty());
1459 assert(std::all_of(package.cbegin(), package.cend(),
1460 [](
const auto &tx) { return tx != nullptr; }));
1464 std::vector<COutPoint> coins_to_uncache;
1468 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(
1469 config,
GetTime(), coins_to_uncache);
1470 return MemPoolAccept(pool, active_chainstate)
1471 .AcceptMultipleTransactions(package, args);
1473 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(
1474 config,
GetTime(), coins_to_uncache);
1475 return MemPoolAccept(pool, active_chainstate)
1476 .AcceptPackage(package, args);
1482 if (test_accept || result.m_state.IsInvalid()) {
1483 for (
const COutPoint &hashTx : coins_to_uncache) {
1496 if (halvings >= 64) {
1507 : m_dbview{
std::move(db_params),
std::move(options)},
1508 m_catcherview(&m_dbview) {}
1510void CoinsViews::InitCache() {
1512 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1517 std::optional<BlockHash> from_snapshot_blockhash)
1518 : m_mempool(mempool), m_blockman(blockman), m_chainman(chainman),
1519 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1525 if (!m_cached_snapshot_base) {
1526 m_cached_snapshot_base =
Assert(
1529 return m_cached_snapshot_base;
1533 bool should_wipe, std::string leveldb_name) {
1540 .cache_bytes = cache_size_bytes,
1541 .memory_only = in_memory,
1542 .wipe_data = should_wipe,
1548void Chainstate::InitCoinsCache(
size_t cache_size_bytes) {
1574 if (chain.Tip() ==
nullptr) {
1583 LogPrintf(
"Leaving InitialBlockDownload (latching to false)\n");
1610 std::string warning =
1611 std::string(
"'Warning: Large-work fork detected, forking after "
1618 LogPrintf(
"%s: Warning: Large fork found\n forking the "
1619 "chain at height %d (%s)\n lasting to height %d "
1620 "(%s).\nChain state database corruption likely.\n",
1627 LogPrintf(
"%s: Warning: Found invalid chain at least ~6 blocks "
1628 "longer than our best chain.\nChain state database "
1629 "corruption likely.\n",
1684 m_avalancheFinalizedBlockIndex = pindexNew->
pprev;
1687 LogPrintf(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n",
1694 LogPrintf(
"%s: current best=%s height=%d log2_work=%f date=%s\n",
1706 pindex->nStatus = pindex->nStatus.withFailed();
1716 if (tx.IsCoinBase()) {
1720 txundo.
vprevout.reserve(tx.vin.size());
1721 for (
const CTxIn &txin : tx.vin) {
1735 const CScript &scriptSig =
ptxTo->vin[
nIn].scriptSig;
1757 bool sigCacheStore,
bool scriptCacheStore,
1761 std::vector<CScriptCheck> *pvChecks) {
1763 assert(!tx.IsCoinBase());
1766 pvChecks->reserve(tx.vin.size());
1776 (pBlockLimitSigChecks &&
1779 "too-many-sigchecks");
1784 int nSigChecksTotal = 0;
1786 for (
size_t i = 0; i < tx.vin.size(); i++) {
1787 const COutPoint &prevout = tx.vin[i].prevout;
1799 &txLimitSigChecks, pBlockLimitSigChecks);
1803 pvChecks->push_back(std::move(check));
1813 uint32_t mandatoryFlags =
1814 flags & ~STANDARD_NOT_MANDATORY_VERIFY_FLAGS;
1815 if (
flags != mandatoryFlags) {
1822 sigCacheStore, txdata);
1826 strprintf(
"non-mandatory-script-verify-flag (%s)",
1842 strprintf(
"mandatory-script-verify-flag-failed (%s)",
1849 nSigChecksOut = nSigChecksTotal;
1851 if (scriptCacheStore && !pvChecks) {
1863 return state.
Error(strMessage);
1868 const COutPoint &out) {
1876 if (undo.GetHeight() == 0) {
1899 view.
AddCoin(out, std::move(undo), !fClean);
1914 error(
"DisconnectBlock(): failure reading undo data");
1918 return ApplyBlockUndo(std::move(blockUndo), block, pindex, view);
1926 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
1927 error(
"DisconnectBlock(): block and undo data inconsistent");
1932 for (
size_t i = 1; i < block.
vtx.size(); i++) {
1933 const CTransaction &tx = *(block.
vtx[i]);
1935 if (txundo.
vprevout.size() != tx.vin.size()) {
1936 error(
"DisconnectBlock(): transaction and undo data inconsistent");
1940 for (
size_t j = 0; j < tx.vin.size(); j++) {
1941 const COutPoint &out = tx.vin[j].prevout;
1953 for (
const auto &ptx : block.
vtx) {
1954 const CTransaction &tx = *ptx;
1955 const TxId &txid = tx.GetId();
1956 const bool is_coinbase = tx.IsCoinBase();
1960 for (
size_t o = 0; o < tx.vout.size(); o++) {
1961 if (tx.vout[o].scriptPubKey.IsUnspendable()) {
1965 COutPoint out(txid, o);
1967 bool is_spent = view.
SpendCoin(out, &coin);
1968 if (!is_spent || tx.vout[o] != coin.
GetTxOut() ||
2104 if (!
CheckBlock(block, state, consensusParams,
2111 return AbortNode(state,
"Corrupt block found indicating potential "
2112 "hardware failure; shutting down");
2114 return error(
"%s: Consensus::CheckBlock: %s", __func__,
2135 bool fScriptChecks =
true;
2144 BlockMap::const_iterator it{
2147 if (it->second.GetAncestor(pindex->
nHeight) == pindex &&
2175 consensusParams) <= 60 * 60 * 24 * 7 * 2);
2197 bool fEnforceBIP30 = !((pindex->
nHeight == 91842 &&
2199 uint256S(
"0x00000000000a4d0a398161ffc163c503763"
2200 "b1f4360639393e0e4c8e300e0caec")) ||
2203 uint256S(
"0x00000000000743f190a18c5577a3c2d2a1f"
2204 "610ae9601ac046a38084ccb7cd721")));
2235 static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2272 (!pindexBIP34height ||
2278 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2279 for (
const auto &tx : block.
vtx) {
2280 for (
size_t o = 0; o < tx->vout.size(); o++) {
2281 if (view.
HaveCoin(COutPoint(tx->GetId(), o))) {
2282 LogPrintf(
"ERROR: ConnectBlock(): tried to overwrite "
2292 int nLockTimeFlags = 0;
2306 std::vector<int> prevheights;
2318 std::vector<TxSigCheckLimiter> nSigChecksTxLimiters;
2319 nSigChecksTxLimiters.resize(block.
vtx.size() - 1);
2322 blockundo.
vtxundo.resize(block.
vtx.size() - 1);
2329 for (
const auto &ptx : block.
vtx) {
2332 }
catch (
const std::logic_error &e) {
2341 LogPrintf(
"ERROR: ConnectBlock(): tried to overwrite transaction\n");
2350 for (
const auto &ptx : block.
vtx) {
2351 const CTransaction &tx = *ptx;
2352 const bool isCoinBase = tx.IsCoinBase();
2353 nInputs += tx.vin.size();
2367 return error(
"%s: Consensus::CheckTxInputs: %s, %s", __func__,
2368 tx.GetId().ToString(), state.
ToString());
2374 LogPrintf(
"ERROR: %s: accumulated fee in the block out of range.\n",
2377 "bad-txns-accumulated-fee-outofrange");
2388 prevheights.resize(tx.vin.size());
2389 for (
size_t j = 0; j < tx.vin.size(); j++) {
2393 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2394 LogPrintf(
"ERROR: %s: contains a non-BIP68-final transaction\n",
2397 "bad-txns-nonfinal");
2402 bool fCacheResults = fJustCheck;
2405 if (!fEnforceSigCheck) {
2412 std::vector<CScriptCheck> vChecks;
2414 if (fScriptChecks &&
2417 nSigChecksRet, nSigChecksTxLimiters[txIndex],
2418 &nSigChecksBlockLimiter, &vChecks)) {
2425 "ConnectBlock(): CheckInputScripts on %s failed with %s",
2426 tx.GetId().ToString(), state.
ToString());
2429 control.
Add(std::move(vChecks));
2443 " - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) "
2444 "[%.2fs (%.2fms/blk)]\n",
2445 (
unsigned)block.
vtx.size(),
MILLI * (nTime3 - nTime2),
2446 MILLI * (nTime3 - nTime2) / block.
vtx.size(),
2447 nInputs <= 1 ? 0 :
MILLI * (nTime3 - nTime2) / (nInputs - 1),
2450 const Amount blockReward =
2452 if (block.
vtx[0]->GetValueOut() > blockReward) {
2453 LogPrintf(
"ERROR: ConnectBlock(): coinbase pays too much (actual=%d vs "
2455 block.
vtx[0]->GetValueOut(), blockReward);
2464 if (!control.
Wait()) {
2466 "blk-bad-inputs",
"parallel script check failed");
2473 " - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n",
2474 nInputs - 1,
MILLI * (nTime4 - nTime2),
2475 nInputs <= 1 ? 0 :
MILLI * (nTime4 - nTime2) / (nInputs - 1),
2482 if (!
m_blockman.WriteUndoDataForBlock(blockundo, state, *pindex)) {
2500 TRACE6(validation, block_connected, block_hash.data(), pindex->
nHeight,
2501 block.
vtx.size(), nInputs, nSigChecksRet,
2503 nTime5 - nTimeStart);
2516Chainstate::GetCoinsCacheSizeState(
size_t max_coins_cache_size_bytes,
2517 size_t max_mempool_size_bytes) {
2521 int64_t nTotalSpace =
2522 max_coins_cache_size_bytes +
2523 std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2526 static constexpr int64_t MAX_BLOCK_COINSDB_USAGE_BYTES =
2528 int64_t large_threshold = std::max(
2529 (9 * nTotalSpace) / 10, nTotalSpace - MAX_BLOCK_COINSDB_USAGE_BYTES);
2531 if (cacheSize > nTotalSpace) {
2532 LogPrintf(
"Cache size (%s) exceeds total space (%s)\n", cacheSize,
2535 }
else if (cacheSize > large_threshold) {
2545 std::set<int> setFilesToPrune;
2546 bool full_flush_completed =
false;
2553 bool fFlushForPrune =
false;
2554 bool fDoFullFlush =
false;
2566 std::optional<std::string> limiting_lock;
2568 for (
const auto &prune_lock :
m_blockman.m_prune_locks) {
2569 if (prune_lock.second.height_first ==
2570 std::numeric_limits<int>::max()) {
2575 const int lock_height{prune_lock.second.height_first -
2577 last_prune = std::max(1, std::min(last_prune, lock_height));
2578 if (last_prune == lock_height) {
2579 limiting_lock = prune_lock.first;
2583 if (limiting_lock) {
2585 limiting_lock.value(), last_prune);
2588 if (nManualPruneHeight > 0) {
2593 std::min(last_prune, nManualPruneHeight),
2605 if (!setFilesToPrune.empty()) {
2606 fFlushForPrune =
true;
2609 "prunedblockfiles",
true);
2614 const auto nNow = GetTime<std::chrono::microseconds>();
2639 fCacheCritical || fPeriodicFlush || fFlushForPrune;
2641 if (fDoFullFlush || fPeriodicWrite) {
2644 return AbortNode(state,
"Disk space is too low!",
2645 _(
"Disk space is too low!"));
2658 "%s: Failed to flush block file.\n",
2670 state,
"Failed to write to block index database");
2675 if (fFlushForPrune) {
2685 if (fDoFullFlush && !
CoinsTip().GetBestBlock().IsNull()) {
2687 strprintf(
"write coins cache to disk (%d coins, %.2fkB)",
2688 coins_count, coins_mem_usage / 1000),
2698 48 * 2 * 2 *
CoinsTip().GetCacheSize())) {
2699 return AbortNode(state,
"Disk space is too low!",
2700 _(
"Disk space is too low!"));
2706 return AbortNode(state,
"Failed to write to coin database");
2709 full_flush_completed =
true;
2714 GetTimeMicros() - nNow.count(), uint32_t(mode), coins_count,
2715 uint64_t(coins_mem_usage), fFlushForPrune);
2718 if (full_flush_completed) {
2723 }
catch (
const std::runtime_error &e) {
2724 return AbortNode(state, std::string(
"System error while flushing: ") +
2733 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
2742 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
2749 const std::string &func_name,
2750 const std::string &
prefix)
2753 LogPrintf(
"%s%s: new best=%s height=%d version=0x%08x log2_work=%f tx=%ld "
2754 "date='%s' progress=%f cache=%.1fMiB(%utxo)\n",
2760 coins_tip.DynamicMemoryUsage() * (1.0 / (1 << 20)),
2761 coins_tip.GetCacheSize());
2764void Chainstate::UpdateTip(
const CBlockIndex *pindexNew) {
2766 const auto &coins_tip =
CoinsTip();
2775 constexpr int BACKGROUND_LOG_INTERVAL = 2000;
2776 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2778 "[background validation] ");
2794 UpdateTipLog(coins_tip, pindexNew, params, __func__,
"");
2821 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2824 return error(
"DisconnectTip(): Failed to read block");
2832 if (DisconnectBlock(block, pindexDelete, view) !=
2834 return error(
"DisconnectTip(): DisconnectBlock %s failed",
2838 bool flushed = view.
Flush();
2848 const int max_height_first{pindexDelete->
nHeight - 1};
2849 for (
auto &prune_lock :
m_blockman.m_prune_locks) {
2850 if (prune_lock.second.height_first <= max_height_first) {
2854 prune_lock.second.height_first = max_height_first;
2856 prune_lock.first, max_height_first);
2869 if (pindexDelete->
pprev !=
nullptr &&
2873 "Disconnecting mempool due to rewind of upgrade block\n");
2874 if (disconnectpool) {
2880 if (disconnectpool) {
2887 UpdateTip(pindexDelete->
pprev);
2907 const std::shared_ptr<const CBlock> &pblock,
2920 std::shared_ptr<const CBlock> pthisBlock;
2922 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
2924 return AbortNode(state,
"Failed to read block");
2926 pthisBlock = pblockNew;
2928 pthisBlock = pblock;
2931 const CBlock &blockConnecting = *pthisBlock;
2942 bool rv =
ConnectBlock(blockConnecting, state, pindexNew, view,
2951 return error(
"%s: ConnectBlock %s failed, %s", __func__,
2972 const Amount blockReward =
2976 std::vector<std::unique_ptr<ParkingPolicy>> parkingPolicies;
2977 parkingPolicies.emplace_back(std::make_unique<MinerFundPolicy>(
2978 consensusParams, *pindexNew, blockConnecting, blockReward));
2994 parkingPolicies.emplace_back(
2995 std::make_unique<RTTPolicy>(consensusParams,
3000 parkingPolicies.emplace_back(
3001 std::make_unique<StakingRewardsPolicy>(
3002 *
avalanche, consensusParams, *pindexNew,
3003 blockConnecting, blockReward));
3006 parkingPolicies.emplace_back(
3007 std::make_unique<PreConsensusPolicy>(
3008 *pindexNew, blockConnecting,
m_mempool));
3013 if (std::find_if_not(parkingPolicies.begin(), parkingPolicies.end(),
3014 [&](
const auto &policy) {
3015 bool ret = (*policy)(blockPolicyState);
3018 "Park block because it "
3019 "violated a block policy: %s\n",
3020 blockPolicyState.ToString());
3023 }) != parkingPolicies.end()) {
3024 pindexNew->nStatus = pindexNew->nStatus.withParked();
3034 " - Connect total: %.2fms [%.2fs (%.2fms/blk)]\n",
3037 bool flushed = view.
Flush();
3055 " - Writing chainstate: %.2fms [%.2fs (%.2fms/blk)]\n",
3061 disconnectpool.removeForBlock(blockConnecting.vtx, *m_mempool);
3066 if (pindexNew->
pprev !=
nullptr &&
3071 "Disconnecting mempool due to acceptance of upgrade block\n");
3072 disconnectpool.importMempool(*m_mempool);
3077 m_chain.SetTip(*pindexNew);
3078 UpdateTip(pindexNew);
3084 " - Connect postprocess: %.2fms [%.2fs (%.2fms/blk)]\n",
3094 if (
this != &m_chainman.ActiveChainstate()) {
3098 m_chainman.MaybeCompleteSnapshotValidation();
3110 std::vector<const CBlockIndex *> &blocksToReconcile,
bool fAutoUnpark) {
3117 std::set<CBlockIndex *, CBlockIndexWorkComparator>::reverse_iterator
3129 if (m_avalancheFinalizedBlockIndex &&
3131 LogPrintf(
"Park block %s because it forks prior to the "
3132 "avalanche finalized chaintip.\n",
3134 pindexNew->nStatus = pindexNew->nStatus.withParked();
3145 bool hasValidAncestor =
true;
3146 while (hasValidAncestor && pindexTest && pindexTest != pindexFork) {
3151 bool fParkedChain = pindexTest->nStatus.isOnParkedChain();
3152 if (fAutoUnpark && fParkedChain) {
3158 if (!pindexTip || !pindexFork) {
3172 pindexExtraPow = pindexExtraPow->
pprev;
3177 requiredWork += (deltaWork >> 1);
3188 LogPrintf(
"Unpark chain up to block %s as it has "
3189 "accumulated enough PoW.\n",
3191 fParkedChain =
false;
3200 bool fInvalidChain = pindexTest->nStatus.isInvalid();
3201 bool fMissingData = !pindexTest->nStatus.hasData();
3202 if (!(fInvalidChain || fParkedChain || fMissingData)) {
3205 pindexTest = pindexTest->
pprev;
3211 hasValidAncestor =
false;
3214 if (fInvalidChain && (
m_chainman.m_best_invalid ==
nullptr ||
3220 if (fParkedChain && (
m_chainman.m_best_parked ==
nullptr ||
3226 LogPrintf(
"Considered switching to better tip %s but that chain "
3227 "contains a%s%s%s block.\n",
3229 fInvalidChain ?
"n invalid" :
"",
3230 fParkedChain ?
" parked" :
"",
3231 fMissingData ?
" missing-data" :
"");
3235 while (pindexTest != pindexFailed) {
3236 if (fInvalidChain || fParkedChain) {
3237 pindexFailed->nStatus =
3238 pindexFailed->nStatus.withFailedParent(fInvalidChain)
3239 .withParkedParent(fParkedChain);
3240 }
else if (fMissingData) {
3246 std::make_pair(pindexFailed->
pprev, pindexFailed));
3249 pindexFailed = pindexFailed->
pprev;
3252 if (fInvalidChain || fParkedChain) {
3259 blocksToReconcile.push_back(pindexNew);
3262 if (hasValidAncestor) {
3295 const std::shared_ptr<const CBlock> &pblock,
bool &fInvalidFound,
3306 bool fBlocksDisconnected =
false;
3309 if (!fBlocksDisconnected) {
3328 "Failed to disconnect block; see debug.log for details");
3332 fBlocksDisconnected =
true;
3336 std::vector<CBlockIndex *> vpindexToConnect;
3337 bool fContinue =
true;
3342 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3343 vpindexToConnect.clear();
3344 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3347 vpindexToConnect.push_back(pindexIter);
3348 pindexIter = pindexIter->
pprev;
3356 if (!
ConnectTip(state, blockPolicyState, pindexConnect,
3357 pindexConnect == pindexMostWork
3359 : std::shared_ptr<const CBlock>(),
3368 fInvalidFound =
true;
3389 if (!pindexOldTip ||
3401 if (fBlocksDisconnected || !disconnectpool.
isEmpty()) {
3407 "Updating mempool due to reorganization or "
3408 "rules upgrade/downgrade\n");
3416 if (fInvalidFound) {
3437 bool fNotify =
false;
3438 bool fInitialBlockDownload =
false;
3443 pindexHeader = chainman.m_best_header;
3445 if (pindexHeader != pindexHeaderOld) {
3447 fInitialBlockDownload = chainman.IsInitialBlockDownload();
3448 pindexHeaderOld = pindexHeader;
3454 chainman.GetNotifications().headerTip(
3470 std::shared_ptr<const CBlock> pblock,
3491 LogPrintf(
"m_disabled is set - this chainstate should not be in "
3492 "operation. Please report this as a bug. %s\n",
3500 bool exited_ibd{
false};
3510 std::vector<const CBlockIndex *> blocksToReconcile;
3511 bool blocks_connected =
false;
3513 const bool fAutoUnpark =
3528 if (pindexMostWork ==
nullptr) {
3534 if (pindexMostWork ==
nullptr ||
3539 bool fInvalidFound =
false;
3540 std::shared_ptr<const CBlock> nullBlockPtr;
3542 state, pindexMostWork,
3543 pblock && pblock->GetHash() ==
3551 blocks_connected =
true;
3553 if (fInvalidFound ||
3554 (pindexMostWork && pindexMostWork->nStatus.isParked())) {
3556 pindexMostWork =
nullptr;
3581 if (blocks_connected) {
3585 if (was_in_ibd && !still_in_ibd) {
3594 pindexFork != pindexNewTip) {
3610 return m_avalancheFinalizedBlockIndex);
3611 for (
const CBlockIndex *pindex : blocksToReconcile) {
3617 if (blocks_connected) {
3619 while (pindexTest && pindexTest != pfinalized) {
3624 avalanche->computeStakingReward(pindexTest);
3625 pindexTest = pindexTest->
pprev;
3631 if (!blocks_connected) {
3635 if (nStopAtHeight && pindexNewTip &&
3636 pindexNewTip->
nHeight >= nStopAtHeight) {
3662 }
while (pindexNewTip != pindexMostWork);
3693 std::numeric_limits<int32_t>::min()) {
3715template <
typename Func>
struct Defer {
3717 Defer(
Func &&f) : func(
std::move(f)) {}
3718 ~Defer() { func(); }
3731 bool pindex_was_in_chain =
false;
3732 int disconnected = 0;
3748 std::multimap<const arith_uint256, CBlockIndex *> candidate_blocks_by_work;
3752 for (
auto &entry :
m_blockman.m_block_index) {
3763 candidate_blocks_by_work.insert(
3764 std::make_pair(candidate->
nChainWork, candidate));
3776 constexpr int maxDisconnectPoolBlocks = 10;
3828 pindex_was_in_chain =
true;
3837 if (optDisconnectPool && disconnected > maxDisconnectPoolBlocks) {
3842 optDisconnectPool =
nullptr;
3858 invalid_walk_tip->nStatus =
3859 invalidate ? invalid_walk_tip->nStatus.withFailed()
3860 : invalid_walk_tip->nStatus.withParked();
3865 if (invalid_walk_tip == to_mark_failed_or_parked->
pprev &&
3866 (invalidate ? to_mark_failed_or_parked->nStatus.hasFailed()
3867 : to_mark_failed_or_parked->nStatus.isParked())) {
3871 to_mark_failed_or_parked->nStatus =
3873 ? to_mark_failed_or_parked->nStatus.withFailed(
false)
3875 : to_mark_failed_or_parked->nStatus.withParked(
false)
3876 .withParkedParent());
3882 auto candidate_it = candidate_blocks_by_work.lower_bound(
3884 while (candidate_it != candidate_blocks_by_work.end()) {
3886 invalid_walk_tip->
pprev)) {
3888 candidate_it = candidate_blocks_by_work.erase(candidate_it);
3897 to_mark_failed_or_parked = invalid_walk_tip;
3913 to_mark_failed_or_parked->nStatus =
3914 invalidate ? to_mark_failed_or_parked->nStatus.withFailed()
3915 : to_mark_failed_or_parked->nStatus.withParked();
3928 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
3930 block_index.HaveNumChainTxs() &&
3943 if (pindex_was_in_chain) {
3946 *to_mark_failed_or_parked->
pprev);
3970template <
typename F>
3974 if (pindex->nStatus != newStatus &&
3977 pindex->nStatus = newStatus;
3993template <
typename F,
typename C,
typename AC>
3995 F f, C fChild, AC fAncestorWasChanged) {
4001 for (
auto pindexAncestor = pindex; pindexAncestor !=
nullptr;
4002 pindexAncestor = pindexAncestor->
pprev) {
4004 pindexDeepestChanged = pindexAncestor;
4010 pindexDeepestChanged) {
4012 pindexReset =
nullptr;
4016 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
4019 fAncestorWasChanged);
4029 return status.withClearedFailureFlags();
4032 return status.withClearedFailureFlags();
4035 return status.withFailedParent(false);
4049 if (is_active_chainstate) {
4053 }
else if (!m_disabled) {
4059 if (snapshot_base->GetAncestor(pindex->
nHeight) == pindex) {
4071 return status.withClearedParkedFlags();
4074 return fClearChildren ? status.withClearedParkedFlags()
4075 : status.withParkedParent(false);
4078 return status.withParkedParent(false);
4098 "The block to mark finalized by avalanche is not on the "
4099 "active chain: %s\n",
4112 m_avalancheFinalizedBlockIndex = pindex;
4122 m_avalancheFinalizedBlockIndex =
nullptr;
4127 return pindex && m_avalancheFinalizedBlockIndex &&
4139 pindexNew->
nTx = block.
vtx.size();
4142 pindexNew->nFile = pos.
nFile;
4143 pindexNew->nDataPos = pos.
nPos;
4144 pindexNew->nUndoPos = 0;
4145 pindexNew->nStatus = pindexNew->nStatus.withData();
4152 std::deque<CBlockIndex *> queue;
4153 queue.push_back(pindexNew);
4157 while (!queue.empty()) {
4170 c->TryAddBlockIndexCandidate(pindex);
4173 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
4174 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
4176 while (range.first != range.second) {
4177 std::multimap<CBlockIndex *, CBlockIndex *>::iterator it =
4179 queue.push_back(it->second);
4184 }
else if (pindexNew->
pprev &&
4187 std::make_pair(pindexNew->
pprev, pindexNew));
4207 "high-hash",
"proof of work failed");
4233 "bad-txnmrklroot",
"hashMerkleRoot mismatch");
4241 "bad-txns-duplicate",
"duplicate transaction");
4250 if (block.
vtx.empty()) {
4252 "bad-cb-missing",
"first tx is not coinbase");
4261 "bad-blk-length",
"size limits failed");
4265 if (currentBlockSize > nMaxBlockSize) {
4267 "bad-blk-length",
"size limits failed");
4275 strprintf(
"Coinbase check failed (txid %s) %s",
4276 block.
vtx[0]->GetId().ToString(),
4282 for (
size_t i = 1; i < block.
vtx.size(); i++) {
4283 auto *tx = block.
vtx[i].get();
4288 strprintf(
"Transaction check failed (txid %s) %s",
4303 return std::all_of(headers.cbegin(), headers.cend(),
4304 [&](
const auto &header) {
4305 return CheckProofOfWork(
4306 header.GetHash(), header.nBits, consensusParams);
4333 const std::optional<CCheckpointData> &test_checkpoints = std::nullopt)
4336 assert(pindexPrev !=
nullptr);
4337 const int nHeight = pindexPrev->nHeight + 1;
4343 LogPrintf(
"bad bits after height: %d\n", pindexPrev->nHeight);
4345 "bad-diffbits",
"incorrect proof of work");
4349 if (chainman.m_options.checkpoints_enabled) {
4351 test_checkpoints ? test_checkpoints.value() : params.
Checkpoints();
4357 "ERROR: %s: rejected by checkpoint lock-in at %d\n",
4360 "checkpoint mismatch");
4368 blockman.GetLastCheckpoint(checkpoints);
4369 if (pcheckpoint && nHeight < pcheckpoint->
nHeight) {
4371 "ERROR: %s: forked chain older than last checkpoint "
4375 "bad-fork-prior-to-checkpoint");
4380 if (block.
GetBlockTime() <= pindexPrev->GetMedianTimePast()) {
4382 "time-too-old",
"block's timestamp is too early");
4386 if (block.
Time() > now + std::chrono::seconds{MAX_FUTURE_BLOCK_TIME}) {
4389 "block timestamp too far in the future");
4423 const int nBlockHeight = active_chain_tip.
nHeight + 1;
4448 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4451 bool enforce_locktime_median_time_past{
false};
4454 assert(pindexPrev !=
nullptr);
4455 enforce_locktime_median_time_past =
true;
4458 const int64_t nMedianTimePast =
4461 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past
4466 const bool fIsMagneticAnomalyEnabled =
4473 const CTransaction *prevTx =
nullptr;
4474 for (
const auto &ptx : block.
vtx) {
4475 const CTransaction &tx = *ptx;
4476 if (fIsMagneticAnomalyEnabled) {
4477 if (prevTx && (tx.GetId() <= prevTx->GetId())) {
4478 if (tx.GetId() == prevTx->GetId()) {
4482 tx.GetId().ToString()));
4487 strprintf(
"Transaction order is invalid (%s < %s)",
4488 tx.GetId().ToString(),
4489 prevTx->GetId().ToString()));
4492 if (prevTx || !tx.IsCoinBase()) {
4510 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4512 block.
vtx[0]->vin[0].scriptSig.begin())) {
4515 "block height mismatch in coinbase");
4530 const std::optional<CCheckpointData> &test_checkpoints) {
4537 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4539 if (miSelf !=
m_blockman.m_block_index.end()) {
4546 if (pindex->nStatus.isInvalid()) {
4559 "%s: Consensus::CheckBlockHeader: %s, %s\n", __func__,
4565 BlockMap::iterator mi{
4569 "header %s has prev block not found: %s\n",
4572 "prev-blk-not-found");
4577 if (pindexPrev->nStatus.isInvalid()) {
4579 "header %s has prev block invalid: %s\n", hash.
ToString(),
4589 "%s: Consensus::ContextualCheckBlockHeader: %s, %s\n",
4620 if (pindexPrev->
GetAncestor(failedit->nHeight) == failedit) {
4621 assert(failedit->nStatus.hasFailed());
4623 while (invalid_walk != failedit) {
4624 invalid_walk->nStatus =
4625 invalid_walk->nStatus.withFailedParent();
4627 invalid_walk = invalid_walk->
pprev;
4630 "header %s has prev block invalid: %s\n",
4639 if (!min_pow_checked) {
4641 "%s: not adding new block header %s, missing anti-dos "
4642 "proof-of-work validation\n",
4645 "too-little-chainwork");
4661 const auto msg =
strprintf(
"Saw new header hash=%s height=%d",
4675 const std::vector<CBlockHeader> &headers,
bool min_pow_checked,
4677 const std::optional<CCheckpointData> &test_checkpoints) {
4685 header, state, &pindex, min_pow_checked, test_checkpoints);
4701 const int64_t blocks_left{
4702 (
GetTime() - last_accepted.GetBlockTime()) /
4704 const double progress{100.0 * last_accepted.nHeight /
4705 (last_accepted.nHeight + blocks_left)};
4706 LogPrintf(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n",
4707 last_accepted.nHeight, progress);
4715 int64_t timestamp) {
4724 if (m_best_header->nChainWork >=
4730 auto now = Now<SteadyMilliseconds>();
4731 if (now < m_last_presync_update + 250ms) {
4734 m_last_presync_update = now;
4738 height, timestamp,
true);
4739 if (initial_download) {
4740 const int64_t blocks_left{(
GetTime() - timestamp) /
4742 const double progress{100.0 * height / (height + blocks_left)};
4743 LogPrintf(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n",
4751 bool *fNewBlock,
bool min_pow_checked) {
4754 const CBlock &block = *pblock;
4761 bool accepted_header{
4765 if (!accepted_header) {
4773 bool fAlreadyHave = pindex->nStatus.hasData();
4785 int64_t chainTipTimeDiff =
4791 LogPrintf(
"Chain tip timestamp-to-received-time difference: hash=%s, "
4794 LogPrintf(
"New block timestamp-to-received-time difference: hash=%s, "
4799 bool fHasMoreOrSameWork =
4807 bool fTooFarAhead{pindex->
nHeight >
4820 if (pindex->
nTx != 0) {
4825 if (!fHasMoreOrSameWork) {
4849 pindex->nStatus = pindex->nStatus.withFailed();
4853 return error(
"%s: %s (block %s)", __func__, state.
ToString(),
4867 std::optional<int> snapshot_base_height = GetSnapshotBaseHeight();
4868 const bool is_background_block =
4870 pindex->
nHeight <= snapshot_base_height;
4872 if (!is_background_block && pindexFork &&
4874 LogPrintf(
"Park block %s as it would cause a deep reorg.\n",
4876 pindex->nStatus = pindex->nStatus.withParked();
4895 if (blockPos.IsNull()) {
4897 "%s: Failed to find position to write new block to disk",
4902 }
catch (
const std::runtime_error &e) {
4903 return AbortNode(state, std::string(
"System error: ") + e.what());
4921 const std::shared_ptr<const CBlock> &block,
bool force_processing,
4922 bool min_pow_checked,
bool *new_block,
4951 ret =
AcceptBlock(block, state, force_processing,
nullptr,
4952 new_block, min_pow_checked);
4957 return error(
"%s: AcceptBlock FAILED (%s)", __func__,
4967 return error(
"%s: ActivateBestChain failed (%s)", __func__,
4972 ? m_ibd_chainstate.get()
4975 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
4976 return error(
"%s: [background] ActivateBestChain failed (%s)", __func__,
4994 false, test_accept);
5010 indexDummy.
pprev = pindexPrev;
5017 adjusted_time_callback())) {
5018 return error(
"%s: Consensus::ContextualCheckBlockHeader: %s", __func__,
5023 return error(
"%s: Consensus::CheckBlock: %s", __func__,
5029 return error(
"%s: Consensus::ContextualCheckBlock: %s", __func__,
5033 if (!chainstate.
ConnectBlock(block, state, &indexDummy, viewNew,
5034 validationOptions,
nullptr,
true)) {
5044 int nManualPruneHeight) {
5047 nManualPruneHeight)) {
5048 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
5054 FopenFn mockable_fopen_function) {
5084 "Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f\n",
5115 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
5119 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
5120 LogPrintf(
"Verifying last %i blocks at level %i\n", nCheckDepth,
5126 int nGoodTransactions = 0;
5129 bool skipped_no_block_data{
false};
5130 bool skipped_l3_checks{
false};
5131 LogPrintf(
"Verification progress: 0%%\n");
5136 pindex = pindex->
pprev) {
5137 const int percentageDone = std::max(
5138 1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() -
5140 (
double)nCheckDepth *
5141 (nCheckLevel >= 4 ? 50 : 100))));
5142 if (reportDone < percentageDone / 10) {
5144 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
5145 reportDone = percentageDone / 10;
5154 !pindex->nStatus.hasData()) {
5157 LogPrintf(
"VerifyDB(): block verification stopping at height %d "
5158 "(no data). This could be due to pruning or use of an "
5159 "assumeutxo snapshot.\n",
5161 skipped_no_block_data =
true;
5170 "Verification error: ReadBlockFromDisk failed at %d, hash=%s\n",
5176 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensusParams,
5179 "Verification error: found bad block at %d, hash=%s (%s)\n",
5186 if (nCheckLevel >= 2 && pindex) {
5190 LogPrintf(
"Verification error: found bad undo data at %d, "
5203 if (nCheckLevel >= 3) {
5207 chainstate.DisconnectBlock(block, pindex, coins);
5209 LogPrintf(
"Verification error: irrecoverable inconsistency "
5210 "in block data at %d, hash=%s\n",
5216 nGoodTransactions = 0;
5217 pindexFailure = pindex;
5219 nGoodTransactions += block.
vtx.size();
5222 skipped_l3_checks =
true;
5231 if (pindexFailure) {
5232 LogPrintf(
"Verification error: coin database inconsistencies found "
5233 "(last %i blocks, %i good transactions before that)\n",
5238 if (skipped_l3_checks) {
5239 LogPrintf(
"Skipped verification of level >=3 (insufficient database "
5240 "cache size). Consider increasing -dbcache.\n");
5247 if (nCheckLevel >= 4 && !skipped_l3_checks) {
5249 const int percentageDone = std::max(
5250 1, std::min(99, 100 -
int(
double(chainstate.
m_chain.
Height() -
5252 double(nCheckDepth) * 50)));
5253 if (reportDone < percentageDone / 10) {
5255 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
5256 reportDone = percentageDone / 10;
5263 LogPrintf(
"Verification error: ReadBlockFromDisk failed at %d, "
5268 if (!chainstate.
ConnectBlock(block, state, pindex, coins,
5270 LogPrintf(
"Verification error: found unconnectable block at "
5271 "%d, hash=%s (%s)\n",
5282 LogPrintf(
"Verification: No coin database inconsistencies in last %i "
5283 "blocks (%i transactions)\n",
5284 block_count, nGoodTransactions);
5286 if (skipped_l3_checks) {
5289 if (skipped_no_block_data) {
5305 return error(
"ReplayBlock(): ReadBlockFromDisk failed at %d, hash=%s",
5315 if (tx->IsCoinBase()) {
5319 for (
const CTxIn &txin : tx->vin) {
5333 std::vector<BlockHash> hashHeads =
db.GetHeadBlocks();
5334 if (hashHeads.empty()) {
5338 if (hashHeads.size() != 2) {
5339 return error(
"ReplayBlocks(): unknown inconsistent state");
5352 if (
m_blockman.m_block_index.count(hashHeads[0]) == 0) {
5354 "ReplayBlocks(): reorganization to unknown block requested");
5357 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
5359 if (!hashHeads[1].IsNull()) {
5361 if (
m_blockman.m_block_index.count(hashHeads[1]) == 0) {
5363 "ReplayBlocks(): reorganization from unknown block requested");
5366 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
5368 assert(pindexFork !=
nullptr);
5372 while (pindexOld != pindexFork) {
5377 return error(
"RollbackBlock(): ReadBlockFromDisk() failed at "
5388 "RollbackBlock(): DisconnectBlock failed at %d, hash=%s",
5400 pindexOld = pindexOld->
pprev;
5404 int nForkHeight = pindexFork ? pindexFork->
nHeight : 0;
5411 _(
"Replaying blocks…"),
5412 (
int)((
nHeight - nForkHeight) * 100.0 /
5413 (pindexNew->
nHeight - nForkHeight)),
5428void Chainstate::ClearBlockIndexCandidates() {
5444 const uint64_t numHeaders{20};
5447 const fs::path filePathTmp = filePath +
".new";
5458 bool missingIndex{
false};
5459 for (uint64_t i = 0; i < numHeaders; i++) {
5461 LogPrintf(
"Missing block index, stopping the headers time "
5462 "dumping after %d blocks.\n",
5464 missingIndex =
true;
5471 index = index->
pprev;
5475 throw std::runtime_error(
strprintf(
"Failed to commit to file %s",
5481 fs::remove(filePathTmp);
5486 throw std::runtime_error(
strprintf(
"Rename failed from %s to %s",
5490 }
catch (
const std::exception &e) {
5491 LogPrintf(
"Failed to dump the headers time: %s.\n", e.what());
5495 LogPrintf(
"Successfully dumped the last %d headers time to %s.\n",
5511 LogPrintf(
"Failed to open header times from disk, skipping.\n");
5520 LogPrintf(
"Unsupported header times file version, skipping.\n");
5527 for (uint64_t i = 0; i < numBlocks; i++) {
5529 int64_t receiveTime;
5532 file >> receiveTime;
5536 LogPrintf(
"Missing index for block %s, stopping the headers "
5537 "time loading after %d blocks.\n",
5544 }
catch (
const std::exception &e) {
5545 LogPrintf(
"Failed to read the headers time file data on disk: %s.\n",
5563 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
5565 std::vector<CBlockIndex *> vSortedByHeight{
5567 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
5579 if (pindex == GetSnapshotBaseBlock() ||
5587 if (pindex->nStatus.isInvalid() &&
5589 pindex->
nChainWork > m_best_invalid->nChainWork)) {
5590 m_best_invalid = pindex;
5593 if (pindex->nStatus.isOnParkedChain() &&
5595 pindex->
nChainWork > m_best_parked->nChainWork)) {
5596 m_best_parked = pindex;
5600 (m_best_header ==
nullptr ||
5602 m_best_header = pindex;
5606 needs_init =
m_blockman.m_block_index.empty();
5616 LogPrintf(
"Initializing databases...\n");
5637 if (blockPos.IsNull()) {
5638 return error(
"%s: writing genesis block to disk failed", __func__);
5643 }
catch (
const std::runtime_error &e) {
5644 return error(
"%s: failed to write genesis block: %s", __func__,
5653 std::multimap<BlockHash, FlatFilePos> *blocks_with_unknown_parent,
5656 assert(!dbp == !blocks_with_unknown_parent);
5670 uint64_t nRewind = blkdat.
GetPos();
5671 while (!blkdat.
eof()) {
5681 unsigned int nSize = 0;
5686 nRewind = blkdat.
GetPos() + 1;
5688 if (memcmp(buf, params.
DiskMagic().data(),
5698 }
catch (
const std::exception &) {
5706 const uint64_t nBlockPos{blkdat.
GetPos()};
5708 dbp->
nPos = nBlockPos;
5710 blkdat.
SetLimit(nBlockPos + nSize);
5718 nRewind = nBlockPos + nSize;
5723 std::shared_ptr<CBlock> pblock{};
5732 "%s: Out of order block %s, parent %s not known\n",
5733 __func__, hash.ToString(),
5735 if (dbp && blocks_with_unknown_parent) {
5736 blocks_with_unknown_parent->emplace(
5745 if (!pindex || !pindex->nStatus.hasData()) {
5748 blkdat.
SetPos(nBlockPos);
5749 pblock = std::make_shared<CBlock>();
5751 nRewind = blkdat.
GetPos();
5754 if (
AcceptBlock(pblock, state,
true, dbp,
nullptr,
5762 pindex->
nHeight % 1000 == 0) {
5765 "Block Import: already had block %s at height %d\n",
5766 hash.ToString(), pindex->
nHeight);
5773 bool genesis_activation_failure =
false;
5774 for (
auto c :
GetAll()) {
5776 if (!c->ActivateBestChain(state,
nullptr,
avalanche)) {
5777 genesis_activation_failure =
true;
5781 if (genesis_activation_failure) {
5795 bool activation_failure =
false;
5796 for (
auto c :
GetAll()) {
5798 if (!c->ActivateBestChain(state, pblock,
avalanche)) {
5800 "failed to activate chain (%s)\n",
5802 activation_failure =
true;
5806 if (activation_failure) {
5813 if (!blocks_with_unknown_parent) {
5819 std::deque<BlockHash> queue;
5820 queue.push_back(hash);
5821 while (!queue.empty()) {
5824 auto range = blocks_with_unknown_parent->equal_range(head);
5825 while (range.first != range.second) {
5826 std::multimap<BlockHash, FlatFilePos>::iterator it =
5828 std::shared_ptr<CBlock> pblockrecursive =
5829 std::make_shared<CBlock>();
5834 "%s: Processing out of order child %s of %s\n",
5835 __func__, pblockrecursive->GetHash().ToString(),
5840 &it->second,
nullptr,
true)) {
5842 queue.push_back(pblockrecursive->GetHash());
5846 blocks_with_unknown_parent->erase(it);
5850 }
catch (
const std::exception &e) {
5871 "%s: unexpected data at file offset 0x%x - %s. "
5873 __func__, (nRewind - 1), e.what());
5876 }
catch (
const std::runtime_error &e) {
5877 AbortNode(std::string(
"System error: ") + e.what());
5880 LogPrintf(
"Loaded %i blocks from external file in %dms\n", nLoaded,
5901 std::multimap<CBlockIndex *, CBlockIndex *> forward;
5902 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
5903 forward.emplace(block_index.pprev, &block_index);
5908 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
5909 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
5910 rangeGenesis = forward.equal_range(
nullptr);
5912 rangeGenesis.first++;
5914 assert(rangeGenesis.first == rangeGenesis.second);
5936 CBlockIndex *pindexFirstNotTransactionsValid =
nullptr;
5942 CBlockIndex *pindexFirstNotScriptsValid =
nullptr;
5948 const CBlockIndex *snap_base{GetSnapshotBaseBlock()};
5949 CBlockIndex *snap_first_missing{}, *snap_first_notx{}, *snap_first_notv{},
5950 *snap_first_nocv{}, *snap_first_nosv{};
5951 auto snap_update_firsts = [&] {
5952 if (pindex == snap_base) {
5953 std::swap(snap_first_missing, pindexFirstMissing);
5954 std::swap(snap_first_notx, pindexFirstNeverProcessed);
5955 std::swap(snap_first_notv, pindexFirstNotTransactionsValid);
5956 std::swap(snap_first_nocv, pindexFirstNotChainValid);
5957 std::swap(snap_first_nosv, pindexFirstNotScriptsValid);
5961 while (pindex !=
nullptr) {
5963 if (pindexFirstInvalid ==
nullptr && pindex->nStatus.hasFailed()) {
5964 pindexFirstInvalid = pindex;
5966 if (pindexFirstParked ==
nullptr && pindex->nStatus.isParked()) {
5967 pindexFirstParked = pindex;
5969 if (pindexFirstMissing ==
nullptr && !pindex->nStatus.hasData()) {
5970 pindexFirstMissing = pindex;
5972 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) {
5973 pindexFirstNeverProcessed = pindex;
5975 if (pindex->
pprev !=
nullptr && pindexFirstNotTreeValid ==
nullptr &&
5977 pindexFirstNotTreeValid = pindex;
5979 if (pindex->
pprev !=
nullptr) {
5980 if (pindexFirstNotTransactionsValid ==
nullptr &&
5982 pindexFirstNotTransactionsValid = pindex;
5984 if (pindexFirstNotChainValid ==
nullptr &&
5986 pindexFirstNotChainValid = pindex;
5988 if (pindexFirstNotScriptsValid ==
nullptr &&
5990 pindexFirstNotScriptsValid = pindex;
5995 if (pindex->
pprev ==
nullptr) {
5999 for (
auto c :
GetAll()) {
6000 if (c->m_chain.Genesis() !=
nullptr) {
6002 assert(pindex == c->m_chain.Genesis());
6017 assert(pindex->nStatus.hasData() == (pindex->
nTx > 0));
6018 assert(pindexFirstMissing == pindexFirstNeverProcessed);
6019 }
else if (pindex->nStatus.hasData()) {
6024 if (pindex->nStatus.hasUndo()) {
6025 assert(pindex->nStatus.hasData());
6027 if (snap_base && snap_base->GetAncestor(pindex->
nHeight) == pindex) {
6039 assert((pindexFirstNeverProcessed ==
nullptr || pindex == snap_base) ==
6041 assert((pindexFirstNotTransactionsValid ==
nullptr ||
6053 assert(pindexFirstNotTreeValid ==
nullptr);
6056 assert(pindexFirstNotTreeValid ==
nullptr);
6060 assert(pindexFirstNotChainValid ==
nullptr);
6064 assert(pindexFirstNotScriptsValid ==
nullptr);
6066 if (pindexFirstInvalid ==
nullptr) {
6069 assert(!pindex->nStatus.isInvalid());
6071 if (pindexFirstParked ==
nullptr) {
6075 assert(!pindex->nStatus.isOnParkedChain());
6078 if (!pindex->
pprev) {
6092 for (
auto c :
GetAll()) {
6093 if (c->m_chain.Tip() ==
nullptr) {
6110 (pindexFirstNeverProcessed ==
nullptr || pindex == snap_base)) {
6114 if (pindexFirstInvalid ==
nullptr) {
6121 GetSnapshotBaseBlock()->GetAncestor(pindex->
nHeight) ==
6135 if (pindexFirstMissing ==
nullptr) {
6136 assert(pindex->nStatus.isOnParkedChain() ||
6137 c->setBlockIndexCandidates.count(pindex));
6145 if (pindex == c->m_chain.Tip() ||
6146 pindex == c->SnapshotBase()) {
6147 assert(c->setBlockIndexCandidates.count(pindex));
6159 assert(c->setBlockIndexCandidates.count(pindex) == 0);
6163 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6164 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6167 bool foundInUnlinked =
false;
6168 while (rangeUnlinked.first != rangeUnlinked.second) {
6169 assert(rangeUnlinked.first->first == pindex->
pprev);
6170 if (rangeUnlinked.first->second == pindex) {
6171 foundInUnlinked =
true;
6174 rangeUnlinked.first++;
6176 if (pindex->
pprev && pindex->nStatus.hasData() &&
6177 pindexFirstNeverProcessed !=
nullptr &&
6178 pindexFirstInvalid ==
nullptr) {
6184 if (!pindex->nStatus.hasData()) {
6186 assert(!foundInUnlinked);
6188 if (pindexFirstMissing ==
nullptr) {
6191 assert(!foundInUnlinked);
6193 if (pindex->
pprev && pindex->nStatus.hasData() &&
6194 pindexFirstNeverProcessed ==
nullptr &&
6195 pindexFirstMissing !=
nullptr) {
6208 for (
auto c :
GetAll()) {
6211 c->setBlockIndexCandidates.count(pindex) == 0) {
6212 if (pindexFirstInvalid ==
nullptr) {
6214 snap_base->GetAncestor(pindex->
nHeight) == pindex) {
6226 snap_update_firsts();
6227 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6228 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6229 range = forward.equal_range(pindex);
6230 if (range.first != range.second) {
6232 pindex = range.first->second;
6240 snap_update_firsts();
6243 if (pindex == pindexFirstInvalid) {
6244 pindexFirstInvalid =
nullptr;
6246 if (pindex == pindexFirstParked) {
6247 pindexFirstParked =
nullptr;
6249 if (pindex == pindexFirstMissing) {
6250 pindexFirstMissing =
nullptr;
6252 if (pindex == pindexFirstNeverProcessed) {
6253 pindexFirstNeverProcessed =
nullptr;
6255 if (pindex == pindexFirstNotTreeValid) {
6256 pindexFirstNotTreeValid =
nullptr;
6258 if (pindex == pindexFirstNotTransactionsValid) {
6259 pindexFirstNotTransactionsValid =
nullptr;
6261 if (pindex == pindexFirstNotChainValid) {
6262 pindexFirstNotChainValid =
nullptr;
6264 if (pindex == pindexFirstNotScriptsValid) {
6265 pindexFirstNotScriptsValid =
nullptr;
6270 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6271 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6272 rangePar = forward.equal_range(pindexPar);
6273 while (rangePar.first->second != pindex) {
6276 assert(rangePar.first != rangePar.second);
6281 if (rangePar.first != rangePar.second) {
6283 pindex = rangePar.first->second;
6295 assert(nNodes == forward.size());
6301 return strprintf(
"Chainstate [%s] @ height %d (%s)",
6307bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size) {
6320 coinsdb_size * (1.0 / 1024 / 1024));
6322 coinstip_size * (1.0 / 1024 / 1024));
6327 if (coinstip_size > old_coinstip_size) {
6344 if (pindex ==
nullptr) {
6349 LogPrintf(
"Internal bug detected: block %d has unset nChainTx (%s %s). "
6350 "Please report this issue here: %s\n",
6356 int64_t nNow = time(
nullptr);
6371 if (m_active_chainstate && m_active_chainstate->m_from_snapshot_blockhash) {
6373 return m_active_chainstate->m_from_snapshot_blockhash;
6375 return std::nullopt;
6380 std::vector<Chainstate *> out;
6383 {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
6385 out.push_back(pchainstate);
6394 assert(!m_ibd_chainstate);
6395 assert(!m_active_chainstate);
6397 m_ibd_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman, *
this);
6398 m_active_chainstate = m_ibd_chainstate.get();
6399 return *m_active_chainstate;
6412 const bool existed{fs::remove(base_blockhash_path)};
6414 LogPrintf(
"[snapshot] snapshot chainstate dir being removed "
6418 }
catch (
const fs::filesystem_error &e) {
6419 LogPrintf(
"[snapshot] failed to remove file %s: %s\n",
6426 LogPrintf(
"Removing leveldb dir at %s\n", path_str);
6430 const bool destroyed = dbwrapper::DestroyDB(path_str, {}).ok();
6433 LogPrintf(
"error: leveldb DestroyDB call failed on %s\n", path_str);
6451 LogPrintf(
"[snapshot] can't activate a snapshot-based chainstate more "
6456 int64_t current_coinsdb_cache_size{0};
6457 int64_t current_coinstip_cache_size{0};
6466 static constexpr double IBD_CACHE_PERC = 0.01;
6467 static constexpr double SNAPSHOT_CACHE_PERC = 0.99;
6480 current_coinsdb_cache_size =
6482 current_coinstip_cache_size =
6488 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
6489 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
6492 auto snapshot_chainstate =
6499 snapshot_chainstate->InitCoinsDB(
6500 static_cast<size_t>(current_coinsdb_cache_size *
6501 SNAPSHOT_CACHE_PERC),
6502 in_memory,
false,
"chainstate");
6503 snapshot_chainstate->InitCoinsCache(
static_cast<size_t>(
6504 current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
6508 coins_file, metadata);
6515 snapshot_ok =
false;
6520 this->MaybeRebalanceCaches();
6529 snapshot_chainstate.reset();
6534 strprintf(
"Failed to remove snapshot chainstate dir (%s). "
6535 "Manually remove it before restarting.\n",
6544 assert(!m_snapshot_chainstate);
6545 m_snapshot_chainstate.swap(snapshot_chainstate);
6546 const bool chaintip_loaded = m_snapshot_chainstate->LoadChainTip();
6549 m_active_chainstate = m_snapshot_chainstate.get();
6551 LogPrintf(
"[snapshot] successfully activated snapshot %s\n",
6554 m_snapshot_chainstate->CoinsTip().DynamicMemoryUsage() /
6557 this->MaybeRebalanceCaches();
6563 bool snapshot_loaded) {
6566 snapshot_loaded ?
"saving snapshot chainstate"
6567 :
"flushing coins cache",
6569 BCLog::LogFlags::ALL);
6571 coins_cache.
Flush();
6575 const char *
what()
const throw()
override {
6576 return "ComputeUTXOStats interrupted by shutdown.";
6600 if (!snapshot_start_block) {
6603 LogPrintf(
"[snapshot] Did not find snapshot start blockheader %s\n",
6608 int base_height = snapshot_start_block->
nHeight;
6611 if (!maybe_au_data) {
6612 LogPrintf(
"[snapshot] assumeutxo height in snapshot metadata not "
6613 "recognized (%d) - refusing to load snapshot\n",
6625 LogPrintf(
"[snapshot] loading coins from snapshot %s\n",
6627 int64_t coins_processed{0};
6629 while (coins_left > 0) {
6631 coins_file >> outpoint;
6633 }
catch (
const std::ios_base::failure &) {
6634 LogPrintf(
"[snapshot] bad snapshot format or truncated snapshot "
6635 "after deserializing %d coins\n",
6636 coins_count - coins_left);
6639 if (coin.
GetHeight() > uint32_t(base_height) ||
6642 std::numeric_limits<
decltype(outpoint.GetN())>::max()) {
6644 "[snapshot] bad snapshot data after deserializing %d coins\n",
6645 coins_count - coins_left);
6654 if (coins_processed % 1000000 == 0) {
6655 LogPrintf(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)\n",
6657 static_cast<float>(coins_processed) * 100 /
6658 static_cast<float>(coins_count),
6666 if (coins_processed % 120000 == 0) {
6671 const auto snapshot_cache_state =
WITH_LOCK(
6672 ::cs_main,
return snapshot_chainstate.GetCoinsCacheSizeState());
6695 bool out_of_coins{
false};
6697 coins_file >> outpoint;
6698 }
catch (
const std::ios_base::failure &) {
6700 out_of_coins =
true;
6702 if (!out_of_coins) {
6703 LogPrintf(
"[snapshot] bad snapshot - coins left over after "
6704 "deserializing %d coins\n",
6709 LogPrintf(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s\n",
6723 std::optional<CCoinsStats> maybe_stats;
6732 if (!maybe_stats.has_value()) {
6733 LogPrintf(
"[snapshot] failed to generate coins stats\n");
6741 LogPrintf(
"[snapshot] bad snapshot content hash: expected %s, got %s\n",
6743 maybe_stats->hashSerialized.ToString());
6759 constexpr int AFTER_GENESIS_START{1};
6761 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height();
6763 index = snapshot_chainstate.
m_chain[i];
6774 assert(index == snapshot_start_block);
6778 LogPrintf(
"[snapshot] validated snapshot (%.2f MB)\n",
6801 if (m_ibd_chainstate.get() == &this->ActiveChainstate() ||
6802 !this->IsUsable(m_snapshot_chainstate.get()) ||
6803 !this->IsUsable(m_ibd_chainstate.get()) ||
6804 !m_ibd_chainstate->m_chain.Tip()) {
6810 const int snapshot_base_height = *
Assert(this->GetSnapshotBaseHeight());
6813 if (index_new.
nHeight < snapshot_base_height) {
6823 _(
"%s failed to validate the -assumeutxo snapshot state. "
6824 "This indicates a hardware problem, or a bug in the software, or "
6825 "a bad software modification that allowed an invalid snapshot to "
6826 "be loaded. As a result of this, the node will shut down and "
6827 "stop using any state that was built on the snapshot, resetting "
6828 "the chain height from %d to %d. On the next restart, the node "
6829 "will resume syncing from %d without using any snapshot data. "
6830 "Please report this incident to %s, including how you obtained "
6831 "the snapshot. The invalid snapshot chainstate will be left on "
6832 "disk in case it is helpful in diagnosing the issue that caused "
6834 PACKAGE_NAME, snapshot_tip_height, snapshot_base_height,
6835 snapshot_base_height, PACKAGE_BUGREPORT);
6838 LogPrintf(
"[snapshot] deleting snapshot, reverting to validated chain, "
6839 "and stopping node\n");
6841 m_active_chainstate = m_ibd_chainstate.get();
6842 m_snapshot_chainstate->m_disabled =
true;
6846 auto rename_result = m_snapshot_chainstate->InvalidateCoinsDBOnDisk();
6847 if (!rename_result) {
6852 shutdown_fnc(user_error);
6857 "[snapshot] supposed base block %s does not match the "
6858 "snapshot base block %s (height %d). Snapshot is not valid.\n",
6860 snapshot_base_height);
6861 handle_invalid_snapshot();
6867 int curr_height = m_ibd_chainstate->m_chain.Height();
6869 assert(snapshot_base_height == curr_height);
6874 CCoinsViewDB &ibd_coins_db = m_ibd_chainstate->CoinsDB();
6875 m_ibd_chainstate->ForceFlushStateToDisk();
6877 const auto &maybe_au_data =
6879 if (!maybe_au_data) {
6880 LogPrintf(
"[snapshot] assumeutxo data not found for height "
6881 "(%d) - refusing to validate snapshot\n",
6883 handle_invalid_snapshot();
6888 std::optional<CCoinsStats> maybe_ibd_stats;
6890 "[snapshot] computing UTXO stats for background chainstate to validate "
6891 "snapshot - this could take a few minutes\n");
6900 if (!maybe_ibd_stats) {
6902 "[snapshot] failed to generate stats for validation coins db\n");
6906 handle_invalid_snapshot();
6909 const auto &ibd_stats = *maybe_ibd_stats;
6918 LogPrintf(
"[snapshot] hash mismatch: actual=%s, expected=%s\n",
6919 ibd_stats.hashSerialized.ToString(),
6921 handle_invalid_snapshot();
6925 LogPrintf(
"[snapshot] snapshot beginning at %s has been fully validated\n",
6928 m_ibd_chainstate->m_disabled =
true;
6929 this->MaybeRebalanceCaches();
6936 assert(m_active_chainstate);
6937 return *m_active_chainstate;
6942 return m_snapshot_chainstate &&
6943 m_active_chainstate == m_snapshot_chainstate.get();
6945void ChainstateManager::MaybeRebalanceCaches() {
6947 bool ibd_usable = this->
IsUsable(m_ibd_chainstate.get());
6948 bool snapshot_usable = this->
IsUsable(m_snapshot_chainstate.get());
6949 assert(ibd_usable || snapshot_usable);
6951 if (ibd_usable && !snapshot_usable) {
6952 LogPrintf(
"[snapshot] allocating all cache to the IBD chainstate\n");
6956 }
else if (snapshot_usable && !ibd_usable) {
6960 "[snapshot] allocating all cache to the snapshot chainstate\n");
6964 }
else if (ibd_usable && snapshot_usable) {
6973 m_snapshot_chainstate->ResizeCoinsCaches(
6976 m_snapshot_chainstate->ResizeCoinsCaches(
6984void ChainstateManager::ResetChainstates() {
6985 m_ibd_chainstate.reset();
6986 m_snapshot_chainstate.reset();
6987 m_active_chainstate =
nullptr;
6996 if (!opts.check_block_index.has_value()) {
6998 opts.config.GetChainParams().DefaultConsistencyChecks();
7001 if (!opts.minimum_chain_work.has_value()) {
7003 opts.config.GetChainParams().GetConsensus().nMinimumChainWork);
7005 if (!opts.assumed_valid_block.has_value()) {
7006 opts.assumed_valid_block =
7007 opts.config.GetChainParams().GetConsensus().defaultAssumeValid;
7009 Assert(opts.adjusted_time_callback);
7010 return std::move(opts);
7016 m_blockman{
std::move(blockman_options)} {}
7018bool ChainstateManager::DetectSnapshotChainstate(
CTxMemPool *mempool) {
7019 assert(!m_snapshot_chainstate);
7024 std::optional<BlockHash> base_blockhash =
7026 if (!base_blockhash) {
7029 LogPrintf(
"[snapshot] detected active snapshot chainstate (%s) - loading\n",
7032 this->ActivateExistingSnapshot(mempool, *base_blockhash);
7037ChainstateManager::ActivateExistingSnapshot(
CTxMemPool *mempool,
7039 assert(!m_snapshot_chainstate);
7040 m_snapshot_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman,
7041 *
this, base_blockhash);
7042 LogPrintf(
"[snapshot] switching active chainstate to %s\n",
7043 m_snapshot_chainstate->ToString());
7044 m_active_chainstate = m_snapshot_chainstate.get();
7045 return *m_snapshot_chainstate;
7052 assert(
cs.m_from_snapshot_blockhash);
7053 auto storage_path_maybe =
cs.CoinsDB().StoragePath();
7055 assert(storage_path_maybe);
7056 return *storage_path_maybe;
7065 auto invalid_path = snapshot_datadir +
"_INVALID";
7068 LogPrintf(
"[snapshot] renaming snapshot datadir %s to %s\n", dbpath,
7075 fs::rename(snapshot_datadir, invalid_path);
7076 }
catch (
const fs::filesystem_error &e) {
7080 LogPrintf(
"%s: error renaming file '%s' -> '%s': %s\n", __func__,
7081 src_str, dest_str, e.what());
7083 "You should resolve this by manually "
7084 "moving or deleting the invalid "
7085 "snapshot directory %s, otherwise you "
7086 "will encounter the same error again "
7087 "on the next startup."),
7088 src_str, dest_str, src_str)};
7093bool ChainstateManager::DeleteSnapshotChainstate() {
7095 Assert(m_snapshot_chainstate);
7096 Assert(m_ibd_chainstate);
7100 LogPrintf(
"Deletion of %s failed. Please remove it manually to "
7101 "continue reindexing.\n",
7105 m_active_chainstate = m_ibd_chainstate.get();
7106 m_snapshot_chainstate.reset();
7118const CBlockIndex *ChainstateManager::GetSnapshotBaseBlock()
const {
7119 return m_active_chainstate ? m_active_chainstate->SnapshotBase() :
nullptr;
7122std::optional<int> ChainstateManager::GetSnapshotBaseHeight()
const {
7123 const CBlockIndex *base = this->GetSnapshotBaseBlock();
7124 return base ? std::make_optional(base->
nHeight) :
std::nullopt;
7127bool ChainstateManager::ValidatedSnapshotCleanup() {
7136 std::optional<fs::path> ibd_chainstate_path_maybe =
7137 get_storage_path(m_ibd_chainstate);
7138 std::optional<fs::path> snapshot_chainstate_path_maybe =
7139 get_storage_path(m_snapshot_chainstate);
7148 if (!ibd_chainstate_path_maybe || !snapshot_chainstate_path_maybe) {
7149 LogPrintf(
"[snapshot] snapshot chainstate cleanup cannot happen with "
7150 "in-memory chainstates. You are testing, right?\n");
7154 const auto &snapshot_chainstate_path = *snapshot_chainstate_path_maybe;
7155 const auto &ibd_chainstate_path = *ibd_chainstate_path_maybe;
7163 this->ResetChainstates();
7168 LogPrintf(
"[snapshot] deleting background chainstate directory (now "
7169 "unnecessary) (%s)\n",
7172 fs::path tmp_old{ibd_chainstate_path +
"_todelete"};
7175 const fs::filesystem_error &err) {
7179 "Rename of '%s' -> '%s' failed. "
7180 "Cannot clean up the background chainstate leveldb directory.",
7185 fs::rename(ibd_chainstate_path, tmp_old);
7186 }
catch (
const fs::filesystem_error &e) {
7187 rename_failed_abort(ibd_chainstate_path, tmp_old, e);
7191 LogPrintf(
"[snapshot] moving snapshot chainstate (%s) to "
7192 "default chainstate directory (%s)\n",
7197 fs::rename(snapshot_chainstate_path, ibd_chainstate_path);
7198 }
catch (
const fs::filesystem_error &e) {
7199 rename_failed_abort(snapshot_chainstate_path, ibd_chainstate_path, e);
7206 LogPrintf(
"Deletion of %s failed. Please remove it manually, as the "
7207 "directory is now unnecessary.\n",
7210 LogPrintf(
"[snapshot] deleted background chainstate directory (%s)\n",
bool IsDAAEnabled(const Consensus::Params ¶ms, int nHeight)
bool IsUAHFenabled(const Consensus::Params ¶ms, int nHeight)
static bool IsPhononEnabled(const Consensus::Params ¶ms, int32_t nHeight)
static bool IsGravitonEnabled(const Consensus::Params ¶ms, int32_t nHeight)
bool IsMagneticAnomalyEnabled(const Consensus::Params ¶ms, int32_t nHeight)
Check if Nov 15, 2018 HF has activated using block height.
bool MoneyRange(const Amount nValue)
static constexpr Amount SATOSHI
static constexpr Amount COIN
arith_uint256 UintToArith256(const uint256 &a)
@ CHAIN
Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends,...
@ TRANSACTIONS
Only first tx is coinbase, 2 <= coinbase input script length <= 100, transactions valid,...
@ SCRIPTS
Scripts & signatures ok.
@ TREE
All parent headers found, difficulty matches, timestamp >= median previous, checkpoint.
arith_uint256 GetBlockProof(const CBlockIndex &block)
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.
bool AreOnTheSameFork(const CBlockIndex *pa, const CBlockIndex *pb)
Check if two block index are on the same fork.
#define Assert(val)
Identity function.
#define Assume(val)
Assume is the identity function.
fs::path GetDataDirNet() const
Get data directory path with appended network identifier.
int64_t GetIntArg(const std::string &strArg, int64_t nDefault) const
Return integer argument or default value.
fs::path GetBlocksDirPath() const
Get blocks directory path.
bool GetBoolArg(const std::string &strArg, bool fDefault) const
Return boolean argument or default value.
Non-refcounted RAII wrapper for FILE*.
bool IsNull() const
Return true if the wrapped FILE* is nullptr, false otherwise.
std::string ToString() const
uint64_t getExcessiveBlockSize() const
BlockValidationOptions withCheckPoW(bool _checkPoW=true) const
BlockValidationOptions withCheckMerkleRoot(bool _checkMerkleRoot=true) const
BlockValidationOptions(const Config &config)
bool shouldValidatePoW() const
bool shouldValidateMerkleRoot() const
std::vector< CTransactionRef > vtx
The block chain is a tree shaped structure starting with the genesis block at the root,...
bool IsValid(enum BlockValidity nUpTo=BlockValidity::TRANSACTIONS) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
std::string ToString() const
CBlockIndex * pprev
pointer to the index of the predecessor of this block
int64_t GetHeaderReceivedTime() const
void MaybeResetChainStats(bool is_snapshot_base_block)
Reset chain tx stats and log a warning if the block is not the snapshot block, and the nChainTx value...
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
const BlockHash * phashBlock
pointer to the hash of the block, if any.
int64_t GetChainTxCount() const
Get the number of transaction in the chain so far.
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.
int64_t GetReceivedTimeDiff() const
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
FlatFilePos GetUndoPos() const EXCLUSIVE_LOCKS_REQUIRED(
bool UpdateChainStats()
Update chain tx stats and return True if this block is the genesis block or all parents have their tx...
CBlockIndex * pskip
pointer to the index of some further predecessor of this block
unsigned int nTx
Number of transactions in this block.
bool RaiseValidity(enum BlockValidity nUpTo) EXCLUSIVE_LOCKS_REQUIRED(
Raise the validity level of this block index entry.
int32_t nVersion
block header
int64_t nTimeReceived
(memory only) block header metadata
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
BlockHash GetBlockHash() const
unsigned int nSize
Size of this block.
int nHeight
height of the entry in the chain. The genesis block has height 0
unsigned int nChainTx
(memory only) Number of transactions in the chain up to and including this block.
Undo information for a CBlock.
std::vector< CTxUndo > vtxundo
Non-refcounted RAII wrapper around a FILE* that implements a ring buffer to deserialize from.
bool SetLimit(uint64_t nPos=std::numeric_limits< uint64_t >::max())
Prevent reading beyond a certain position.
uint64_t GetPos() const
return the current reading position
void FindByte(std::byte byte)
search for a given byte in the stream, and remain positioned on it
void SkipTo(const uint64_t file_pos)
Move the read position ahead in the stream to the given position.
bool SetPos(uint64_t nPos)
rewind to a given reading position
bool eof() const
check whether we're at the end of the source file
An in-memory indexed chain of blocks.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
void SetTip(CBlockIndex &block)
Set/initialize a chain with a given tip.
CBlockIndex * Genesis() const
Returns the index entry for the genesis block of this chain, or nullptr if none.
CBlockIndex * Next(const CBlockIndex *pindex) const
Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip...
int Height() const
Return the maximal height in the chain.
const CBlockIndex * FindFork(const CBlockIndex *pindex) const
Find the last common block between this chain and a block index entry.
bool Contains(const CBlockIndex *pindex) const
Efficiently check whether a block is present in this chain.
CBlockLocator GetLocator() const
Return a CBlockLocator that refers to the tip of this chain.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
const CBlock & GenesisBlock() const
const CMessageHeader::MessageMagic & DiskMagic() const
const ChainTxData & TxData() const
const Consensus::Params & GetConsensus() const
uint64_t PruneAfterHeight() const
std::optional< AssumeutxoData > AssumeutxoForHeight(int height) const
const CCheckpointData & Checkpoints() const
RAII-style controller object for a CCheckQueue that guarantees the passed queue is finished before co...
void Add(std::vector< T > &&vChecks)
Queue for verifications that have to be performed.
void SetBackend(CCoinsView &viewIn)
CCoinsView that adds a memory cache for transactions to another CCoinsView.
void AddCoin(const COutPoint &outpoint, Coin coin, bool possible_overwrite)
Add a coin.
BlockHash GetBestBlock() const override
Retrieve the block hash whose state this CCoinsView currently represents.
bool SpendCoin(const COutPoint &outpoint, Coin *moveto=nullptr)
Spend a coin.
void Uncache(const COutPoint &outpoint)
Removes the UTXO with the given outpoint from the cache, if it is not modified.
void SetBestBlock(const BlockHash &hashBlock)
unsigned int GetCacheSize() const
Calculate the size of the cache (in number of transaction outputs)
bool GetCoin(const COutPoint &outpoint, Coin &coin) const override
Retrieve the Coin (unspent transaction output) for a given outpoint.
bool HaveCoinInCache(const COutPoint &outpoint) const
Check if we have the given utxo already loaded in this cache.
bool Flush()
Push the modifications applied to this cache to its base and wipe local state.
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::optional< fs::path > StoragePath()
void ResizeCache(size_t new_cache_size) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Dynamically alter the underlying leveldb cache size.
Abstract view on the open txout dataset.
virtual bool GetCoin(const COutPoint &outpoint, Coin &coin) const
Retrieve the Coin (unspent transaction output) for a given outpoint.
CCoinsView that brings transactions from a mempool into view.
Fee rate in satoshis per kilobyte: Amount / kB.
void TransactionAddedToMempool(const CTransactionRef &, std::shared_ptr< const std::vector< Coin > >, uint64_t mempool_sequence)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void BlockConnected(ChainstateRole, const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockChecked(const CBlock &, const BlockValidationState &)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void ChainStateFlushed(ChainstateRole, const CBlockLocator &)
void insert(Span< const uint8_t > vKey)
bool contains(Span< const uint8_t > vKey) const
Closure representing one script verification.
ScriptError GetScriptError() const
ScriptExecutionMetrics GetScriptExecutionMetrics() const
TxSigCheckLimiter * pTxLimitSigChecks
ScriptExecutionMetrics metrics
PrecomputedTransactionData txdata
const CTransaction * ptxTo
CheckInputsLimiter * pBlockLimitSigChecks
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it.
void AddTransactionsUpdated(unsigned int n)
const int64_t m_max_size_bytes
size_t DynamicMemoryUsage() const
void SetLoadTried(bool load_tried)
Set whether or not we've made an attempt to load the mempool (regardless of whether the attempt was s...
Restore the UTXO in a Coin at a given COutPoint.
std::vector< Coin > vprevout
VerifyDBResult VerifyDB(Chainstate &chainstate, CCoinsView &coinsview, int nCheckLevel, int nCheckDepth) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
kernel::Notifications & m_notifications
Chainstate stores and provides an API to update our local knowledge of the current best chain.
bool IsBlockAvalancheFinalized(const CBlockIndex *pindex) const EXCLUSIVE_LOCKS_REQUIRED(!cs_avalancheFinalizedBlockIndex)
Checks if a block is finalized by avalanche voting.
const std::optional< BlockHash > m_from_snapshot_blockhash
The blockhash which is the base of the snapshot this chainstate was created from.
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...
void CheckForkWarningConditionsOnNewFork(CBlockIndex *pindexNewForkTip) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ActivateBestChain(BlockValidationState &state, std::shared_ptr< const CBlock > pblock=nullptr, avalanche::Processor *const avalanche=nullptr) EXCLUSIVE_LOCKS_REQUIRED(!m_chainstate_mutex
Find the best known block, and make it the tip of the block chain.
Mutex m_chainstate_mutex
The ChainState Mutex.
void UpdateFlags(CBlockIndex *pindex, CBlockIndex *&pindexReset, F f, C fChild, AC fAncestorWasChanged) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
CChain m_chain
The current chain of blockheaders we consult and build on.
bool HasCoinsViews() const
Does this chainstate have a UTXO set attached?
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.
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
bool ActivateBestChainStep(BlockValidationState &state, CBlockIndex *pindexMostWork, const std::shared_ptr< const CBlock > &pblock, bool &fInvalidFound, const avalanche::Processor *const avalanche=nullptr) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Try to make some progress towards making pindexMostWork the active block.
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.
void UnparkBlockImpl(CBlockIndex *pindex, bool fClearChildren) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
void CheckForkWarningConditions() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Chainstate(CTxMemPool *mempool, node::BlockManager &blockman, ChainstateManager &chainman, std::optional< BlockHash > from_snapshot_blockhash=std::nullopt)
void InvalidBlockFound(CBlockIndex *pindex, const BlockValidationState &state) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Mutex cs_avalancheFinalizedBlockIndex
void ForceFlushStateToDisk()
Unconditionally flush all changes to disk.
bool LoadGenesisBlock()
Ensures we have a genesis block in the block tree, possibly writing one to disk.
void UpdateTip(const CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(std::chrono::microsecond m_last_write)
Check warning conditions and do some notifications on new chain tip set.
void UnparkBlockAndChildren(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove parked status from a block and its descendants.
CTxMemPool * m_mempool
Optional mempool that is kept in sync with the chain.
void LoadMempool(const fs::path &load_path, fsbridge::FopenFn mockable_fopen_function=fsbridge::fopen)
Load the persisted mempool from disk.
CCoinsViewDB & CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(
std::chrono::microseconds m_last_flush
bool DisconnectTip(BlockValidationState &state, DisconnectedBlockTransactions *disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Disconnect m_chain's tip.
bool UnwindBlock(BlockValidationState &state, CBlockIndex *pindex, bool invalidate) EXCLUSIVE_LOCKS_REQUIRED(m_chainstate_mutex
bool InvalidateBlock(BlockValidationState &state, CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(!m_chainstate_mutex
Mark a block as invalid.
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.
const CBlockIndex *SnapshotBase() EXCLUSIVE_LOCKS_REQUIRED(std::set< CBlockIndex *, CBlockIndexWorkComparator > setBlockIndexCandidates
The base of the snapshot this chainstate was created from.
CRollingBloomFilter m_filterParkingPoliciesApplied
Filter to prevent parking a block due to block policies more than once.
bool ReplayBlocks()
Replay blocks that aren't fully applied to the database.
bool AvalancheFinalizeBlock(CBlockIndex *pindex, avalanche::Processor &avalanche) EXCLUSIVE_LOCKS_REQUIRED(!cs_avalancheFinalizedBlockIndex)
Mark a block as finalized by avalanche.
void ResetBlockFailureFlags(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove invalidity status from a block and its descendants.
void PruneBlockIndexCandidates()
Delete all entries in setBlockIndexCandidates that are worse than the current tip.
DisconnectResult DisconnectBlock(const CBlock &block, const CBlockIndex *pindex, CCoinsViewCache &view) EXCLUSIVE_LOCKS_REQUIRED(boo ConnectBlock)(const CBlock &block, BlockValidationState &state, CBlockIndex *pindex, CCoinsViewCache &view, BlockValidationOptions options, Amount *blockFees=nullptr, bool fJustCheck=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Apply the effects of this block (with given index) on the UTXO set represented by coins.
CBlockIndex const * m_best_fork_tip
bool ConnectTip(BlockValidationState &state, BlockPolicyValidationState &blockPolicyState, CBlockIndex *pindexNew, const std::shared_ptr< const CBlock > &pblock, DisconnectedBlockTransactions &disconnectpool, const avalanche::Processor *const avalanche=nullptr) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Connect a new block to m_chain.
void TryAddBlockIndexCandidate(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
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.
ChainstateRole GetRole() const EXCLUSIVE_LOCKS_REQUIRED(void InitCoinsDB(size_t cache_size_bytes, bool in_memory, bool should_wipe, std::string leveldb_name="chainstate")
Return the current role of the chainstate.
CBlockIndex const * m_best_fork_base
void InvalidChainFound(CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(cs_main
void UnparkBlock(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove parked status from a block.
bool PreciousBlock(BlockValidationState &state, CBlockIndex *pindex, avalanche::Processor *const avalanche=nullptr) EXCLUSIVE_LOCKS_REQUIRED(!m_chainstate_mutex
Mark a block as precious and reorganize.
void ClearAvalancheFinalizedBlock() EXCLUSIVE_LOCKS_REQUIRED(!cs_avalancheFinalizedBlockIndex)
Clear avalanche finalization.
void ClearBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex * FindForkInGlobalIndex(const CBlockLocator &locator) const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Find the last common block of this chain and a locator.
CBlockIndex * FindMostWorkChain(std::vector< const CBlockIndex * > &blocksToReconcile, bool fAutoUnpark) 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...
bool UpdateFlagsForBlock(CBlockIndex *pindexBase, CBlockIndex *pindex, F f) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ParkBlock(BlockValidationState &state, CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(!m_chainstate_mutex
Park a block.
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.
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
SnapshotCompletionResult MaybeCompleteSnapshotValidation(std::function< void(bilingual_str)> shutdown_fnc=[](bilingual_str msg) { AbortNode(msg.original, msg);}) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetSnapshotBaseBlock() const EXCLUSIVE_LOCKS_REQUIRED(Chainstate ActiveChainstate)() const
Once the background validation chainstate has reached the height which is the base of the UTXO snapsh...
std::atomic< int32_t > nBlockSequenceId
Every received block is assigned a unique and increasing identifier, so we know which one to give pri...
bool DetectSnapshotChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(void ResetChainstates() EXCLUSIVE_LOCKS_REQUIRED(bool DeleteSnapshotChainstate() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &ActivateExistingSnapshot(CTxMemPool *mempool, BlockHash base_blockhash) EXCLUSIVE_LOCKS_REQUIRED(bool ValidatedSnapshotCleanup() EXCLUSIVE_LOCKS_REQUIRED(boo DumpRecentHeadersTime)(const fs::path &filePath) const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
When starting up, search the datadir for a chainstate based on a UTXO snapshot that is in the process...
const Config & GetConfig() const
std::optional< int > GetSnapshotBaseHeight() const EXCLUSIVE_LOCKS_REQUIRED(bool IsUsable(const Chainstate *const pchainstate) const EXCLUSIVE_LOCKS_REQUIRED(
Return the height of the base block of the snapshot in use, if one exists, else nullopt.
int64_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.
int64_t m_total_coinsdb_cache
The total number of bytes available for us to use across all leveldb coins databases.
bool AcceptBlockHeader(const CBlockHeader &block, BlockValidationState &state, CBlockIndex **ppindex, bool min_pow_checked, const std::optional< CCheckpointData > &test_checkpoints=std::nullopt) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
If a block header hasn't already been seen, call CheckBlockHeader on it, ensure that it doesn't desce...
kernel::Notifications & GetNotifications() const
bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network)
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
bool ProcessNewBlock(const std::shared_ptr< const CBlock > &block, bool force_processing, bool min_pow_checked, bool *new_block, avalanche::Processor *const avalanche=nullptr) LOCKS_EXCLUDED(cs_main)
Process an incoming block.
bool LoadRecentHeadersTime(const fs::path &filePath) EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Load the recent block headers reception time from a file.
void LoadExternalBlockFile(FILE *fileIn, FlatFilePos *dbp=nullptr, std::multimap< BlockHash, FlatFilePos > *blocks_with_unknown_parent=nullptr, avalanche::Processor *const avalanche=nullptr)
Import blocks from an external file.
std::optional< BlockHash > SnapshotBlockhash() const
bool IsSnapshotValidated() const EXCLUSIVE_LOCKS_REQUIRED(
Is there a snapshot in use and has it been fully validated?
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
bool BackgroundSyncInProgress() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
The state of a background sync (for net processing)
std::atomic< bool > m_cached_finished_ibd
Whether initial block download has ended and IsInitialBlockDownload should return false from now on.
bool PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
int ActiveHeight() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
bool ActivateSnapshot(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Construct and activate a Chainstate on the basis of UTXO snapshot data.
bool IsSnapshotActive() const
bool AcceptBlock(const std::shared_ptr< const CBlock > &pblock, BlockValidationState &state, 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).
const CChainParams & GetParams() const
bool ProcessNewBlockHeaders(const std::vector< CBlockHeader > &block, bool min_pow_checked, BlockValidationState &state, const CBlockIndex **ppindex=nullptr, const std::optional< CCheckpointData > &test_checkpoints=std::nullopt) LOCKS_EXCLUDED(cs_main)
Process incoming block headers.
const Consensus::Params & GetConsensus() const
const arith_uint256 & MinimumChainWork() const
void CheckBlockIndex()
Make various assertions about the state of the block index.
bool LoadBlockIndex() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Load the block tree and coins database from disk, initializing state if we're running with -reindex.
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
void MaybeRebalanceCaches() EXCLUSIVE_LOCKS_REQUIRED(void ReportHeadersPresync(const arith_uint256 &work, int64_t height, int64_t timestamp)
Check to see if caches are out of balance and if so, call ResizeCoinsCaches() as needed.
arith_uint256 nLastPreciousChainwork
chainwork for the last block that preciousblock has been applied to.
const BlockHash & AssumedValidBlock() const
ChainstateManager(Options options, node::BlockManager::Options blockman_options)
Chainstate &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(std::vector< Chainstate * GetAll)()
Instantiate a new chainstate.
std::set< CBlockIndex * > m_failed_blocks
In order to efficiently track invalidity of headers, we keep the set of blocks which we tried to conn...
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...
uint32_t GetHeight() const
CoinsViews(DBParams db_params, CoinsViewOptions options)
This constructor initializes CCoinsViewDB and CCoinsViewErrorCatcher instances, but it does not creat...
virtual const CChainParams & GetChainParams() const =0
void updateMempoolForReorg(Chainstate &active_chainstate, bool fAddToMempool, CTxMemPool &pool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Make mempool consistent after a reorg, by re-adding or recursively erasing disconnected block transac...
void addForBlock(const std::vector< CTransactionRef > &vtx, CTxMemPool &pool) EXCLUSIVE_LOCKS_REQUIRED(pool.cs)
void importMempool(CTxMemPool &pool) EXCLUSIVE_LOCKS_REQUIRED(pool.cs)
Different type to mark Mutex at global scope.
static RCUPtr acquire(T *&ptrIn)
Acquire ownership of some pointer.
The script cache is a map using a key/value element, that caches the success of executing a specific ...
static TxSigCheckLimiter getDisabled()
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
256-bit unsigned big integer.
std::string ToString() const
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
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 warning(const std::string &warning)
virtual void progress(const bilingual_str &title, int progress_percent, bool resume_possible)
virtual void blockTip(SynchronizationState state, CBlockIndex &index)
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
bool ReadBlockFromDisk(CBlock &block, const FlatFilePos &pos) const
Functions for disk access for blocks.
RecursiveMutex cs_LastBlockFile
bool CheckBlockDataAvailability(const CBlockIndex &upper_block LIFETIMEBOUND, const CBlockIndex &lower_block LIFETIMEBOUND) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetFirstStoredBlock(const CBlockIndex &start_block LIFETIMEBOUND, const CBlockIndex *lower_block=nullptr) EXCLUSIVE_LOCKS_REQUIRED(boo m_have_pruned)
Check if all blocks in the [upper_block, lower_block] range have data available.
void FindFilesToPrune(std::set< int > &setFilesToPrune, uint64_t nPruneAfterHeight, int chain_tip_height, int prune_height, bool is_ibd)
Prune block and undo files (blk???.dat and undo???.dat) so that the disk space used is less than a us...
bool WriteUndoDataForBlock(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos SaveBlockToDisk(const CBlock &block, int nHeight, const FlatFilePos *dbp)
Store block on disk.
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, int chain_tip_height)
Calculate the block/rev files to delete based on height specified by user with RPC command pruneblock...
CBlockIndex * LookupBlockIndex(const BlockHash &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool LoadingBlocks() const
bool UndoReadFromDisk(CBlockUndo &blockundo, const CBlockIndex &index) const
void UnlinkPrunedFiles(const std::set< int > &setFilesToPrune) const
Actually unlink the specified files.
bool WriteBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(bool LoadBlockIndexDB(const std::optional< BlockHash > &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.
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
bool FlushBlockFile(bool fFinalize=false, bool finalize_undo=false)
Return false if block file or undo file flushing fails.
bool m_check_for_pruning
Global flag to indicate we should check to see if there are block/undo files that should be deleted.
bool IsPruneMode() const
Whether running in -prune mode.
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::string FormatFullVersion()
static constexpr int CLIENT_VERSION
bitcoind-res.rc includes this file, but it cannot cope with real c++ code.
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.
@ BLOCK_CHECKPOINT
the block failed to meet one of our checkpoints
@ 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)
@ TX_MISSING_INPUTS
transaction was missing some of its inputs
@ TX_CHILD_BEFORE_PARENT
This tx outputs are already spent in the mempool.
@ TX_MEMPOOL_POLICY
violated mempool's fee/size/descendant/etc limits
@ TX_PACKAGE_RECONSIDERABLE
fails some policy, but might be acceptable if submitted in a (different) package
@ TX_PREMATURE_SPEND
transaction spends a coinbase too early, or violates locktime/sequence locks
@ TX_DUPLICATE
Tx already in mempool or in the chain.
@ TX_INPUTS_NOT_STANDARD
inputs failed policy rules
@ TX_CONFLICT
Tx conflicts with a finalized tx, i.e.
@ TX_NOT_STANDARD
otherwise didn't meet our local policy rules
@ TX_AVALANCHE_RECONSIDERABLE
fails some policy, but might be reconsidered by avalanche voting
@ TX_NO_MEMPOOL
this node does not have a mempool so can't validate the transaction
@ TX_CONSENSUS
invalid by consensus rules
static constexpr unsigned int LOCKTIME_VERIFY_SEQUENCE
Flags for nSequence and nLockTime locks.
static const uint64_t MAX_TX_SIZE
The maximum allowed size for a transaction, in bytes.
uint64_t GetMaxBlockSigChecksCount(uint64_t maxBlockSize)
Compute the maximum number of sigchecks that can be contained in a block given the MAXIMUM block size...
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
bool DeploymentActiveAfter(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep)
Determine if a deployment is active for the next block.
bool DeploymentActiveAt(const CBlockIndex &index, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep)
Determine if a deployment is active for this block.
bool RenameOver(fs::path src, fs::path dest)
bool CheckDiskSpace(const fs::path &dir, uint64_t additional_bytes)
bool FileCommit(FILE *file)
Ensure file contents are fully committed to disk, using a platform-specific feature analogous to fsyn...
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, uint32_t flags, const BaseSignatureChecker &checker, ScriptExecutionMetrics &metricsOut, ScriptError *serror)
Execute an unlocking and locking script together.
ChainstateRole
This enum describes the various roles a specific Chainstate instance can take.
bool error(const char *fmt, const Args &...args)
#define LogPrintLevel(category, level,...)
#define LogPrint(category,...)
uint256 BlockMerkleRoot(const CBlock &block, bool *mutated)
Compute the Merkle root of the transactions in a block.
bool CheckBlock(const CCheckpointData &data, int nHeight, const BlockHash &hash)
Returns true if block passes checkpoint checks.
bool CheckTxInputs(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, int nSpendHeight, Amount &txfee)
Check whether all inputs of this transaction are valid (no double spends and amounts).
static bool exists(const path &p)
static std::string PathToString(const path &path)
Convert path object to byte string.
FILE * fopen(const fs::path &p, const char *mode)
std::string get_filesystem_error_message(const fs::filesystem_error &e)
std::function< FILE *(const fs::path &, const char *)> FopenFn
static bool ComputeUTXOStats(CCoinsView *view, CCoinsStats &stats, T hash_obj, const std::function< void()> &interruption_point)
Calculate statistics about the unspent transaction output set.
bool LoadMempool(CTxMemPool &pool, const fs::path &load_path, Chainstate &active_chainstate, FopenFn mockable_fopen_function)
static const unsigned int UNDOFILE_CHUNK_SIZE
The pre-allocation chunk size for rev?????.dat files (since 0.8)
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 > FindSnapshotChainstateDir()
Return a path to the snapshot-based chainstate dir, if one exists.
std::unordered_map< BlockHash, CBlockIndex, BlockHasher > BlockMap
std::optional< BlockHash > ReadSnapshotBaseBlockhash(const fs::path &chaindir)
static constexpr unsigned int BLOCKFILE_CHUNK_SIZE
The pre-allocation chunk size for blk?????.dat files (since 0.8)
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate) EXCLUSIVE_LOCKS_REQUIRED(std::optional< BlockHash > ReadSnapshotBaseBlockhash(const 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::atomic_bool fReindex
bool Func(const std::string &str, Span< const char > &sp)
Parse a function call.
Implement std::hash so RCUPtr can be used as a key for maps or sets.
bilingual_str ErrorString(const Result< T > &result)
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 CheckPackage(const Package &txns, PackageValidationState &state)
Context-free package policy checks:
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.
bool AreInputsStandard(const CTransaction &tx, const CCoinsViewCache &mapInputs, uint32_t flags)
Check transaction inputs to mitigate two potential denial-of-service attacks:
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 uint32_t STANDARD_SCRIPT_VERIFY_FLAGS
Standard script verification flags that standard transactions will comply with.
static constexpr uint32_t STANDARD_LOCKTIME_VERIFY_FLAGS
Used as the flags parameter to sequence and nLocktime checks in non-consensus code.
bool CheckProofOfWork(const BlockHash &hash, uint32_t nBits, const Consensus::Params ¶ms)
Check whether a block hash satisfies the proof-of-work requirement specified by nBits.
uint32_t GetNextWorkRequired(const CBlockIndex *pindexPrev, const CBlockHeader *pblock, const CChainParams &chainParams)
std::shared_ptr< const CTransaction > CTransactionRef
uint256 GetRandHash() noexcept
reverse_range< T > reverse_iterate(T &x)
std::string ScriptErrorString(const ScriptError serror)
@ SIGCHECKS_LIMIT_EXCEEDED
@ SCRIPT_VERIFY_SIGPUSHONLY
@ SCRIPT_VERIFY_CHECKLOCKTIMEVERIFY
@ SCRIPT_ENABLE_REPLAY_PROTECTION
@ SCRIPT_ENABLE_SCHNORR_MULTISIG
@ SCRIPT_VERIFY_STRICTENC
@ SCRIPT_ENFORCE_SIGCHECKS
@ SCRIPT_VERIFY_CLEANSTACK
@ SCRIPT_VERIFY_MINIMALDATA
@ SCRIPT_VERIFY_CHECKSEQUENCEVERIFY
@ SCRIPT_ENABLE_SIGHASH_FORKID
void AddKeyInScriptCache(ScriptCacheKey key, int nSigChecks)
Add an entry in the cache.
bool IsKeyInScriptCache(ScriptCacheKey key, bool erase, int &nSigChecksOut)
Check if a given key is in the cache, and if so, return its values.
size_t GetSerializeSize(const T &t, int nVersion=0)
bool ShutdownRequested()
Returns true if a shutdown is requested, false otherwise.
void StartShutdown()
Request shutdown of the application.
std::string ToString(const T &t)
Locale-independent version of std::to_string.
static constexpr Amount zero() noexcept
Holds configuration for use during UTXO snapshot load and validation.
AssumeutxoHash hash_serialized
The expected hash of the deserialized UTXO set.
unsigned int nChainTx
Used to populate the nChainTx value, which is used during BlockManager::LoadBlockIndex().
A BlockHash is a unqiue identifier for a block.
bool isValid(enum BlockValidity nUpTo=BlockValidity::TRANSACTIONS) const
Check whether this block index entry is valid up to the passed validity level.
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
std::vector< BlockHash > vHave
Holds various statistics on transactions within a chain.
User-controlled performance and debug options.
Parameters that influence chain consensus.
int shibusawaActivationTime
Unix time used for MTP activation of 15 Nov 2025 12:00:00 UTC upgrade.
int BIP34Height
Block height and hash at which BIP34 becomes active.
int nSubsidyHalvingInterval
BlockHash hashGenesisBlock
int64_t nPowTargetSpacing
bool fPowAllowMinDifficultyBlocks
Application-specific storage settings.
fs::path path
Location in the filesystem where leveldb data will be stored.
Validation result for a transaction evaluated by MemPoolAccept (single or package).
const ResultType m_result_type
Result type.
@ VALID
Fully validated, valid.
static MempoolAcceptResult Failure(TxValidationState state)
static MempoolAcceptResult FeeFailure(TxValidationState state, CFeeRate effective_feerate, const std::vector< TxId > &txids_fee_calculations)
static MempoolAcceptResult Success(int64_t vsize, Amount fees, CFeeRate effective_feerate, const std::vector< TxId > &txids_fee_calculations)
Constructor for success case.
static MempoolAcceptResult MempoolTx(int64_t vsize, Amount fees)
Constructor for already-in-mempool case.
std::chrono::time_point< NodeClock > time_point
Validation result for package mempool acceptance.
Precompute sighash midstate to avoid quadratic hashing.
const char * what() const override
A TxId is the identifier of a transaction.
An options struct for BlockManager, more ergonomically referred to as BlockManager::Options due to th...
An options struct for ChainstateManager, more ergonomically referred to as ChainstateManager::Options...
const std::function< NodeClock::time_point()> adjusted_time_callback
std::optional< bool > 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.
bool store_recent_headers_time
If set, store and load the last few block headers reception time to speed up RTT bootstraping.
CoinsViewOptions coins_view
#define AssertLockNotHeld(cs)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define EXCLUSIVE_LOCKS_REQUIRED(...)
#define LOCKS_EXCLUDED(...)
#define NO_THREAD_SAFETY_ANALYSIS
int64_t GetTimeMicros()
Returns the system time (not mockable)
int64_t GetTimeMillis()
Returns the system time (not mockable)
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.
#define LOG_TIME_MILLIS_WITH_CATEGORY(end_msg, log_category)
#define LOG_TIME_MILLIS_WITH_CATEGORY_MSG_ONCE(end_msg, log_category)
#define TRACE6(context, event, a, b, c, d, e, f)
#define TRACE5(context, event, a, b, c, d, e)
bilingual_str _(const char *psz)
Translation function.
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
bool CheckRegularTransaction(const CTransaction &tx, TxValidationState &state)
Context-independent validity checks for coinbase and non-coinbase transactions.
bool CheckCoinbase(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...
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 ContextualCheckTransaction(const Consensus::Params ¶ms, const CTransaction &tx, TxValidationState &state, int nHeight, int64_t nMedianTimePast)
Context dependent validity checks for non coinbase transactions.
static const uint32_t MEMPOOL_HEIGHT
Fake height value used in Coins to signify they are only in the memory pool(since 0....
uint256 uint256S(const char *str)
uint256 from const char *.
static bool DeleteCoinsDBFromDisk(const fs::path &db_path, bool is_snapshot) EXCLUSIVE_LOCKS_REQUIRED(
static bool NotifyHeaderTip(ChainstateManager &chainman) LOCKS_EXCLUDED(cs_main)
void StartScriptCheckWorkerThreads(int threads_num)
Run instances of script checking worker threads.
static int64_t nTimeConnectTotal
GlobalMutex g_best_block_mutex
Amount GetBlockSubsidy(int nHeight, const Consensus::Params &consensusParams)
std::condition_variable g_best_block_cv
std::optional< LockPoints > CalculateLockPointsAtTip(CBlockIndex *tip, const CCoinsView &coins_view, const CTransaction &tx)
Calculate LockPoints required to check if transaction will be BIP68 final in the next block to be cre...
arith_uint256 CalculateHeadersWork(const std::vector< CBlockHeader > &headers)
Return the sum of the work on a given set of headers.
DisconnectResult ApplyBlockUndo(CBlockUndo &&blockUndo, const CBlock &block, const CBlockIndex *pindex, CCoinsViewCache &view)
Undo a block from the block and the undoblock data.
double GuessVerificationProgress(const ChainTxData &data, const CBlockIndex *pindex)
Guess how far we are in the verification process at the given block index require cs_main if pindex h...
MempoolAcceptResult AcceptToMemoryPool(Chainstate &active_chainstate, const CTransactionRef &tx, int64_t accept_time, bool bypass_limits, bool test_accept, unsigned int heightOverride)
Try to add a transaction to the mempool.
static int64_t nBlocksTotal
static int64_t nTimePostConnect
static bool CheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, const Consensus::Params ¶ms, BlockValidationOptions validationOptions)
Return true if the provided block header is valid.
static SynchronizationState GetSynchronizationState(bool init)
static void SnapshotUTXOHashBreakpoint()
static int64_t nTimeFlush
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 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 uint32_t GetNextBlockScriptFlags(const CBlockIndex *pindex, const ChainstateManager &chainman)
const CBlockIndex * g_best_block
Used to notify getblocktemplate RPC of new tips.
bool HasValidProofOfWork(const std::vector< CBlockHeader > &headers, const Consensus::Params &consensusParams)
Check with the proof of work on each blockheader matches the value in nBits.
static constexpr std::chrono::hours DATABASE_FLUSH_INTERVAL
Time to wait between flushing chainstate to disk.
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept)
Validate (and maybe submit) a package to the mempool.
static CCheckQueue< CScriptCheck > scriptcheckqueue(128)
static ChainstateManager::Options && Flatten(ChainstateManager::Options &&opts)
Apply default chain params to nullopt members.
static constexpr int PRUNE_LOCK_BUFFER
The number of blocks to keep below the deepest prune lock.
static int64_t nTimeVerify
void StopScriptCheckWorkerThreads()
Stop all of the script checking worker threads.
static void LimitValidationInterfaceQueue() LOCKS_EXCLUDED(cs_main)
return CheckInputScripts(tx, state, view, flags, true, true, txdata, nSigChecksOut)
static bool CheckInputsFromMempoolAndCache(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &view, const CTxMemPool &pool, const uint32_t flags, PrecomputedTransactionData &txdata, int &nSigChecksOut, CCoinsViewCache &coins_tip) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Checks to avoid mempool polluting consensus critical paths since cached signature and script validity...
void SpendCoins(CCoinsViewCache &view, const CTransaction &tx, CTxUndo &txundo, int nHeight)
Mark all the coins corresponding to a given transaction inputs as spent.
static int64_t nTimeTotal
static int64_t nTimeConnect
bool CheckBlock(const CBlock &block, BlockValidationState &state, const Consensus::Params ¶ms, BlockValidationOptions validationOptions)
Functions for validating blocks and updating the block tree.
bool ContextualCheckTransactionForCurrentBlock(const CBlockIndex &active_chain_tip, const Consensus::Params ¶ms, const CTransaction &tx, TxValidationState &state)
const std::vector< std::string > CHECKLEVEL_DOC
Documentation for argument 'checklevel'.
DisconnectResult UndoCoinSpend(Coin &&undo, CCoinsViewCache &view, const COutPoint &out)
Restore the UTXO in a Coin at a given COutPoint.
static int64_t nTimeIndex
bool TestBlockValidity(BlockValidationState &state, const CChainParams ¶ms, Chainstate &chainstate, const CBlock &block, CBlockIndex *pindexPrev, const std::function< NodeClock::time_point()> &adjusted_time_callback, BlockValidationOptions validationOptions)
void PruneBlockFilesManual(Chainstate &active_chainstate, int nManualPruneHeight)
Prune block files up to a given height.
static void FlushSnapshotToDisk(CCoinsViewCache &coins_cache, bool snapshot_loaded)
bool AbortNode(BlockValidationState &state, const std::string &strMessage, const bilingual_str &userMessage)
void UpdateCoins(CCoinsViewCache &view, const CTransaction &tx, CTxUndo &txundo, int nHeight)
Apply the effects of this transaction on the UTXO set represented by view.
static bool ContextualCheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, BlockManager &blockman, ChainstateManager &chainman, const CBlockIndex *pindexPrev, NodeClock::time_point now, const std::optional< CCheckpointData > &test_checkpoints=std::nullopt) EXCLUSIVE_LOCKS_REQUIRED(
Context-dependent validity checks.
static int64_t nTimeForks
static constexpr uint64_t HEADERS_TIME_VERSION
static int64_t nTimeCheck
static fs::path GetSnapshotCoinsDBPath(Chainstate &cs) EXCLUSIVE_LOCKS_REQUIRED(
static void UpdateTipLog(const CCoinsViewCache &coins_tip, const CBlockIndex *tip, const CChainParams ¶ms, const std::string &func_name, const std::string &prefix) EXCLUSIVE_LOCKS_REQUIRED(
static constexpr std::chrono::hours DATABASE_WRITE_INTERVAL
Time to wait between writing blocks/block index to disk.
static int64_t nTimeChainState
static int64_t nTimeReadFromDisk
static bool IsReplayProtectionEnabled(const Consensus::Params ¶ms, int64_t nMedianTimePast)
#define MIN_TRANSACTION_SIZE
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...
@ BASE_BLOCKHASH_MISMATCH
SynchronizationState
Current sync state passed to tip changed callbacks.
@ LARGE
The cache is at >= 90% capacity.
@ CRITICAL
The coins cache is in immediate need of a flush.
static const int DEFAULT_STOPATHEIGHT
Default for -stopatheight.
CMainSignals & GetMainSignals()
void SyncWithValidationInterfaceQueue()
This is a synonym for the following, which asserts certain locks are not held: std::promise<void> pro...
static const int PROTOCOL_VERSION
network protocol versioning
void SetfLargeWorkInvalidChainFound(bool flag)
void SetfLargeWorkForkFound(bool flag)
bool GetfLargeWorkForkFound()