99 "level 0 reads the blocks from disk",
100 "level 1 verifies block validity",
101 "level 2 verifies undo data",
102 "level 3 checks disconnection of tip blocks",
103 "level 4 tries to reconnect the blocks",
104 "each level includes the checks of the previous levels",
121 : excessiveBlockSize(config.GetMaxBlockSize()), checkPoW(true),
122 checkMerkleRoot(true) {}
159std::optional<std::vector<int>> CalculatePrevHeights(
const CBlockIndex &tip,
161 const CTransaction &tx) {
162 std::vector<int> prev_heights;
163 prev_heights.resize(tx.vin.size());
164 for (
size_t i = 0; i < tx.vin.size(); ++i) {
165 const CTxIn &txin = tx.vin[i];
167 if (!coins.
GetCoin(txin.prevout, coin)) {
168 LogPrintf(
"ERROR: %s: Missing input %d in transaction \'%s\'\n",
169 __func__, i, tx.GetId().GetHex());
174 prev_heights[i] = tip.
nHeight + 1;
185 const CTransaction &tx) {
188 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
189 if (!prev_heights.has_value()) {
194 next_tip.
pprev = tip;
227 int64_t nMedianTimePast) {
228 return nMedianTimePast >=
gArgs.
GetIntArg(
"-replayprotectionactivationtime",
234 if (pindexPrev ==
nullptr) {
255 for (
const CTxIn &txin : tx.vin) {
256 const Coin &coin = view.AccessCoin(txin.prevout);
271 assert(txFrom->GetId() == txin.prevout.GetTxId());
272 assert(txFrom->vout.size() > txin.prevout.GetN());
275 const Coin &coinFromUTXOSet = coins_tip.AccessCoin(txin.prevout);
284 true, txdata, nSigChecksOut);
292 : m_pool(mempool), m_view(&m_dummy),
293 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
294 m_active_chainstate(active_chainstate) {}
300 const int64_t m_accept_time;
301 const bool m_bypass_limits;
309 std::vector<COutPoint> &m_coins_to_uncache;
310 const bool m_test_accept;
311 const unsigned int m_heightOverride;
317 const bool m_package_submission;
323 const bool m_package_feerates;
326 static ATMPArgs SingleAccept(
const Config &config, int64_t accept_time,
328 std::vector<COutPoint> &coins_to_uncache,
330 unsigned int heightOverride) {
348 PackageTestAccept(
const Config &config, int64_t accept_time,
349 std::vector<COutPoint> &coins_to_uncache) {
365 PackageChildWithParents(
const Config &config, int64_t accept_time,
366 std::vector<COutPoint> &coins_to_uncache) {
380 static ATMPArgs SingleInPackageAccept(
const ATMPArgs &package_args) {
382 package_args.m_config,
383 package_args.m_accept_time,
385 package_args.m_coins_to_uncache,
386 package_args.m_test_accept,
387 package_args.m_heightOverride,
399 ATMPArgs(
const Config &config, int64_t accept_time,
bool bypass_limits,
400 std::vector<COutPoint> &coins_to_uncache,
bool test_accept,
401 unsigned int height_override,
bool package_submission,
402 bool package_feerates)
403 : m_config{config}, m_accept_time{accept_time},
404 m_bypass_limits{bypass_limits},
405 m_coins_to_uncache{coins_to_uncache}, m_test_accept{test_accept},
406 m_heightOverride{height_override},
407 m_package_submission{package_submission},
408 m_package_feerates(package_feerates) {}
423 AcceptMultipleTransactions(
const std::vector<CTransactionRef> &txns,
440 AcceptSubPackage(
const std::vector<CTransactionRef> &subpackage,
458 const uint32_t next_block_script_verify_flags)
460 m_next_block_script_verify_flags(next_block_script_verify_flags) {
467 std::unique_ptr<CTxMemPoolEntry> m_entry;
505 const uint32_t m_next_block_script_verify_flags;
506 int m_sig_checks_standard;
514 bool PreChecks(ATMPArgs &args, Workspace &ws)
521 bool ConsensusScriptChecks(
const ATMPArgs &args, Workspace &ws)
527 bool Finalize(
const ATMPArgs &args, Workspace &ws)
535 bool SubmitPackage(
const ATMPArgs &args, std::vector<Workspace> &workspaces,
537 std::map<TxId, MempoolAcceptResult> &results)
541 bool CheckFeeRate(
size_t package_size,
size_t package_vsize,
547 const Amount mempoolRejectFee =
548 m_pool.GetMinFee().GetFee(package_vsize);
551 package_fee < mempoolRejectFee) {
552 return state.Invalid(
554 "mempool min fee not met",
555 strprintf(
"%d < %d", package_fee, mempoolRejectFee));
560 if (package_fee < m_pool.m_min_relay_feerate.GetFee(package_size)) {
561 return state.Invalid(
563 "min relay fee not met",
565 m_pool.m_min_relay_feerate.GetFee(package_size)));
580bool MemPoolAccept::PreChecks(ATMPArgs &args, Workspace &ws) {
584 const CTransaction &tx = *ws.m_ptx;
585 const TxId &txid = ws.m_ptx->GetId();
588 const int64_t nAcceptTime = args.m_accept_time;
589 const bool bypass_limits = args.m_bypass_limits;
590 std::vector<COutPoint> &coins_to_uncache = args.m_coins_to_uncache;
591 const unsigned int heightOverride = args.m_heightOverride;
603 if (m_pool.m_require_standard &&
605 m_pool.m_permit_bare_multisig,
606 m_pool.m_dust_relay_feerate, reason)) {
615 *
Assert(m_active_chainstate.m_chain.Tip()),
616 args.m_config.GetChainParams().GetConsensus(), tx, ctxState)) {
625 if (m_pool.exists(txid)) {
627 "txn-already-in-mempool");
631 for (
const CTxIn &txin : tx.vin) {
632 if (
const auto ptxConflicting = m_pool.GetConflictTx(txin.prevout)) {
633 if (m_pool.isAvalancheFinalized(ptxConflicting->GetId())) {
635 "finalized-tx-conflict");
640 "txn-mempool-conflict");
644 m_view.SetBackend(m_viewmempool);
648 for (
const CTxIn &txin : tx.vin) {
650 coins_to_uncache.push_back(txin.prevout);
657 if (!m_view.HaveCoin(txin.prevout)) {
659 for (
size_t out = 0; out < tx.vout.size(); out++) {
664 "txn-already-known");
671 "bad-txns-inputs-missingorspent");
676 if (!m_view.HaveInputs(tx)) {
678 "bad-txns-inputs-spent");
682 m_view.GetBestBlock();
687 m_view.SetBackend(m_dummy);
689 assert(m_active_chainstate.m_blockman.LookupBlockIndex(
690 m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
699 m_active_chainstate.m_chain.Tip(), m_view, tx)};
700 if (!lock_points.has_value() ||
710 m_active_chainstate.m_chain.Height() + 1,
717 if (m_pool.m_require_standard &&
720 "bad-txns-nonstandard-inputs");
724 ws.m_modified_fees = ws.m_base_fees;
725 m_pool.ApplyDelta(txid, ws.m_modified_fees);
727 unsigned int nSize = tx.GetTotalSize();
730 const uint32_t scriptVerifyFlags =
734 ws.m_precomputed_txdata, ws.m_sig_checks_standard)) {
739 ws.m_entry = std::make_unique<CTxMemPoolEntry>(
740 ptx, ws.m_base_fees, nAcceptTime,
741 heightOverride ? heightOverride : m_active_chainstate.m_chain.Height(),
742 ws.m_sig_checks_standard, lock_points.value());
744 ws.m_vsize = ws.m_entry->GetTxVirtualSize();
752 if (!bypass_limits &&
754 m_pool.m_min_relay_feerate.GetFee(ws.m_ptx->GetTotalSize())) {
761 m_pool.m_min_relay_feerate.GetFee(nSize)));
766 if (!bypass_limits && !args.m_package_feerates &&
767 !CheckFeeRate(nSize, ws.m_vsize, ws.m_modified_fees, state)) {
774bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs &args, Workspace &ws) {
777 const CTransaction &tx = *ws.m_ptx;
778 const TxId &txid = tx.GetId();
792 int nSigChecksConsensus;
794 tx, state, m_view, m_pool, ws.m_next_block_script_verify_flags,
795 ws.m_precomputed_txdata, nSigChecksConsensus,
796 m_active_chainstate.CoinsTip())) {
800 LogPrintf(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against "
801 "latest-block but not STANDARD flags %s, %s\n",
806 if (ws.m_sig_checks_standard != nSigChecksConsensus) {
811 "%s: BUG! PLEASE REPORT THIS! SigChecks count differed between "
812 "standard and consensus flags in %s",
821 std::vector<Coin> spent_coins;
822 spent_coins.reserve(ptx->vin.size());
823 for (
const CTxIn &input : ptx->vin) {
825 const bool coinFound = coins_view.
GetCoin(input.prevout, coin);
827 spent_coins.push_back(std::move(coin));
832bool MemPoolAccept::Finalize(
const ATMPArgs &args, Workspace &ws) {
835 const TxId &txid = ws.m_ptx->GetId();
837 const bool bypass_limits = args.m_bypass_limits;
842 m_pool.addUnchecked(entry);
846 std::make_shared<
const std::vector<Coin>>(
847 getSpentCoins(ws.m_ptx, m_view)),
848 m_pool.GetAndIncrementSequence());
855 if (!args.m_package_submission && !bypass_limits) {
856 m_pool.LimitSize(m_active_chainstate.CoinsTip());
857 if (!m_pool.exists(txid)) {
867bool MemPoolAccept::SubmitPackage(
868 const ATMPArgs &args, std::vector<Workspace> &workspaces,
870 std::map<TxId, MempoolAcceptResult> &results) {
875 workspaces.cbegin(), workspaces.cend(),
876 [
this](
const auto &ws) { return !m_pool.exists(ws.m_ptx->GetId()); }));
878 bool all_submitted =
true;
885 for (Workspace &ws : workspaces) {
886 if (!ConsensusScriptChecks(args, ws)) {
887 results.emplace(ws.m_ptx->GetId(),
890 all_submitted =
false;
893 strprintf(
"BUG! PolicyScriptChecks succeeded but "
894 "ConsensusScriptChecks failed: %s",
895 ws.m_ptx->GetId().ToString()));
905 if (!Finalize(args, ws)) {
906 results.emplace(ws.m_ptx->GetId(),
909 all_submitted =
false;
911 strprintf(
"BUG! Adding to mempool failed: %s",
912 ws.m_ptx->GetId().ToString()));
918 m_pool.LimitSize(m_active_chainstate.CoinsTip());
920 std::vector<TxId> all_package_txids;
921 all_package_txids.reserve(workspaces.size());
922 std::transform(workspaces.cbegin(), workspaces.cend(),
923 std::back_inserter(all_package_txids),
924 [](
const auto &ws) { return ws.m_ptx->GetId(); });
928 for (Workspace &ws : workspaces) {
929 const auto effective_feerate =
930 args.m_package_feerates
931 ? ws.m_package_feerate
933 static_cast<uint32_t
>(ws.m_vsize)};
934 const auto effective_feerate_txids =
935 args.m_package_feerates ? all_package_txids
936 : std::vector<TxId>({ws.m_ptx->GetId()});
937 results.emplace(ws.m_ptx->GetId(),
940 effective_feerate_txids));
942 return all_submitted;
953 const CBlockIndex *tip = m_active_chainstate.m_chain.Tip();
958 const std::vector<TxId> single_txid{ws.m_ptx->GetId()};
963 if (!PreChecks(args, ws)) {
964 if (ws.m_state.GetResult() ==
969 ws.m_state,
CFeeRate(ws.m_modified_fees, ws.m_vsize),
975 if (!ConsensusScriptChecks(args, ws)) {
979 const TxId txid = ptx->GetId();
989 if (
auto it = m_pool.mapNextTx.lower_bound(COutPoint{txid, 0});
990 it != m_pool.mapNextTx.end() && it->first->GetTxId() == txid) {
991 LogPrintf(
"%s: BUG! PLEASE REPORT THIS! Attempt to add txid %s, but "
992 "its outputs are already spent in the "
996 "txn-child-before-parent");
1000 const CFeeRate effective_feerate{ws.m_modified_fees,
1001 static_cast<uint32_t
>(ws.m_vsize)};
1003 if (args.m_test_accept) {
1005 effective_feerate, single_txid);
1008 if (!Finalize(args, ws)) {
1012 Assume(ws.m_state.GetResult() ==
1015 ws.m_state,
CFeeRate(ws.m_modified_fees, ws.m_vsize), single_txid);
1019 effective_feerate, single_txid);
1023 const std::vector<CTransactionRef> &txns, ATMPArgs &args) {
1033 std::vector<Workspace> workspaces{};
1034 workspaces.reserve(txns.size());
1036 txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1037 [
this](
const auto &tx) {
1039 tx, GetNextBlockScriptFlags(m_active_chainstate.m_chain.Tip(),
1040 m_active_chainstate.m_chainman));
1042 std::map<TxId, MempoolAcceptResult> results;
1048 std::vector<TxId> valid_txids;
1049 for (Workspace &ws : workspaces) {
1050 if (!PreChecks(args, ws)) {
1052 "transaction failed");
1055 results.emplace(ws.m_ptx->GetId(),
1058 std::move(results));
1062 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1063 valid_txids.push_back(ws.m_ptx->GetId());
1078 const auto m_total_size = std::accumulate(
1079 workspaces.cbegin(), workspaces.cend(), int64_t{0},
1080 [](int64_t
sum,
auto &ws) { return sum + ws.m_ptx->GetTotalSize(); });
1081 const auto m_total_vsize =
1082 std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1083 [](int64_t
sum,
auto &ws) { return sum + ws.m_vsize; });
1084 const auto m_total_modified_fees = std::accumulate(
1085 workspaces.cbegin(), workspaces.cend(),
Amount::zero(),
1086 [](
Amount sum,
auto &ws) { return sum + ws.m_modified_fees; });
1087 const CFeeRate package_feerate(m_total_modified_fees, m_total_vsize);
1088 std::vector<TxId> all_package_txids;
1089 all_package_txids.reserve(workspaces.size());
1090 std::transform(workspaces.cbegin(), workspaces.cend(),
1091 std::back_inserter(all_package_txids),
1092 [](
const auto &ws) { return ws.m_ptx->GetId(); });
1094 if (args.m_package_feerates &&
1095 !CheckFeeRate(m_total_size, m_total_vsize, m_total_modified_fees,
1096 placeholder_state)) {
1098 "transaction failed");
1100 package_state, {{workspaces.back().m_ptx->GetId(),
1103 CFeeRate(m_total_modified_fees, m_total_vsize),
1104 all_package_txids)}});
1107 for (Workspace &ws : workspaces) {
1108 ws.m_package_feerate = package_feerate;
1109 const TxId &ws_txid = ws.m_ptx->GetId();
1110 if (args.m_test_accept &&
1111 std::find(valid_txids.begin(), valid_txids.end(), ws_txid) !=
1112 valid_txids.end()) {
1113 const auto effective_feerate =
1114 args.m_package_feerates
1115 ? ws.m_package_feerate
1117 static_cast<uint32_t
>(ws.m_vsize)};
1118 const auto effective_feerate_txids =
1119 args.m_package_feerates ? all_package_txids
1120 : std::vector<TxId>{ws.m_ptx->GetId()};
1124 results.emplace(ws_txid,
1126 ws.m_vsize, ws.m_base_fees, effective_feerate,
1127 effective_feerate_txids));
1131 if (args.m_test_accept) {
1135 if (!SubmitPackage(args, workspaces, package_state, results)) {
1144MemPoolAccept::AcceptSubPackage(
const std::vector<CTransactionRef> &subpackage,
1150 if (subpackage.size() > 1) {
1151 return AcceptMultipleTransactions(subpackage, args);
1153 const auto &tx = subpackage.front();
1154 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(args);
1155 const auto single_res = AcceptSingleTransaction(tx, single_args);
1157 if (single_res.m_result_type !=
1160 "transaction failed");
1163 {{tx->GetId(), single_res}});
1193 for (
const auto &outpoint : m_viewmempool.GetNonBaseCoins()) {
1197 m_view.Uncache(outpoint);
1200 m_viewmempool.Reset();
1216 if (!
CheckPackage(package, package_state_quit_early)) {
1225 "package-not-child-with-parents");
1230 assert(package.size() > 1);
1233 const auto &child = package.back();
1234 std::unordered_set<TxId, SaltedTxIdHasher> unconfirmed_parent_txids;
1236 package.cbegin(), package.cend() - 1,
1237 std::inserter(unconfirmed_parent_txids, unconfirmed_parent_txids.end()),
1238 [](
const auto &tx) { return tx->GetId(); });
1247 const CCoinsViewCache &coins_tip_cache = m_active_chainstate.CoinsTip();
1248 for (
const auto &input : child->vin) {
1250 args.m_coins_to_uncache.push_back(input.prevout);
1257 m_view.
SetBackend(m_active_chainstate.CoinsTip());
1258 const auto package_or_confirmed = [
this, &unconfirmed_parent_txids](
1259 const auto &input) {
1260 return unconfirmed_parent_txids.count(input.prevout.GetTxId()) > 0 ||
1261 m_view.HaveCoin(input.prevout);
1263 if (!std::all_of(child->vin.cbegin(), child->vin.cend(),
1264 package_or_confirmed)) {
1265 package_state_quit_early.
Invalid(
1267 "package-not-child-with-unconfirmed-parents");
1273 m_view.SetBackend(m_dummy);
1279 std::map<TxId, MempoolAcceptResult> results_final;
1284 std::map<TxId, MempoolAcceptResult> individual_results_nonfinal;
1285 bool quit_early{
false};
1286 std::vector<CTransactionRef> txns_package_eval;
1287 for (
const auto &tx : package) {
1288 const auto &txid = tx->GetId();
1291 if (m_pool.exists(txid)) {
1305 auto iter = m_pool.GetIter(txid);
1306 assert(iter != std::nullopt);
1308 (*iter.value())->GetTxSize(),
1309 (*iter.value())->GetFee()));
1313 const auto single_package_res = AcceptSubPackage({tx}, args);
1314 const auto &single_res = single_package_res.m_tx_results.at(txid);
1315 if (single_res.m_result_type ==
1320 assert(m_pool.exists(txid));
1321 results_final.emplace(txid, single_res);
1322 }
else if (single_res.m_state.GetResult() !=
1324 single_res.m_state.GetResult() !=
1339 package_state_quit_early.
Invalid(
1341 individual_results_nonfinal.emplace(txid, single_res);
1343 individual_results_nonfinal.emplace(txid, single_res);
1344 txns_package_eval.push_back(tx);
1349 auto multi_submission_result =
1350 quit_early || txns_package_eval.empty()
1352 : AcceptSubPackage(txns_package_eval, args);
1354 multi_submission_result.m_state;
1359 m_pool.LimitSize(m_active_chainstate.CoinsTip());
1361 for (
const auto &tx : package) {
1362 const auto &txid = tx->GetId();
1363 if (multi_submission_result.m_tx_results.count(txid) > 0) {
1366 Assume(results_final.count(txid) == 0);
1370 const auto &txresult =
1371 multi_submission_result.m_tx_results.at(txid);
1372 if (txresult.m_result_type ==
1374 !m_pool.exists(txid)) {
1376 "transaction failed");
1380 results_final.emplace(
1383 results_final.emplace(txid, txresult);
1385 }
else if (
const auto final_it{results_final.find(txid)};
1386 final_it != results_final.end()) {
1389 Assume(final_it->second.m_result_type !=
1391 Assume(individual_results_nonfinal.count(txid) == 0);
1392 if (!m_pool.exists(tx->GetId())) {
1394 "transaction failed");
1399 results_final.erase(txid);
1400 results_final.emplace(
1403 }
else if (
const auto non_final_it{
1404 individual_results_nonfinal.find(txid)};
1405 non_final_it != individual_results_nonfinal.end()) {
1406 Assume(non_final_it->second.m_result_type ==
1409 results_final.emplace(txid, non_final_it->second);
1412 Assume(results_final.size() == package.size());
1414 std::move(results_final));
1420 int64_t accept_time,
bool bypass_limits,
1422 unsigned int heightOverride) {
1427 std::vector<COutPoint> coins_to_uncache;
1428 auto args = MemPoolAccept::ATMPArgs::SingleAccept(
1430 coins_to_uncache, test_accept, heightOverride);
1432 .AcceptSingleTransaction(tx, args);
1440 for (
const COutPoint &outpoint : coins_to_uncache) {
1457 assert(!package.empty());
1458 assert(std::all_of(package.cbegin(), package.cend(),
1459 [](
const auto &tx) { return tx != nullptr; }));
1463 std::vector<COutPoint> coins_to_uncache;
1467 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(
1468 config,
GetTime(), coins_to_uncache);
1469 return MemPoolAccept(pool, active_chainstate)
1470 .AcceptMultipleTransactions(package, args);
1472 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(
1473 config,
GetTime(), coins_to_uncache);
1474 return MemPoolAccept(pool, active_chainstate)
1475 .AcceptPackage(package, args);
1481 if (test_accept || result.m_state.IsInvalid()) {
1482 for (
const COutPoint &hashTx : coins_to_uncache) {
1495 if (halvings >= 64) {
1506 : m_dbview{
std::move(db_params),
std::move(options)},
1507 m_catcherview(&m_dbview) {}
1509void CoinsViews::InitCache() {
1511 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1516 std::optional<BlockHash> from_snapshot_blockhash)
1517 : m_mempool(mempool), m_blockman(blockman), m_chainman(chainman),
1518 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1524 if (!m_cached_snapshot_base) {
1525 m_cached_snapshot_base =
Assert(
1528 return m_cached_snapshot_base;
1532 bool should_wipe, std::string leveldb_name) {
1539 .cache_bytes = cache_size_bytes,
1540 .memory_only = in_memory,
1541 .wipe_data = should_wipe,
1547void Chainstate::InitCoinsCache(
size_t cache_size_bytes) {
1582 LogPrintf(
"Leaving InitialBlockDownload (latching to false)\n");
1609 std::string warning =
1610 std::string(
"'Warning: Large-work fork detected, forking after "
1617 LogPrintf(
"%s: Warning: Large fork found\n forking the "
1618 "chain at height %d (%s)\n lasting to height %d "
1619 "(%s).\nChain state database corruption likely.\n",
1626 LogPrintf(
"%s: Warning: Found invalid chain at least ~6 blocks "
1627 "longer than our best chain.\nChain state database "
1628 "corruption likely.\n",
1683 m_avalancheFinalizedBlockIndex = pindexNew->
pprev;
1686 LogPrintf(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n",
1693 LogPrintf(
"%s: current best=%s height=%d log2_work=%f date=%s\n",
1705 pindex->nStatus = pindex->nStatus.withFailed();
1715 if (tx.IsCoinBase()) {
1719 txundo.
vprevout.reserve(tx.vin.size());
1720 for (
const CTxIn &txin : tx.vin) {
1734 const CScript &scriptSig =
ptxTo->vin[
nIn].scriptSig;
1756 bool sigCacheStore,
bool scriptCacheStore,
1760 std::vector<CScriptCheck> *pvChecks) {
1762 assert(!tx.IsCoinBase());
1765 pvChecks->reserve(tx.vin.size());
1775 (pBlockLimitSigChecks &&
1778 "too-many-sigchecks");
1783 int nSigChecksTotal = 0;
1785 for (
size_t i = 0; i < tx.vin.size(); i++) {
1786 const COutPoint &prevout = tx.vin[i].prevout;
1798 &txLimitSigChecks, pBlockLimitSigChecks);
1802 pvChecks->push_back(std::move(check));
1812 uint32_t mandatoryFlags =
1813 flags & ~STANDARD_NOT_MANDATORY_VERIFY_FLAGS;
1814 if (
flags != mandatoryFlags) {
1821 sigCacheStore, txdata);
1825 strprintf(
"non-mandatory-script-verify-flag (%s)",
1841 strprintf(
"mandatory-script-verify-flag-failed (%s)",
1848 nSigChecksOut = nSigChecksTotal;
1850 if (scriptCacheStore && !pvChecks) {
1862 return state.
Error(strMessage);
1867 const COutPoint &out) {
1875 if (undo.GetHeight() == 0) {
1898 view.
AddCoin(out, std::move(undo), !fClean);
1913 error(
"DisconnectBlock(): failure reading undo data");
1917 return ApplyBlockUndo(std::move(blockUndo), block, pindex, view);
1925 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
1926 error(
"DisconnectBlock(): block and undo data inconsistent");
1931 for (
size_t i = 1; i < block.
vtx.size(); i++) {
1932 const CTransaction &tx = *(block.
vtx[i]);
1934 if (txundo.
vprevout.size() != tx.vin.size()) {
1935 error(
"DisconnectBlock(): transaction and undo data inconsistent");
1939 for (
size_t j = 0; j < tx.vin.size(); j++) {
1940 const COutPoint &out = tx.vin[j].prevout;
1952 for (
const auto &ptx : block.
vtx) {
1953 const CTransaction &tx = *ptx;
1954 const TxId &txid = tx.GetId();
1955 const bool is_coinbase = tx.IsCoinBase();
1959 for (
size_t o = 0; o < tx.vout.size(); o++) {
1960 if (tx.vout[o].scriptPubKey.IsUnspendable()) {
1964 COutPoint out(txid, o);
1966 bool is_spent = view.
SpendCoin(out, &coin);
1967 if (!is_spent || tx.vout[o] != coin.
GetTxOut() ||
2103 if (!
CheckBlock(block, state, consensusParams,
2110 return AbortNode(state,
"Corrupt block found indicating potential "
2111 "hardware failure; shutting down");
2113 return error(
"%s: Consensus::CheckBlock: %s", __func__,
2134 bool fScriptChecks =
true;
2143 BlockMap::const_iterator it{
2146 if (it->second.GetAncestor(pindex->
nHeight) == pindex &&
2174 consensusParams) <= 60 * 60 * 24 * 7 * 2);
2196 bool fEnforceBIP30 = !((pindex->
nHeight == 91842 &&
2198 uint256S(
"0x00000000000a4d0a398161ffc163c503763"
2199 "b1f4360639393e0e4c8e300e0caec")) ||
2202 uint256S(
"0x00000000000743f190a18c5577a3c2d2a1f"
2203 "610ae9601ac046a38084ccb7cd721")));
2234 static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2271 (!pindexBIP34height ||
2277 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2278 for (
const auto &tx : block.
vtx) {
2279 for (
size_t o = 0; o < tx->vout.size(); o++) {
2280 if (view.
HaveCoin(COutPoint(tx->GetId(), o))) {
2281 LogPrintf(
"ERROR: ConnectBlock(): tried to overwrite "
2291 int nLockTimeFlags = 0;
2305 std::vector<int> prevheights;
2317 std::vector<TxSigCheckLimiter> nSigChecksTxLimiters;
2318 nSigChecksTxLimiters.resize(block.
vtx.size() - 1);
2321 blockundo.
vtxundo.resize(block.
vtx.size() - 1);
2328 for (
const auto &ptx : block.
vtx) {
2331 }
catch (
const std::logic_error &e) {
2340 LogPrintf(
"ERROR: ConnectBlock(): tried to overwrite transaction\n");
2349 for (
const auto &ptx : block.
vtx) {
2350 const CTransaction &tx = *ptx;
2351 const bool isCoinBase = tx.IsCoinBase();
2352 nInputs += tx.vin.size();
2366 return error(
"%s: Consensus::CheckTxInputs: %s, %s", __func__,
2367 tx.GetId().ToString(), state.
ToString());
2373 LogPrintf(
"ERROR: %s: accumulated fee in the block out of range.\n",
2376 "bad-txns-accumulated-fee-outofrange");
2387 prevheights.resize(tx.vin.size());
2388 for (
size_t j = 0; j < tx.vin.size(); j++) {
2392 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2393 LogPrintf(
"ERROR: %s: contains a non-BIP68-final transaction\n",
2396 "bad-txns-nonfinal");
2401 bool fCacheResults = fJustCheck;
2404 if (!fEnforceSigCheck) {
2411 std::vector<CScriptCheck> vChecks;
2413 if (fScriptChecks &&
2416 nSigChecksRet, nSigChecksTxLimiters[txIndex],
2417 &nSigChecksBlockLimiter, &vChecks)) {
2424 "ConnectBlock(): CheckInputScripts on %s failed with %s",
2425 tx.GetId().ToString(), state.
ToString());
2428 control.
Add(std::move(vChecks));
2442 " - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) "
2443 "[%.2fs (%.2fms/blk)]\n",
2444 (
unsigned)block.
vtx.size(),
MILLI * (nTime3 - nTime2),
2445 MILLI * (nTime3 - nTime2) / block.
vtx.size(),
2446 nInputs <= 1 ? 0 :
MILLI * (nTime3 - nTime2) / (nInputs - 1),
2449 const Amount blockReward =
2451 if (block.
vtx[0]->GetValueOut() > blockReward) {
2452 LogPrintf(
"ERROR: ConnectBlock(): coinbase pays too much (actual=%d vs "
2454 block.
vtx[0]->GetValueOut(), blockReward);
2463 if (!control.
Wait()) {
2465 "blk-bad-inputs",
"parallel script check failed");
2472 " - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n",
2473 nInputs - 1,
MILLI * (nTime4 - nTime2),
2474 nInputs <= 1 ? 0 :
MILLI * (nTime4 - nTime2) / (nInputs - 1),
2481 if (!
m_blockman.WriteUndoDataForBlock(blockundo, state, *pindex)) {
2499 TRACE6(validation, block_connected, block_hash.data(), pindex->
nHeight,
2500 block.
vtx.size(), nInputs, nSigChecksRet,
2502 nTime5 - nTimeStart);
2515Chainstate::GetCoinsCacheSizeState(
size_t max_coins_cache_size_bytes,
2516 size_t max_mempool_size_bytes) {
2520 int64_t nTotalSpace =
2521 max_coins_cache_size_bytes +
2522 std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2525 static constexpr int64_t MAX_BLOCK_COINSDB_USAGE_BYTES =
2527 int64_t large_threshold = std::max(
2528 (9 * nTotalSpace) / 10, nTotalSpace - MAX_BLOCK_COINSDB_USAGE_BYTES);
2530 if (cacheSize > nTotalSpace) {
2531 LogPrintf(
"Cache size (%s) exceeds total space (%s)\n", cacheSize,
2534 }
else if (cacheSize > large_threshold) {
2544 std::set<int> setFilesToPrune;
2545 bool full_flush_completed =
false;
2552 bool fFlushForPrune =
false;
2553 bool fDoFullFlush =
false;
2565 std::optional<std::string> limiting_lock;
2567 for (
const auto &prune_lock :
m_blockman.m_prune_locks) {
2568 if (prune_lock.second.height_first ==
2569 std::numeric_limits<int>::max()) {
2574 const int lock_height{prune_lock.second.height_first -
2576 last_prune = std::max(1, std::min(last_prune, lock_height));
2577 if (last_prune == lock_height) {
2578 limiting_lock = prune_lock.first;
2582 if (limiting_lock) {
2584 limiting_lock.value(), last_prune);
2587 if (nManualPruneHeight > 0) {
2592 std::min(last_prune, nManualPruneHeight),
2603 if (!setFilesToPrune.empty()) {
2604 fFlushForPrune =
true;
2607 "prunedblockfiles",
true);
2612 const auto nNow = GetTime<std::chrono::microseconds>();
2637 fCacheCritical || fPeriodicFlush || fFlushForPrune;
2639 if (fDoFullFlush || fPeriodicWrite) {
2642 return AbortNode(state,
"Disk space is too low!",
2643 _(
"Disk space is too low!"));
2662 state,
"Failed to write to block index database");
2667 if (fFlushForPrune) {
2677 if (fDoFullFlush && !
CoinsTip().GetBestBlock().IsNull()) {
2679 strprintf(
"write coins cache to disk (%d coins, %.2fkB)",
2680 coins_count, coins_mem_usage / 1000),
2690 48 * 2 * 2 *
CoinsTip().GetCacheSize())) {
2691 return AbortNode(state,
"Disk space is too low!",
2692 _(
"Disk space is too low!"));
2698 return AbortNode(state,
"Failed to write to coin database");
2701 full_flush_completed =
true;
2706 GetTimeMicros() - nNow.count(), uint32_t(mode), coins_count,
2707 uint64_t(coins_mem_usage), fFlushForPrune);
2710 if (full_flush_completed) {
2714 }
catch (
const std::runtime_error &e) {
2715 return AbortNode(state, std::string(
"System error while flushing: ") +
2724 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
2733 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
2740 const std::string &func_name,
2741 const std::string &
prefix)
2744 LogPrintf(
"%s%s: new best=%s height=%d version=0x%08x log2_work=%f tx=%ld "
2745 "date='%s' progress=%f cache=%.1fMiB(%utxo)\n",
2751 coins_tip.DynamicMemoryUsage() * (1.0 / (1 << 20)),
2752 coins_tip.GetCacheSize());
2755void Chainstate::UpdateTip(
const CBlockIndex *pindexNew) {
2757 const auto &coins_tip =
CoinsTip();
2766 constexpr int BACKGROUND_LOG_INTERVAL = 2000;
2767 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2769 "[background validation] ");
2785 UpdateTipLog(coins_tip, pindexNew, params, __func__,
"");
2812 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2815 return error(
"DisconnectTip(): Failed to read block");
2823 if (DisconnectBlock(block, pindexDelete, view) !=
2825 return error(
"DisconnectTip(): DisconnectBlock %s failed",
2829 bool flushed = view.
Flush();
2839 const int max_height_first{pindexDelete->
nHeight - 1};
2840 for (
auto &prune_lock :
m_blockman.m_prune_locks) {
2841 if (prune_lock.second.height_first <= max_height_first) {
2845 prune_lock.second.height_first = max_height_first;
2847 prune_lock.first, max_height_first);
2860 if (pindexDelete->
pprev !=
nullptr &&
2864 "Disconnecting mempool due to rewind of upgrade block\n");
2865 if (disconnectpool) {
2871 if (disconnectpool) {
2878 UpdateTip(pindexDelete->
pprev);
2898 const std::shared_ptr<const CBlock> &pblock,
2911 std::shared_ptr<const CBlock> pthisBlock;
2913 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
2915 return AbortNode(state,
"Failed to read block");
2917 pthisBlock = pblockNew;
2919 pthisBlock = pblock;
2922 const CBlock &blockConnecting = *pthisBlock;
2933 bool rv =
ConnectBlock(blockConnecting, state, pindexNew, view,
2942 return error(
"%s: ConnectBlock %s failed, %s", __func__,
2963 const Amount blockReward =
2967 std::vector<std::unique_ptr<ParkingPolicy>> parkingPolicies;
2968 parkingPolicies.emplace_back(std::make_unique<MinerFundPolicy>(
2969 consensusParams, *pindexNew, blockConnecting, blockReward));
2985 parkingPolicies.emplace_back(
2986 std::make_unique<RTTPolicy>(consensusParams,
2991 parkingPolicies.emplace_back(
2992 std::make_unique<StakingRewardsPolicy>(
2993 *
avalanche, consensusParams, *pindexNew,
2994 blockConnecting, blockReward));
2997 parkingPolicies.emplace_back(
2998 std::make_unique<PreConsensusPolicy>(
2999 *pindexNew, blockConnecting,
m_mempool));
3004 if (std::find_if_not(parkingPolicies.begin(), parkingPolicies.end(),
3005 [&](
const auto &policy) {
3006 bool ret = (*policy)(blockPolicyState);
3009 "Park block because it "
3010 "violated a block policy: %s\n",
3011 blockPolicyState.ToString());
3014 }) != parkingPolicies.end()) {
3015 pindexNew->nStatus = pindexNew->nStatus.withParked();
3025 " - Connect total: %.2fms [%.2fs (%.2fms/blk)]\n",
3028 bool flushed = view.
Flush();
3046 " - Writing chainstate: %.2fms [%.2fs (%.2fms/blk)]\n",
3052 disconnectpool.removeForBlock(blockConnecting.vtx, *m_mempool);
3057 if (pindexNew->
pprev !=
nullptr &&
3062 "Disconnecting mempool due to acceptance of upgrade block\n");
3063 disconnectpool.importMempool(*m_mempool);
3068 m_chain.SetTip(*pindexNew);
3069 UpdateTip(pindexNew);
3075 " - Connect postprocess: %.2fms [%.2fs (%.2fms/blk)]\n",
3085 if (
this != &m_chainman.ActiveChainstate()) {
3089 m_chainman.MaybeCompleteSnapshotValidation();
3101 std::vector<const CBlockIndex *> &blocksToReconcile,
bool fAutoUnpark) {
3108 std::set<CBlockIndex *, CBlockIndexWorkComparator>::reverse_iterator
3120 if (m_avalancheFinalizedBlockIndex &&
3122 LogPrintf(
"Park block %s because it forks prior to the "
3123 "avalanche finalized chaintip.\n",
3125 pindexNew->nStatus = pindexNew->nStatus.withParked();
3136 bool hasValidAncestor =
true;
3137 while (hasValidAncestor && pindexTest && pindexTest != pindexFork) {
3142 bool fParkedChain = pindexTest->nStatus.isOnParkedChain();
3143 if (fAutoUnpark && fParkedChain) {
3149 if (!pindexTip || !pindexFork) {
3163 pindexExtraPow = pindexExtraPow->
pprev;
3168 requiredWork += (deltaWork >> 1);
3179 LogPrintf(
"Unpark chain up to block %s as it has "
3180 "accumulated enough PoW.\n",
3182 fParkedChain =
false;
3191 bool fInvalidChain = pindexTest->nStatus.isInvalid();
3192 bool fMissingData = !pindexTest->nStatus.hasData();
3193 if (!(fInvalidChain || fParkedChain || fMissingData)) {
3196 pindexTest = pindexTest->
pprev;
3202 hasValidAncestor =
false;
3205 if (fInvalidChain && (
m_chainman.m_best_invalid ==
nullptr ||
3211 if (fParkedChain && (
m_chainman.m_best_parked ==
nullptr ||
3217 LogPrintf(
"Considered switching to better tip %s but that chain "
3218 "contains a%s%s%s block.\n",
3220 fInvalidChain ?
"n invalid" :
"",
3221 fParkedChain ?
" parked" :
"",
3222 fMissingData ?
" missing-data" :
"");
3226 while (pindexTest != pindexFailed) {
3227 if (fInvalidChain || fParkedChain) {
3228 pindexFailed->nStatus =
3229 pindexFailed->nStatus.withFailedParent(fInvalidChain)
3230 .withParkedParent(fParkedChain);
3231 }
else if (fMissingData) {
3237 std::make_pair(pindexFailed->
pprev, pindexFailed));
3240 pindexFailed = pindexFailed->
pprev;
3243 if (fInvalidChain || fParkedChain) {
3250 blocksToReconcile.push_back(pindexNew);
3253 if (hasValidAncestor) {
3286 const std::shared_ptr<const CBlock> &pblock,
bool &fInvalidFound,
3297 bool fBlocksDisconnected =
false;
3300 if (!fBlocksDisconnected) {
3319 "Failed to disconnect block; see debug.log for details");
3323 fBlocksDisconnected =
true;
3327 std::vector<CBlockIndex *> vpindexToConnect;
3328 bool fContinue =
true;
3333 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3334 vpindexToConnect.clear();
3335 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3338 vpindexToConnect.push_back(pindexIter);
3339 pindexIter = pindexIter->
pprev;
3347 if (!
ConnectTip(state, blockPolicyState, pindexConnect,
3348 pindexConnect == pindexMostWork
3350 : std::shared_ptr<const CBlock>(),
3359 fInvalidFound =
true;
3380 if (!pindexOldTip ||
3392 if (fBlocksDisconnected || !disconnectpool.
isEmpty()) {
3398 "Updating mempool due to reorganization or "
3399 "rules upgrade/downgrade\n");
3407 if (fInvalidFound) {
3427 bool fNotify =
false;
3428 bool fInitialBlockDownload =
false;
3433 pindexHeader = chainstate.m_chainman.m_best_header;
3435 if (pindexHeader != pindexHeaderOld) {
3437 fInitialBlockDownload = chainstate.IsInitialBlockDownload();
3438 pindexHeaderOld = pindexHeader;
3444 chainstate.m_chainman.GetNotifications().headerTip(
3460 std::shared_ptr<const CBlock> pblock,
3481 LogPrintf(
"m_disabled is set - this chainstate should not be in "
3482 "operation. Please report this as a bug. %s\n",
3499 std::vector<const CBlockIndex *> blocksToReconcile;
3500 bool blocks_connected =
false;
3502 const bool fAutoUnpark =
3516 if (pindexMostWork ==
nullptr) {
3522 if (pindexMostWork ==
nullptr ||
3527 bool fInvalidFound =
false;
3528 std::shared_ptr<const CBlock> nullBlockPtr;
3530 state, pindexMostWork,
3531 pblock && pblock->GetHash() ==
3539 blocks_connected =
true;
3541 if (fInvalidFound ||
3542 (pindexMostWork && pindexMostWork->nStatus.isParked())) {
3544 pindexMostWork =
nullptr;
3569 if (blocks_connected) {
3576 if (pindexFork != pindexNewTip) {
3593 return m_avalancheFinalizedBlockIndex);
3594 for (
const CBlockIndex *pindex : blocksToReconcile) {
3600 if (blocks_connected) {
3602 while (pindexTest && pindexTest != pfinalized) {
3607 avalanche->computeStakingReward(pindexTest);
3608 pindexTest = pindexTest->
pprev;
3614 if (!blocks_connected) {
3618 if (nStopAtHeight && pindexNewTip &&
3619 pindexNewTip->
nHeight >= nStopAtHeight) {
3637 }
while (pindexNewTip != pindexMostWork);
3668 std::numeric_limits<int32_t>::min()) {
3690template <
typename Func>
struct Defer {
3692 Defer(
Func &&f) : func(
std::move(f)) {}
3693 ~Defer() { func(); }
3706 bool pindex_was_in_chain =
false;
3707 int disconnected = 0;
3723 std::multimap<const arith_uint256, CBlockIndex *> candidate_blocks_by_work;
3727 for (
auto &entry :
m_blockman.m_block_index) {
3738 candidate_blocks_by_work.insert(
3739 std::make_pair(candidate->
nChainWork, candidate));
3751 constexpr int maxDisconnectPoolBlocks = 10;
3803 pindex_was_in_chain =
true;
3812 if (optDisconnectPool && disconnected > maxDisconnectPoolBlocks) {
3817 optDisconnectPool =
nullptr;
3833 invalid_walk_tip->nStatus =
3834 invalidate ? invalid_walk_tip->nStatus.withFailed()
3835 : invalid_walk_tip->nStatus.withParked();
3840 if (invalid_walk_tip == to_mark_failed_or_parked->
pprev &&
3841 (invalidate ? to_mark_failed_or_parked->nStatus.hasFailed()
3842 : to_mark_failed_or_parked->nStatus.isParked())) {
3846 to_mark_failed_or_parked->nStatus =
3848 ? to_mark_failed_or_parked->nStatus.withFailed(
false)
3850 : to_mark_failed_or_parked->nStatus.withParked(
false)
3851 .withParkedParent());
3857 auto candidate_it = candidate_blocks_by_work.lower_bound(
3859 while (candidate_it != candidate_blocks_by_work.end()) {
3861 invalid_walk_tip->
pprev)) {
3863 candidate_it = candidate_blocks_by_work.erase(candidate_it);
3872 to_mark_failed_or_parked = invalid_walk_tip;
3888 to_mark_failed_or_parked->nStatus =
3889 invalidate ? to_mark_failed_or_parked->nStatus.withFailed()
3890 : to_mark_failed_or_parked->nStatus.withParked();
3903 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
3905 block_index.HaveTxsDownloaded() &&
3918 if (pindex_was_in_chain) {
3921 *to_mark_failed_or_parked->
pprev);
3945template <
typename F>
3949 if (pindex->nStatus != newStatus &&
3952 pindex->nStatus = newStatus;
3968template <
typename F,
typename C,
typename AC>
3970 F f, C fChild, AC fAncestorWasChanged) {
3976 for (
auto pindexAncestor = pindex; pindexAncestor !=
nullptr;
3977 pindexAncestor = pindexAncestor->
pprev) {
3979 pindexDeepestChanged = pindexAncestor;
3985 pindexDeepestChanged) {
3987 pindexReset =
nullptr;
3991 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
3994 fAncestorWasChanged);
4004 return status.withClearedFailureFlags();
4007 return status.withClearedFailureFlags();
4010 return status.withFailedParent(false);
4024 if (is_active_chainstate) {
4028 }
else if (!m_disabled) {
4034 if (snapshot_base->GetAncestor(pindex->
nHeight) == pindex) {
4046 return status.withClearedParkedFlags();
4049 return fClearChildren ? status.withClearedParkedFlags()
4050 : status.withParkedParent(false);
4053 return status.withParkedParent(false);
4073 "The block to mark finalized by avalanche is not on the "
4074 "active chain: %s\n",
4087 m_avalancheFinalizedBlockIndex = pindex;
4097 m_avalancheFinalizedBlockIndex =
nullptr;
4102 return pindex && m_avalancheFinalizedBlockIndex &&
4114 pindexNew->
nTx = block.
vtx.size();
4116 pindexNew->nFile = pos.
nFile;
4117 pindexNew->nDataPos = pos.
nPos;
4118 pindexNew->nUndoPos = 0;
4119 pindexNew->nStatus = pindexNew->nStatus.withData();
4126 std::deque<CBlockIndex *> queue;
4127 queue.push_back(pindexNew);
4131 while (!queue.empty()) {
4144 c->TryAddBlockIndexCandidate(pindex);
4147 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
4148 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
4150 while (range.first != range.second) {
4151 std::multimap<CBlockIndex *, CBlockIndex *>::iterator it =
4153 queue.push_back(it->second);
4158 }
else if (pindexNew->
pprev &&
4161 std::make_pair(pindexNew->
pprev, pindexNew));
4181 "high-hash",
"proof of work failed");
4207 "bad-txnmrklroot",
"hashMerkleRoot mismatch");
4215 "bad-txns-duplicate",
"duplicate transaction");
4224 if (block.
vtx.empty()) {
4226 "bad-cb-missing",
"first tx is not coinbase");
4235 "bad-blk-length",
"size limits failed");
4239 if (currentBlockSize > nMaxBlockSize) {
4241 "bad-blk-length",
"size limits failed");
4249 strprintf(
"Coinbase check failed (txid %s) %s",
4250 block.
vtx[0]->GetId().ToString(),
4256 for (
size_t i = 1; i < block.
vtx.size(); i++) {
4257 auto *tx = block.
vtx[i].get();
4262 strprintf(
"Transaction check failed (txid %s) %s",
4277 return std::all_of(headers.cbegin(), headers.cend(),
4278 [&](
const auto &header) {
4279 return CheckProofOfWork(
4280 header.GetHash(), header.nBits, consensusParams);
4307 const std::optional<CCheckpointData> &test_checkpoints = std::nullopt)
4310 assert(pindexPrev !=
nullptr);
4311 const int nHeight = pindexPrev->nHeight + 1;
4317 LogPrintf(
"bad bits after height: %d\n", pindexPrev->nHeight);
4319 "bad-diffbits",
"incorrect proof of work");
4323 if (chainman.m_options.checkpoints_enabled) {
4325 test_checkpoints ? test_checkpoints.value() : params.
Checkpoints();
4331 "ERROR: %s: rejected by checkpoint lock-in at %d\n",
4334 "checkpoint mismatch");
4342 blockman.GetLastCheckpoint(checkpoints);
4343 if (pcheckpoint && nHeight < pcheckpoint->
nHeight) {
4345 "ERROR: %s: forked chain older than last checkpoint "
4349 "bad-fork-prior-to-checkpoint");
4354 if (block.
GetBlockTime() <= pindexPrev->GetMedianTimePast()) {
4356 "time-too-old",
"block's timestamp is too early");
4360 if (block.
Time() > now + std::chrono::seconds{MAX_FUTURE_BLOCK_TIME}) {
4363 "block timestamp too far in the future");
4397 const int nBlockHeight = active_chain_tip.
nHeight + 1;
4422 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4425 bool enforce_locktime_median_time_past{
false};
4428 assert(pindexPrev !=
nullptr);
4429 enforce_locktime_median_time_past =
true;
4432 const int64_t nMedianTimePast =
4435 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past
4440 const bool fIsMagneticAnomalyEnabled =
4447 const CTransaction *prevTx =
nullptr;
4448 for (
const auto &ptx : block.
vtx) {
4449 const CTransaction &tx = *ptx;
4450 if (fIsMagneticAnomalyEnabled) {
4451 if (prevTx && (tx.GetId() <= prevTx->GetId())) {
4452 if (tx.GetId() == prevTx->GetId()) {
4456 tx.GetId().ToString()));
4461 strprintf(
"Transaction order is invalid (%s < %s)",
4462 tx.GetId().ToString(),
4463 prevTx->GetId().ToString()));
4466 if (prevTx || !tx.IsCoinBase()) {
4484 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4486 block.
vtx[0]->vin[0].scriptSig.begin())) {
4489 "block height mismatch in coinbase");
4504 const std::optional<CCheckpointData> &test_checkpoints) {
4511 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4513 if (miSelf !=
m_blockman.m_block_index.end()) {
4520 if (pindex->nStatus.isInvalid()) {
4533 "%s: Consensus::CheckBlockHeader: %s, %s\n", __func__,
4539 BlockMap::iterator mi{
4543 "header %s has prev block not found: %s\n",
4546 "prev-blk-not-found");
4551 if (pindexPrev->nStatus.isInvalid()) {
4553 "header %s has prev block invalid: %s\n", hash.
ToString(),
4563 "%s: Consensus::ContextualCheckBlockHeader: %s, %s\n",
4594 if (pindexPrev->
GetAncestor(failedit->nHeight) == failedit) {
4595 assert(failedit->nStatus.hasFailed());
4597 while (invalid_walk != failedit) {
4598 invalid_walk->nStatus =
4599 invalid_walk->nStatus.withFailedParent();
4601 invalid_walk = invalid_walk->
pprev;
4604 "header %s has prev block invalid: %s\n",
4613 if (!min_pow_checked) {
4615 "%s: not adding new block header %s, missing anti-dos "
4616 "proof-of-work validation\n",
4619 "too-little-chainwork");
4635 const auto msg =
strprintf(
"Saw new header hash=%s height=%d",
4649 const std::vector<CBlockHeader> &headers,
bool min_pow_checked,
4651 const std::optional<CCheckpointData> &test_checkpoints) {
4659 header, state, &pindex, min_pow_checked, test_checkpoints);
4676 const int64_t blocks_left{
4677 (
GetTime() - last_accepted.GetBlockTime()) /
4679 const double progress{100.0 * last_accepted.nHeight /
4680 (last_accepted.nHeight + blocks_left)};
4681 LogPrintf(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n",
4682 last_accepted.nHeight, progress);
4690 int64_t timestamp) {
4700 if (m_best_header->nChainWork >=
4706 auto now = Now<SteadyMilliseconds>();
4707 if (now < m_last_presync_update + 250ms) {
4710 m_last_presync_update = now;
4712 bool initial_download = chainstate.IsInitialBlockDownload();
4714 height, timestamp,
true);
4715 if (initial_download) {
4716 const int64_t blocks_left{(
GetTime() - timestamp) /
4718 const double progress{100.0 * height / (height + blocks_left)};
4719 LogPrintf(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n",
4727 bool *fNewBlock,
bool min_pow_checked) {
4730 const CBlock &block = *pblock;
4737 bool accepted_header{
4741 if (!accepted_header) {
4749 bool fAlreadyHave = pindex->nStatus.hasData();
4761 int64_t chainTipTimeDiff =
4767 LogPrintf(
"Chain tip timestamp-to-received-time difference: hash=%s, "
4770 LogPrintf(
"New block timestamp-to-received-time difference: hash=%s, "
4775 bool fHasMoreOrSameWork =
4783 bool fTooFarAhead{pindex->
nHeight >
4796 if (pindex->
nTx != 0) {
4801 if (!fHasMoreOrSameWork) {
4825 pindex->nStatus = pindex->nStatus.withFailed();
4829 return error(
"%s: %s (block %s)", __func__, state.
ToString(),
4842 LogPrintf(
"Park block %s as it would cause a deep reorg.\n",
4844 pindex->nStatus = pindex->nStatus.withParked();
4864 if (blockPos.IsNull()) {
4866 "%s: Failed to find position to write new block to disk",
4871 }
catch (
const std::runtime_error &e) {
4872 return AbortNode(state, std::string(
"System error: ") + e.what());
4890 const std::shared_ptr<const CBlock> &block,
bool force_processing,
4891 bool min_pow_checked,
bool *new_block,
4920 ret =
AcceptBlock(block, state, force_processing,
nullptr,
4921 new_block, min_pow_checked);
4926 return error(
"%s: AcceptBlock FAILED (%s)", __func__,
4936 return error(
"%s: ActivateBestChain failed (%s)", __func__,
4954 false, test_accept);
4970 indexDummy.
pprev = pindexPrev;
4977 adjusted_time_callback())) {
4978 return error(
"%s: Consensus::ContextualCheckBlockHeader: %s", __func__,
4983 return error(
"%s: Consensus::CheckBlock: %s", __func__,
4989 return error(
"%s: Consensus::ContextualCheckBlock: %s", __func__,
4993 if (!chainstate.
ConnectBlock(block, state, &indexDummy, viewNew,
4994 validationOptions,
nullptr,
true)) {
5004 int nManualPruneHeight) {
5007 nManualPruneHeight)) {
5008 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
5014 FopenFn mockable_fopen_function) {
5044 "Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f\n",
5075 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
5079 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
5080 LogPrintf(
"Verifying last %i blocks at level %i\n", nCheckDepth,
5086 int nGoodTransactions = 0;
5089 bool skipped_no_block_data{
false};
5090 bool skipped_l3_checks{
false};
5091 LogPrintf(
"Verification progress: 0%%\n");
5096 pindex = pindex->
pprev) {
5097 const int percentageDone = std::max(
5098 1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() -
5100 (
double)nCheckDepth *
5101 (nCheckLevel >= 4 ? 50 : 100))));
5102 if (reportDone < percentageDone / 10) {
5104 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
5105 reportDone = percentageDone / 10;
5114 !pindex->nStatus.hasData()) {
5117 LogPrintf(
"VerifyDB(): block verification stopping at height %d "
5118 "(no data). This could be due to pruning or use of an "
5119 "assumeutxo snapshot.\n",
5121 skipped_no_block_data =
true;
5130 "Verification error: ReadBlockFromDisk failed at %d, hash=%s\n",
5136 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensusParams,
5139 "Verification error: found bad block at %d, hash=%s (%s)\n",
5146 if (nCheckLevel >= 2 && pindex) {
5150 LogPrintf(
"Verification error: found bad undo data at %d, "
5163 if (nCheckLevel >= 3) {
5167 chainstate.DisconnectBlock(block, pindex, coins);
5169 LogPrintf(
"Verification error: irrecoverable inconsistency "
5170 "in block data at %d, hash=%s\n",
5176 nGoodTransactions = 0;
5177 pindexFailure = pindex;
5179 nGoodTransactions += block.
vtx.size();
5182 skipped_l3_checks =
true;
5191 if (pindexFailure) {
5192 LogPrintf(
"Verification error: coin database inconsistencies found "
5193 "(last %i blocks, %i good transactions before that)\n",
5198 if (skipped_l3_checks) {
5199 LogPrintf(
"Skipped verification of level >=3 (insufficient database "
5200 "cache size). Consider increasing -dbcache.\n");
5207 if (nCheckLevel >= 4 && !skipped_l3_checks) {
5209 const int percentageDone = std::max(
5210 1, std::min(99, 100 -
int(
double(chainstate.
m_chain.
Height() -
5212 double(nCheckDepth) * 50)));
5213 if (reportDone < percentageDone / 10) {
5215 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
5216 reportDone = percentageDone / 10;
5223 LogPrintf(
"Verification error: ReadBlockFromDisk failed at %d, "
5228 if (!chainstate.
ConnectBlock(block, state, pindex, coins,
5230 LogPrintf(
"Verification error: found unconnectable block at "
5231 "%d, hash=%s (%s)\n",
5242 LogPrintf(
"Verification: No coin database inconsistencies in last %i "
5243 "blocks (%i transactions)\n",
5244 block_count, nGoodTransactions);
5246 if (skipped_l3_checks) {
5249 if (skipped_no_block_data) {
5265 return error(
"ReplayBlock(): ReadBlockFromDisk failed at %d, hash=%s",
5275 if (tx->IsCoinBase()) {
5279 for (
const CTxIn &txin : tx->vin) {
5293 std::vector<BlockHash> hashHeads =
db.GetHeadBlocks();
5294 if (hashHeads.empty()) {
5298 if (hashHeads.size() != 2) {
5299 return error(
"ReplayBlocks(): unknown inconsistent state");
5312 if (
m_blockman.m_block_index.count(hashHeads[0]) == 0) {
5314 "ReplayBlocks(): reorganization to unknown block requested");
5317 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
5319 if (!hashHeads[1].IsNull()) {
5321 if (
m_blockman.m_block_index.count(hashHeads[1]) == 0) {
5323 "ReplayBlocks(): reorganization from unknown block requested");
5326 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
5328 assert(pindexFork !=
nullptr);
5332 while (pindexOld != pindexFork) {
5337 return error(
"RollbackBlock(): ReadBlockFromDisk() failed at "
5348 "RollbackBlock(): DisconnectBlock failed at %d, hash=%s",
5360 pindexOld = pindexOld->
pprev;
5364 int nForkHeight = pindexFork ? pindexFork->
nHeight : 0;
5371 _(
"Replaying blocks…"),
5372 (
int)((
nHeight - nForkHeight) * 100.0 /
5373 (pindexNew->
nHeight - nForkHeight)),
5388void Chainstate::ClearBlockIndexCandidates() {
5404 const uint64_t numHeaders{20};
5407 const fs::path filePathTmp = filePath +
".new";
5418 bool missingIndex{
false};
5419 for (uint64_t i = 0; i < numHeaders; i++) {
5421 LogPrintf(
"Missing block index, stopping the headers time "
5422 "dumping after %d blocks.\n",
5424 missingIndex =
true;
5431 index = index->
pprev;
5435 throw std::runtime_error(
strprintf(
"Failed to commit to file %s",
5441 fs::remove(filePathTmp);
5446 throw std::runtime_error(
strprintf(
"Rename failed from %s to %s",
5450 }
catch (
const std::exception &e) {
5451 LogPrintf(
"Failed to dump the headers time: %s.\n", e.what());
5455 LogPrintf(
"Successfully dumped the last %d headers time to %s.\n",
5471 LogPrintf(
"Failed to open header times from disk, skipping.\n");
5480 LogPrintf(
"Unsupported header times file version, skipping.\n");
5487 for (uint64_t i = 0; i < numBlocks; i++) {
5489 int64_t receiveTime;
5492 file >> receiveTime;
5496 LogPrintf(
"Missing index for block %s, stopping the headers "
5497 "time loading after %d blocks.\n",
5504 }
catch (
const std::exception &e) {
5505 LogPrintf(
"Failed to read the headers time file data on disk: %s.\n",
5523 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
5525 std::vector<CBlockIndex *> vSortedByHeight{
5527 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
5539 if (pindex == GetSnapshotBaseBlock() ||
5547 if (pindex->nStatus.isInvalid() &&
5549 pindex->
nChainWork > m_best_invalid->nChainWork)) {
5550 m_best_invalid = pindex;
5553 if (pindex->nStatus.isOnParkedChain() &&
5555 pindex->
nChainWork > m_best_parked->nChainWork)) {
5556 m_best_parked = pindex;
5560 (m_best_header ==
nullptr ||
5562 m_best_header = pindex;
5566 needs_init =
m_blockman.m_block_index.empty();
5576 LogPrintf(
"Initializing databases...\n");
5597 if (blockPos.IsNull()) {
5598 return error(
"%s: writing genesis block to disk failed", __func__);
5603 }
catch (
const std::runtime_error &e) {
5604 return error(
"%s: failed to write genesis block: %s", __func__,
5613 std::multimap<BlockHash, FlatFilePos> *blocks_with_unknown_parent,
5616 assert(!dbp == !blocks_with_unknown_parent);
5630 uint64_t nRewind = blkdat.
GetPos();
5631 while (!blkdat.
eof()) {
5641 unsigned int nSize = 0;
5646 nRewind = blkdat.
GetPos() + 1;
5648 if (memcmp(buf, params.
DiskMagic().data(),
5658 }
catch (
const std::exception &) {
5666 const uint64_t nBlockPos{blkdat.
GetPos()};
5668 dbp->
nPos = nBlockPos;
5670 blkdat.
SetLimit(nBlockPos + nSize);
5678 nRewind = nBlockPos + nSize;
5683 std::shared_ptr<CBlock> pblock{};
5692 "%s: Out of order block %s, parent %s not known\n",
5693 __func__, hash.ToString(),
5695 if (dbp && blocks_with_unknown_parent) {
5696 blocks_with_unknown_parent->emplace(
5705 if (!pindex || !pindex->nStatus.hasData()) {
5708 blkdat.
SetPos(nBlockPos);
5709 pblock = std::make_shared<CBlock>();
5711 nRewind = blkdat.
GetPos();
5714 if (
AcceptBlock(pblock, state,
true, dbp,
nullptr,
5722 pindex->
nHeight % 1000 == 0) {
5725 "Block Import: already had block %s at height %d\n",
5726 hash.ToString(), pindex->
nHeight);
5733 bool genesis_activation_failure =
false;
5734 for (
auto c :
GetAll()) {
5736 if (!c->ActivateBestChain(state,
nullptr,
avalanche)) {
5737 genesis_activation_failure =
true;
5741 if (genesis_activation_failure) {
5755 bool activation_failure =
false;
5756 for (
auto c :
GetAll()) {
5758 if (!c->ActivateBestChain(state, pblock,
avalanche)) {
5760 "failed to activate chain (%s)\n",
5762 activation_failure =
true;
5766 if (activation_failure) {
5773 if (!blocks_with_unknown_parent) {
5779 std::deque<BlockHash> queue;
5780 queue.push_back(hash);
5781 while (!queue.empty()) {
5784 auto range = blocks_with_unknown_parent->equal_range(head);
5785 while (range.first != range.second) {
5786 std::multimap<BlockHash, FlatFilePos>::iterator it =
5788 std::shared_ptr<CBlock> pblockrecursive =
5789 std::make_shared<CBlock>();
5794 "%s: Processing out of order child %s of %s\n",
5795 __func__, pblockrecursive->GetHash().ToString(),
5800 &it->second,
nullptr,
true)) {
5802 queue.push_back(pblockrecursive->GetHash());
5806 blocks_with_unknown_parent->erase(it);
5810 }
catch (
const std::exception &e) {
5831 "%s: unexpected data at file offset 0x%x - %s. "
5833 __func__, (nRewind - 1), e.what());
5836 }
catch (
const std::runtime_error &e) {
5837 AbortNode(std::string(
"System error: ") + e.what());
5840 LogPrintf(
"Loaded %i blocks from external file in %dms\n", nLoaded,
5861 std::multimap<CBlockIndex *, CBlockIndex *> forward;
5862 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
5863 forward.emplace(block_index.pprev, &block_index);
5868 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
5869 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
5870 rangeGenesis = forward.equal_range(
nullptr);
5872 rangeGenesis.first++;
5874 assert(rangeGenesis.first == rangeGenesis.second);
5894 CBlockIndex *pindexFirstNotTransactionsValid =
nullptr;
5900 CBlockIndex *pindexFirstNotScriptsValid =
nullptr;
5903 while (pindex !=
nullptr) {
5905 if (pindexFirstAssumeValid ==
nullptr &&
5906 pindex->nStatus.isAssumedValid()) {
5907 pindexFirstAssumeValid = pindex;
5909 if (pindexFirstInvalid ==
nullptr && pindex->nStatus.hasFailed()) {
5910 pindexFirstInvalid = pindex;
5912 if (pindexFirstParked ==
nullptr && pindex->nStatus.isParked()) {
5913 pindexFirstParked = pindex;
5915 if (pindexFirstMissing ==
nullptr && !pindex->nStatus.hasData()) {
5916 pindexFirstMissing = pindex;
5918 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) {
5919 pindexFirstNeverProcessed = pindex;
5921 if (pindex->
pprev !=
nullptr && pindexFirstNotTreeValid ==
nullptr &&
5923 pindexFirstNotTreeValid = pindex;
5926 if (pindexFirstNotTransactionsValid ==
nullptr &&
5928 pindexFirstNotTransactionsValid = pindex;
5930 if (pindexFirstNotChainValid ==
nullptr &&
5932 pindexFirstNotChainValid = pindex;
5934 if (pindexFirstNotScriptsValid ==
nullptr &&
5936 pindexFirstNotScriptsValid = pindex;
5941 if (pindex->
pprev ==
nullptr) {
5945 for (
auto c :
GetAll()) {
5946 if (c->m_chain.Genesis() !=
nullptr) {
5948 assert(pindex == c->m_chain.Genesis());
5965 assert(pindex->nStatus.hasData() == (pindex->
nTx > 0));
5966 if (pindexFirstAssumeValid ==
nullptr) {
5971 assert(pindexFirstMissing == pindexFirstNeverProcessed);
5973 }
else if (pindex->nStatus.hasData()) {
5978 if (pindex->nStatus.hasUndo()) {
5979 assert(pindex->nStatus.hasData());
5990 assert((pindex->nStatus.getValidity() >=
5998 assert((pindexFirstNeverProcessed ==
nullptr) ==
6000 assert((pindexFirstNotTransactionsValid ==
nullptr) ==
6012 assert(pindexFirstNotTreeValid ==
nullptr);
6015 assert(pindexFirstNotTreeValid ==
nullptr);
6019 assert(pindexFirstNotChainValid ==
nullptr);
6023 assert(pindexFirstNotScriptsValid ==
nullptr);
6025 if (pindexFirstInvalid ==
nullptr) {
6028 assert(!pindex->nStatus.isInvalid());
6030 if (pindexFirstParked ==
nullptr) {
6034 assert(!pindex->nStatus.isOnParkedChain());
6037 for (
auto c :
GetAll()) {
6038 if (c->m_chain.Tip() ==
nullptr) {
6042 pindexFirstNeverProcessed ==
nullptr) {
6043 if (pindexFirstInvalid ==
nullptr) {
6048 GetSnapshotBaseBlock()->GetAncestor(pindex->
nHeight) ==
6054 if (pindexFirstMissing ==
nullptr) {
6055 assert(pindex->nStatus.isOnParkedChain() ||
6056 c->setBlockIndexCandidates.count(pindex));
6060 if (pindex == c->m_chain.Tip()) {
6061 assert(c->setBlockIndexCandidates.count(pindex));
6073 assert(c->setBlockIndexCandidates.count(pindex) == 0);
6077 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6078 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6081 bool foundInUnlinked =
false;
6082 while (rangeUnlinked.first != rangeUnlinked.second) {
6083 assert(rangeUnlinked.first->first == pindex->
pprev);
6084 if (rangeUnlinked.first->second == pindex) {
6085 foundInUnlinked =
true;
6088 rangeUnlinked.first++;
6090 if (pindex->
pprev && pindex->nStatus.hasData() &&
6091 pindexFirstNeverProcessed !=
nullptr &&
6092 pindexFirstInvalid ==
nullptr) {
6098 if (!pindex->nStatus.hasData()) {
6100 assert(!foundInUnlinked);
6102 if (pindexFirstMissing ==
nullptr) {
6105 assert(!foundInUnlinked);
6107 if (pindex->
pprev && pindex->nStatus.hasData() &&
6108 pindexFirstNeverProcessed ==
nullptr &&
6109 pindexFirstMissing !=
nullptr) {
6115 pindexFirstAssumeValid !=
nullptr);
6125 for (
auto c :
GetAll()) {
6128 c->setBlockIndexCandidates.count(pindex) == 0) {
6129 if (pindexFirstInvalid ==
nullptr) {
6130 if (is_active || GetSnapshotBaseBlock()->GetAncestor(
6143 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6144 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6145 range = forward.equal_range(pindex);
6146 if (range.first != range.second) {
6148 pindex = range.first->second;
6158 if (pindex == pindexFirstInvalid) {
6159 pindexFirstInvalid =
nullptr;
6161 if (pindex == pindexFirstParked) {
6162 pindexFirstParked =
nullptr;
6164 if (pindex == pindexFirstMissing) {
6165 pindexFirstMissing =
nullptr;
6167 if (pindex == pindexFirstNeverProcessed) {
6168 pindexFirstNeverProcessed =
nullptr;
6170 if (pindex == pindexFirstNotTreeValid) {
6171 pindexFirstNotTreeValid =
nullptr;
6173 if (pindex == pindexFirstNotTransactionsValid) {
6174 pindexFirstNotTransactionsValid =
nullptr;
6176 if (pindex == pindexFirstNotChainValid) {
6177 pindexFirstNotChainValid =
nullptr;
6179 if (pindex == pindexFirstNotScriptsValid) {
6180 pindexFirstNotScriptsValid =
nullptr;
6182 if (pindex == pindexFirstAssumeValid) {
6183 pindexFirstAssumeValid =
nullptr;
6188 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6189 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6190 rangePar = forward.equal_range(pindexPar);
6191 while (rangePar.first->second != pindex) {
6194 assert(rangePar.first != rangePar.second);
6199 if (rangePar.first != rangePar.second) {
6201 pindex = rangePar.first->second;
6213 assert(nNodes == forward.size());
6219 return strprintf(
"Chainstate [%s] @ height %d (%s)",
6225bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size) {
6238 coinsdb_size * (1.0 / 1024 / 1024));
6240 coinstip_size * (1.0 / 1024 / 1024));
6245 if (coinstip_size > old_coinstip_size) {
6262 if (pindex ==
nullptr) {
6266 int64_t nNow = time(
nullptr);
6281 if (m_active_chainstate && m_active_chainstate->m_from_snapshot_blockhash) {
6283 return m_active_chainstate->m_from_snapshot_blockhash;
6285 return std::nullopt;
6290 std::vector<Chainstate *> out;
6293 {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
6295 out.push_back(pchainstate);
6304 assert(!m_ibd_chainstate);
6305 assert(!m_active_chainstate);
6307 m_ibd_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman, *
this);
6308 m_active_chainstate = m_ibd_chainstate.get();
6309 return *m_active_chainstate;
6315 const auto assumeutxo_found = valid_assumeutxos_map.find(height);
6317 if (assumeutxo_found != valid_assumeutxos_map.end()) {
6318 return &assumeutxo_found->second;
6332 const bool existed{fs::remove(base_blockhash_path)};
6334 LogPrintf(
"[snapshot] snapshot chainstate dir being removed "
6338 }
catch (
const fs::filesystem_error &e) {
6339 LogPrintf(
"[snapshot] failed to remove file %s: %s\n",
6346 LogPrintf(
"Removing leveldb dir at %s\n", path_str);
6350 const bool destroyed = dbwrapper::DestroyDB(path_str, {}).ok();
6353 LogPrintf(
"error: leveldb DestroyDB call failed on %s\n", path_str);
6371 LogPrintf(
"[snapshot] can't activate a snapshot-based chainstate more "
6376 int64_t current_coinsdb_cache_size{0};
6377 int64_t current_coinstip_cache_size{0};
6386 static constexpr double IBD_CACHE_PERC = 0.01;
6387 static constexpr double SNAPSHOT_CACHE_PERC = 0.99;
6400 current_coinsdb_cache_size =
6402 current_coinstip_cache_size =
6408 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
6409 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
6412 auto snapshot_chainstate =
6419 snapshot_chainstate->InitCoinsDB(
6420 static_cast<size_t>(current_coinsdb_cache_size *
6421 SNAPSHOT_CACHE_PERC),
6422 in_memory,
false,
"chainstate");
6423 snapshot_chainstate->InitCoinsCache(
static_cast<size_t>(
6424 current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
6428 coins_file, metadata);
6435 snapshot_ok =
false;
6440 this->MaybeRebalanceCaches();
6449 snapshot_chainstate.reset();
6454 strprintf(
"Failed to remove snapshot chainstate dir (%s). "
6455 "Manually remove it before restarting.\n",
6464 assert(!m_snapshot_chainstate);
6465 m_snapshot_chainstate.swap(snapshot_chainstate);
6466 const bool chaintip_loaded = m_snapshot_chainstate->LoadChainTip();
6469 m_active_chainstate = m_snapshot_chainstate.get();
6471 LogPrintf(
"[snapshot] successfully activated snapshot %s\n",
6474 m_snapshot_chainstate->CoinsTip().DynamicMemoryUsage() /
6477 this->MaybeRebalanceCaches();
6483 bool snapshot_loaded) {
6486 snapshot_loaded ?
"saving snapshot chainstate"
6487 :
"flushing coins cache",
6489 BCLog::LogFlags::ALL);
6491 coins_cache.
Flush();
6495 const char *
what()
const throw()
override {
6496 return "ComputeUTXOStats interrupted by shutdown.";
6520 if (!snapshot_start_block) {
6523 LogPrintf(
"[snapshot] Did not find snapshot start blockheader %s\n",
6528 int base_height = snapshot_start_block->
nHeight;
6531 if (!maybe_au_data) {
6532 LogPrintf(
"[snapshot] assumeutxo height in snapshot metadata not "
6533 "recognized (%d) - refusing to load snapshot\n",
6545 LogPrintf(
"[snapshot] loading coins from snapshot %s\n",
6547 int64_t coins_processed{0};
6549 while (coins_left > 0) {
6551 coins_file >> outpoint;
6553 }
catch (
const std::ios_base::failure &) {
6554 LogPrintf(
"[snapshot] bad snapshot format or truncated snapshot "
6555 "after deserializing %d coins\n",
6556 coins_count - coins_left);
6559 if (coin.
GetHeight() > uint32_t(base_height) ||
6562 std::numeric_limits<
decltype(outpoint.GetN())>::max()) {
6564 "[snapshot] bad snapshot data after deserializing %d coins\n",
6565 coins_count - coins_left);
6574 if (coins_processed % 1000000 == 0) {
6575 LogPrintf(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)\n",
6577 static_cast<float>(coins_processed) * 100 /
6578 static_cast<float>(coins_count),
6586 if (coins_processed % 120000 == 0) {
6591 const auto snapshot_cache_state =
WITH_LOCK(
6592 ::cs_main,
return snapshot_chainstate.GetCoinsCacheSizeState());
6615 bool out_of_coins{
false};
6617 coins_file >> outpoint;
6618 }
catch (
const std::ios_base::failure &) {
6620 out_of_coins =
true;
6622 if (!out_of_coins) {
6623 LogPrintf(
"[snapshot] bad snapshot - coins left over after "
6624 "deserializing %d coins\n",
6629 LogPrintf(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s\n",
6643 std::optional<CCoinsStats> maybe_stats;
6652 if (!maybe_stats.has_value()) {
6653 LogPrintf(
"[snapshot] failed to generate coins stats\n");
6661 LogPrintf(
"[snapshot] bad snapshot content hash: expected %s, got %s\n",
6663 maybe_stats->hashSerialized.ToString());
6680 constexpr int AFTER_GENESIS_START{1};
6682 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height();
6684 index = snapshot_chainstate.
m_chain[i];
6699 index->nStatus = index->nStatus.withAssumedValid();
6714 LogPrintf(
"[snapshot] validated snapshot (%.2f MB)\n",
6737 if (m_ibd_chainstate.get() == &this->ActiveChainstate() ||
6738 !this->IsUsable(m_snapshot_chainstate.get()) ||
6739 !this->IsUsable(m_ibd_chainstate.get()) ||
6740 !m_ibd_chainstate->m_chain.Tip()) {
6746 const int snapshot_base_height = *
Assert(this->GetSnapshotBaseHeight());
6749 if (index_new.
nHeight < snapshot_base_height) {
6759 _(
"%s failed to validate the -assumeutxo snapshot state. "
6760 "This indicates a hardware problem, or a bug in the software, or "
6761 "a bad software modification that allowed an invalid snapshot to "
6762 "be loaded. As a result of this, the node will shut down and "
6763 "stop using any state that was built on the snapshot, resetting "
6764 "the chain height from %d to %d. On the next restart, the node "
6765 "will resume syncing from %d without using any snapshot data. "
6766 "Please report this incident to %s, including how you obtained "
6767 "the snapshot. The invalid snapshot chainstate will be left on "
6768 "disk in case it is helpful in diagnosing the issue that caused "
6770 PACKAGE_NAME, snapshot_tip_height, snapshot_base_height,
6771 snapshot_base_height, PACKAGE_BUGREPORT);
6774 LogPrintf(
"[snapshot] deleting snapshot, reverting to validated chain, "
6775 "and stopping node\n");
6777 m_active_chainstate = m_ibd_chainstate.get();
6778 m_snapshot_chainstate->m_disabled =
true;
6782 auto rename_result = m_snapshot_chainstate->InvalidateCoinsDBOnDisk();
6783 if (!rename_result) {
6788 shutdown_fnc(user_error);
6793 "[snapshot] supposed base block %s does not match the "
6794 "snapshot base block %s (height %d). Snapshot is not valid.\n",
6796 snapshot_base_height);
6797 handle_invalid_snapshot();
6803 int curr_height = m_ibd_chainstate->m_chain.Height();
6805 assert(snapshot_base_height == curr_height);
6810 CCoinsViewDB &ibd_coins_db = m_ibd_chainstate->CoinsDB();
6811 m_ibd_chainstate->ForceFlushStateToDisk();
6814 if (!maybe_au_data) {
6815 LogPrintf(
"[snapshot] assumeutxo data not found for height "
6816 "(%d) - refusing to validate snapshot\n",
6818 handle_invalid_snapshot();
6823 std::optional<CCoinsStats> maybe_ibd_stats;
6825 "[snapshot] computing UTXO stats for background chainstate to validate "
6826 "snapshot - this could take a few minutes\n");
6835 if (!maybe_ibd_stats) {
6837 "[snapshot] failed to generate stats for validation coins db\n");
6841 handle_invalid_snapshot();
6844 const auto &ibd_stats = *maybe_ibd_stats;
6853 LogPrintf(
"[snapshot] hash mismatch: actual=%s, expected=%s\n",
6854 ibd_stats.hashSerialized.ToString(),
6856 handle_invalid_snapshot();
6860 LogPrintf(
"[snapshot] snapshot beginning at %s has been fully validated\n",
6863 m_ibd_chainstate->m_disabled =
true;
6864 this->MaybeRebalanceCaches();
6871 assert(m_active_chainstate);
6872 return *m_active_chainstate;
6877 return m_snapshot_chainstate &&
6878 m_active_chainstate == m_snapshot_chainstate.get();
6880void ChainstateManager::MaybeRebalanceCaches() {
6882 bool ibd_usable = this->
IsUsable(m_ibd_chainstate.get());
6883 bool snapshot_usable = this->
IsUsable(m_snapshot_chainstate.get());
6884 assert(ibd_usable || snapshot_usable);
6886 if (ibd_usable && !snapshot_usable) {
6887 LogPrintf(
"[snapshot] allocating all cache to the IBD chainstate\n");
6891 }
else if (snapshot_usable && !ibd_usable) {
6895 "[snapshot] allocating all cache to the snapshot chainstate\n");
6899 }
else if (ibd_usable && snapshot_usable) {
6905 if (m_snapshot_chainstate->IsInitialBlockDownload()) {
6908 m_snapshot_chainstate->ResizeCoinsCaches(
6911 m_snapshot_chainstate->ResizeCoinsCaches(
6919void ChainstateManager::ResetChainstates() {
6920 m_ibd_chainstate.reset();
6921 m_snapshot_chainstate.reset();
6922 m_active_chainstate =
nullptr;
6931 if (!opts.check_block_index.has_value()) {
6933 opts.config.GetChainParams().DefaultConsistencyChecks();
6936 if (!opts.minimum_chain_work.has_value()) {
6938 opts.config.GetChainParams().GetConsensus().nMinimumChainWork);
6940 if (!opts.assumed_valid_block.has_value()) {
6941 opts.assumed_valid_block =
6942 opts.config.GetChainParams().GetConsensus().defaultAssumeValid;
6944 Assert(opts.adjusted_time_callback);
6945 return std::move(opts);
6951 m_blockman{
std::move(blockman_options)} {}
6953bool ChainstateManager::DetectSnapshotChainstate(
CTxMemPool *mempool) {
6954 assert(!m_snapshot_chainstate);
6959 std::optional<BlockHash> base_blockhash =
6961 if (!base_blockhash) {
6964 LogPrintf(
"[snapshot] detected active snapshot chainstate (%s) - loading\n",
6967 this->ActivateExistingSnapshot(mempool, *base_blockhash);
6972ChainstateManager::ActivateExistingSnapshot(
CTxMemPool *mempool,
6974 assert(!m_snapshot_chainstate);
6975 m_snapshot_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman,
6976 *
this, base_blockhash);
6977 LogPrintf(
"[snapshot] switching active chainstate to %s\n",
6978 m_snapshot_chainstate->ToString());
6979 m_active_chainstate = m_snapshot_chainstate.get();
6980 return *m_snapshot_chainstate;
6989 assert(storage_path_maybe);
6990 fs::path snapshot_datadir = *storage_path_maybe;
6995 auto invalid_path = snapshot_datadir +
"_INVALID";
6998 LogPrintf(
"[snapshot] renaming snapshot datadir %s to %s\n", dbpath,
7005 fs::rename(snapshot_datadir, invalid_path);
7006 }
catch (
const fs::filesystem_error &e) {
7010 LogPrintf(
"%s: error renaming file '%s' -> '%s': %s\n", __func__,
7011 src_str, dest_str, e.what());
7013 "You should resolve this by manually "
7014 "moving or deleting the invalid "
7015 "snapshot directory %s, otherwise you "
7016 "will encounter the same error again "
7017 "on the next startup."),
7018 src_str, dest_str, src_str)};
7023const CBlockIndex *ChainstateManager::GetSnapshotBaseBlock()
const {
7024 return m_active_chainstate ? m_active_chainstate->SnapshotBase() :
nullptr;
7027std::optional<int> ChainstateManager::GetSnapshotBaseHeight()
const {
7028 const CBlockIndex *base = this->GetSnapshotBaseBlock();
7029 return base ? std::make_optional(base->
nHeight) :
std::nullopt;
7032bool ChainstateManager::ValidatedSnapshotCleanup() {
7041 std::optional<fs::path> ibd_chainstate_path_maybe =
7042 get_storage_path(m_ibd_chainstate);
7043 std::optional<fs::path> snapshot_chainstate_path_maybe =
7044 get_storage_path(m_snapshot_chainstate);
7053 if (!ibd_chainstate_path_maybe || !snapshot_chainstate_path_maybe) {
7054 LogPrintf(
"[snapshot] snapshot chainstate cleanup cannot happen with "
7055 "in-memory chainstates. You are testing, right?\n");
7059 const auto &snapshot_chainstate_path = *snapshot_chainstate_path_maybe;
7060 const auto &ibd_chainstate_path = *ibd_chainstate_path_maybe;
7068 this->ResetChainstates();
7073 LogPrintf(
"[snapshot] deleting background chainstate directory (now "
7074 "unnecessary) (%s)\n",
7077 fs::path tmp_old{ibd_chainstate_path +
"_todelete"};
7080 const fs::filesystem_error &err) {
7084 "Rename of '%s' -> '%s' failed. "
7085 "Cannot clean up the background chainstate leveldb directory.",
7090 fs::rename(ibd_chainstate_path, tmp_old);
7091 }
catch (
const fs::filesystem_error &e) {
7092 rename_failed_abort(ibd_chainstate_path, tmp_old, e);
7096 LogPrintf(
"[snapshot] moving snapshot chainstate (%s) to "
7097 "default chainstate directory (%s)\n",
7102 fs::rename(snapshot_chainstate_path, ibd_chainstate_path);
7103 }
catch (
const fs::filesystem_error &e) {
7104 rename_failed_abort(snapshot_chainstate_path, ibd_chainstate_path, e);
7111 LogPrintf(
"Deletion of %s failed. Please remove it manually, as the "
7112 "directory is now unnecessary.\n",
7115 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
bool IsAssumedValid() const EXCLUSIVE_LOCKS_REQUIRED(
int64_t GetHeaderReceivedTime() const
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.
bool HaveTxsDownloaded() const
Check whether this block's and all previous blocks' transactions have been downloaded (and stored to ...
int64_t GetChainTxCount() const
Get the number of transaction in the chain so far.
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.
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
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
const MapAssumeutxo & Assumeutxo() const
Get allowed assumeutxo configuration.
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 BlockConnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
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(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()
void ClearBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network)
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.
void InitCoinsDB(size_t cache_size_bytes, bool in_memory, bool should_wipe, std::string leveldb_name="chainstate")
Initialize the CoinsViews UTXO set database management data structures.
CBlockIndex const * m_best_fork_base
void InvalidChainFound(CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(cs_main
std::atomic< bool > m_cached_finished_ibd
Whether this chainstate is undergoing initial block download.
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.
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...
const CBlockIndex *GetSnapshotBaseBlock() const EXCLUSIVE_LOCKS_REQUIRED(std::optional< int > GetSnapshotBaseHeight() const EXCLUSIVE_LOCKS_REQUIRED(bool IsUsable(const Chainstate *const pchainstate) const EXCLUSIVE_LOCKS_REQUIRED(
Returns nullptr if no snapshot has been loaded.
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(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
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
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())
SnapshotCompletionResult MaybeCompleteSnapshotValidation(std::function< void(bilingual_str)> shutdown_fnc=[](bilingual_str msg) { AbortNode(msg.original, msg);}) EXCLUSIVE_LOCKS_REQUIRED(Chainstate & ActiveChainstate() const
Once the background validation chainstate has reached the height which is the base of the UTXO snapsh...
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
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...
const CBlockIndex *GetFirstStoredBlock(const CBlockIndex &start_block) EXCLUSIVE_LOCKS_REQUIRED(bool m_have_pruned
Find the first block that is not pruned.
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() 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 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.
void FlushBlockFile(bool fFinalize=false, bool finalize_undo=false)
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.
std::map< int, const AssumeutxoData > MapAssumeutxo
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.
const AssumeutxoHash hash_serialized
The expected hash of the deserialized UTXO set.
const 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 BIP34Height
Block height and hash at which BIP34 becomes active.
int nSubsidyHalvingInterval
BlockHash hashGenesisBlock
int schumpeterActivationTime
Unix time used for MTP activation of 15 May 2025 12:00:00 UTC upgrade.
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(
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.
const AssumeutxoData * ExpectedAssumeutxo(const int height, const CChainParams &chainparams)
Return the expected assumeutxo value for a given height, if one exists.
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
static bool NotifyHeaderTip(Chainstate &chainstate) LOCKS_EXCLUDED(cs_main)
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 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()