103 "level 0 reads the blocks from disk",
104 "level 1 verifies block validity",
105 "level 2 verifies undo data",
106 "level 3 checks disconnection of tip blocks",
107 "level 4 tries to reconnect the blocks",
108 "each level includes the checks of the previous levels",
125 : excessiveBlockSize(config.GetMaxBlockSize()), checkPoW(true),
126 checkMerkleRoot(true) {}
163std::optional<std::vector<int>> CalculatePrevHeights(
const CBlockIndex &tip,
165 const CTransaction &tx) {
166 std::vector<int> prev_heights;
167 prev_heights.resize(tx.vin.size());
168 for (
size_t i = 0; i < tx.vin.size(); ++i) {
169 const CTxIn &txin = tx.vin[i];
171 if (!coins.
GetCoin(txin.prevout, coin)) {
172 LogPrintf(
"ERROR: %s: Missing input %d in transaction \'%s\'\n",
173 __func__, i, tx.GetId().GetHex());
178 prev_heights[i] = tip.
nHeight + 1;
189 const CTransaction &tx) {
192 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
193 if (!prev_heights.has_value()) {
198 next_tip.
pprev = tip;
233 const std::optional<int64_t> activation_time) {
234 if (pindexPrev ==
nullptr) {
257 for (
const CTxIn &txin : tx.vin) {
273 assert(txFrom->GetId() == txin.prevout.GetTxId());
274 assert(txFrom->vout.size() > txin.prevout.GetN());
287 validation_cache, nSigChecksOut);
295 : m_pool(mempool), m_view(&m_dummy),
296 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
297 m_active_chainstate(active_chainstate) {}
303 const int64_t m_accept_time;
304 const bool m_bypass_limits;
312 std::vector<COutPoint> &m_coins_to_uncache;
313 const bool m_test_accept;
314 const unsigned int m_heightOverride;
320 const bool m_package_submission;
326 const bool m_package_feerates;
329 static ATMPArgs SingleAccept(
const Config &config, int64_t accept_time,
331 std::vector<COutPoint> &coins_to_uncache,
333 unsigned int heightOverride) {
351 PackageTestAccept(
const Config &config, int64_t accept_time,
352 std::vector<COutPoint> &coins_to_uncache) {
368 PackageChildWithParents(
const Config &config, int64_t accept_time,
369 std::vector<COutPoint> &coins_to_uncache) {
383 static ATMPArgs SingleInPackageAccept(
const ATMPArgs &package_args) {
385 package_args.m_config,
386 package_args.m_accept_time,
388 package_args.m_coins_to_uncache,
389 package_args.m_test_accept,
390 package_args.m_heightOverride,
402 ATMPArgs(
const Config &config, int64_t accept_time,
bool bypass_limits,
403 std::vector<COutPoint> &coins_to_uncache,
bool test_accept,
404 unsigned int height_override,
bool package_submission,
405 bool package_feerates)
406 : m_config{config}, m_accept_time{accept_time},
407 m_bypass_limits{bypass_limits},
408 m_coins_to_uncache{coins_to_uncache}, m_test_accept{test_accept},
409 m_heightOverride{height_override},
410 m_package_submission{package_submission},
411 m_package_feerates(package_feerates) {}
426 AcceptMultipleTransactions(
const std::vector<CTransactionRef> &txns,
443 AcceptSubPackage(
const std::vector<CTransactionRef> &subpackage,
461 const uint32_t next_block_script_verify_flags)
463 m_next_block_script_verify_flags(next_block_script_verify_flags) {
470 std::unique_ptr<CTxMemPoolEntry> m_entry;
508 const uint32_t m_next_block_script_verify_flags;
509 int m_sig_checks_standard;
517 bool PreChecks(ATMPArgs &args, Workspace &ws)
524 bool ConsensusScriptChecks(
const ATMPArgs &args, Workspace &ws)
530 bool Finalize(
const ATMPArgs &args, Workspace &ws)
538 bool SubmitPackage(
const ATMPArgs &args, std::vector<Workspace> &workspaces,
540 std::map<TxId, MempoolAcceptResult> &results)
544 bool CheckFeeRate(
size_t package_size,
size_t package_vsize,
550 const Amount mempoolRejectFee =
551 m_pool.GetMinFee().GetFee(package_vsize);
554 package_fee < mempoolRejectFee) {
557 "mempool min fee not met",
558 strprintf(
"%d < %d", package_fee, mempoolRejectFee));
563 if (package_fee < m_pool.m_min_relay_feerate.GetFee(package_size)) {
566 "min relay fee not met",
568 m_pool.m_min_relay_feerate.GetFee(package_size)));
575 return m_active_chainstate.m_chainman.m_validation_cache;
587bool MemPoolAccept::PreChecks(ATMPArgs &args, Workspace &ws) {
591 const CTransaction &tx = *ws.m_ptx;
592 const TxId &txid = ws.m_ptx->GetId();
595 const int64_t nAcceptTime = args.m_accept_time;
596 const bool bypass_limits = args.m_bypass_limits;
597 std::vector<COutPoint> &coins_to_uncache = args.m_coins_to_uncache;
598 const unsigned int heightOverride = args.m_heightOverride;
610 if (m_pool.m_require_standard &&
612 m_pool.m_permit_bare_multisig,
613 m_pool.m_dust_relay_feerate, reason)) {
622 *
Assert(m_active_chainstate.m_chain.Tip()),
623 args.m_config.GetChainParams().GetConsensus(), tx, ctxState)) {
632 if (m_pool.exists(txid)) {
634 "txn-already-in-mempool");
638 for (
const CTxIn &txin : tx.vin) {
639 if (
const auto ptxConflicting = m_pool.GetConflictTx(txin.prevout)) {
640 if (m_pool.isAvalancheFinalizedPreConsensus(
641 ptxConflicting->GetId())) {
643 "finalized-tx-conflict");
648 "txn-mempool-conflict");
652 m_view.SetBackend(m_viewmempool);
656 for (
const CTxIn &txin : tx.vin) {
658 coins_to_uncache.push_back(txin.prevout);
665 if (!m_view.HaveCoin(txin.prevout)) {
667 for (
size_t out = 0; out < tx.vout.size(); out++) {
672 "txn-already-known");
679 "bad-txns-inputs-missingorspent");
684 if (!m_view.HaveInputs(tx)) {
686 "bad-txns-inputs-spent");
690 m_view.GetBestBlock();
695 m_view.SetBackend(m_dummy);
697 assert(m_active_chainstate.m_blockman.LookupBlockIndex(
698 m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
707 m_active_chainstate.m_chain.Tip(), m_view, tx)};
708 if (!lock_points.has_value() ||
718 m_active_chainstate.m_chain.Height() + 1,
725 if (m_pool.m_require_standard &&
728 "bad-txns-nonstandard-inputs");
732 ws.m_modified_fees = ws.m_base_fees;
733 m_pool.ApplyDelta(txid, ws.m_modified_fees);
735 unsigned int nSize = tx.GetTotalSize();
738 const uint32_t scriptVerifyFlags =
742 ws.m_precomputed_txdata, GetValidationCache(),
743 ws.m_sig_checks_standard)) {
748 ws.m_entry = std::make_unique<CTxMemPoolEntry>(
749 ptx, ws.m_base_fees, nAcceptTime,
750 heightOverride ? heightOverride : m_active_chainstate.m_chain.Height(),
751 ws.m_sig_checks_standard, lock_points.value());
753 ws.m_vsize = ws.m_entry->GetTxVirtualSize();
761 if (!bypass_limits &&
763 m_pool.m_min_relay_feerate.GetFee(ws.m_ptx->GetTotalSize())) {
770 m_pool.m_min_relay_feerate.GetFee(nSize)));
775 if (!bypass_limits && !args.m_package_feerates &&
776 !CheckFeeRate(nSize, ws.m_vsize, ws.m_modified_fees, state)) {
783bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs &args, Workspace &ws) {
786 const CTransaction &tx = *ws.m_ptx;
787 const TxId &txid = tx.GetId();
801 int nSigChecksConsensus;
803 tx, state, m_view, m_pool, ws.m_next_block_script_verify_flags,
804 ws.m_precomputed_txdata, GetValidationCache(), nSigChecksConsensus,
805 m_active_chainstate.CoinsTip())) {
809 LogPrintf(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against "
810 "latest-block but not STANDARD flags %s, %s\n",
815 if (ws.m_sig_checks_standard != nSigChecksConsensus) {
820 "%s: BUG! PLEASE REPORT THIS! SigChecks count differed between "
821 "standard and consensus flags in %s\n",
828bool MemPoolAccept::Finalize(
const ATMPArgs &args, Workspace &ws) {
831 const TxId &txid = ws.m_ptx->GetId();
833 const bool bypass_limits = args.m_bypass_limits;
838 m_pool.addUnchecked(entry);
842 std::make_shared<
const std::vector<Coin>>(
844 m_pool.GetAndIncrementSequence());
851 if (!args.m_package_submission && !bypass_limits) {
852 m_pool.LimitSize(m_active_chainstate.CoinsTip());
853 if (!m_pool.exists(txid)) {
863bool MemPoolAccept::SubmitPackage(
864 const ATMPArgs &args, std::vector<Workspace> &workspaces,
866 std::map<TxId, MempoolAcceptResult> &results) {
871 workspaces.cbegin(), workspaces.cend(),
872 [
this](
const auto &ws) { return !m_pool.exists(ws.m_ptx->GetId()); }));
874 bool all_submitted =
true;
881 for (Workspace &ws : workspaces) {
882 if (!ConsensusScriptChecks(args, ws)) {
883 results.emplace(ws.m_ptx->GetId(),
886 all_submitted =
false;
889 strprintf(
"BUG! PolicyScriptChecks succeeded but "
890 "ConsensusScriptChecks failed: %s",
891 ws.m_ptx->GetId().ToString()));
901 if (!Finalize(args, ws)) {
902 results.emplace(ws.m_ptx->GetId(),
905 all_submitted =
false;
907 strprintf(
"BUG! Adding to mempool failed: %s",
908 ws.m_ptx->GetId().ToString()));
914 m_pool.LimitSize(m_active_chainstate.CoinsTip());
916 std::vector<TxId> all_package_txids;
917 all_package_txids.reserve(workspaces.size());
918 std::transform(workspaces.cbegin(), workspaces.cend(),
919 std::back_inserter(all_package_txids),
920 [](
const auto &ws) { return ws.m_ptx->GetId(); });
924 for (Workspace &ws : workspaces) {
925 const auto effective_feerate =
926 args.m_package_feerates
927 ? ws.m_package_feerate
929 static_cast<uint32_t
>(ws.m_vsize)};
930 const auto effective_feerate_txids =
931 args.m_package_feerates ? all_package_txids
932 : std::vector<TxId>({ws.m_ptx->GetId()});
933 results.emplace(ws.m_ptx->GetId(),
936 effective_feerate_txids));
938 return all_submitted;
949 const CBlockIndex *tip = m_active_chainstate.m_chain.Tip();
954 const std::vector<TxId> single_txid{ws.m_ptx->GetId()};
959 if (!PreChecks(args, ws)) {
960 if (ws.m_state.GetResult() ==
965 ws.m_state,
CFeeRate(ws.m_modified_fees, ws.m_vsize),
971 if (!ConsensusScriptChecks(args, ws)) {
975 const TxId txid = ptx->GetId();
985 if (
auto it = m_pool.mapNextTx.lower_bound(COutPoint{txid, 0});
986 it != m_pool.mapNextTx.end() && it->first->GetTxId() == txid) {
987 LogPrintf(
"%s: BUG! PLEASE REPORT THIS! Attempt to add txid %s, but "
988 "its outputs are already spent in the "
992 "txn-child-before-parent");
996 const CFeeRate effective_feerate{ws.m_modified_fees,
997 static_cast<uint32_t
>(ws.m_vsize)};
999 if (args.m_test_accept) {
1001 effective_feerate, single_txid);
1004 if (!Finalize(args, ws)) {
1008 Assume(ws.m_state.GetResult() ==
1011 ws.m_state,
CFeeRate(ws.m_modified_fees, ws.m_vsize), single_txid);
1015 effective_feerate, single_txid);
1019 const std::vector<CTransactionRef> &txns, ATMPArgs &args) {
1029 std::vector<Workspace> workspaces{};
1030 workspaces.reserve(txns.size());
1032 txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1033 [
this](
const auto &tx) {
1035 tx, GetNextBlockScriptFlags(m_active_chainstate.m_chain.Tip(),
1036 m_active_chainstate.m_chainman));
1038 std::map<TxId, MempoolAcceptResult> results;
1044 std::vector<TxId> valid_txids;
1045 for (Workspace &ws : workspaces) {
1046 if (!PreChecks(args, ws)) {
1048 "transaction failed");
1051 results.emplace(ws.m_ptx->GetId(),
1054 std::move(results));
1058 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1059 valid_txids.push_back(ws.m_ptx->GetId());
1074 const auto m_total_size = std::accumulate(
1075 workspaces.cbegin(), workspaces.cend(), int64_t{0},
1076 [](int64_t
sum,
auto &ws) { return sum + ws.m_ptx->GetTotalSize(); });
1077 const auto m_total_vsize =
1078 std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1079 [](int64_t
sum,
auto &ws) { return sum + ws.m_vsize; });
1080 const auto m_total_modified_fees = std::accumulate(
1081 workspaces.cbegin(), workspaces.cend(),
Amount::zero(),
1082 [](
Amount sum,
auto &ws) { return sum + ws.m_modified_fees; });
1083 const CFeeRate package_feerate(m_total_modified_fees, m_total_vsize);
1084 std::vector<TxId> all_package_txids;
1085 all_package_txids.reserve(workspaces.size());
1086 std::transform(workspaces.cbegin(), workspaces.cend(),
1087 std::back_inserter(all_package_txids),
1088 [](
const auto &ws) { return ws.m_ptx->GetId(); });
1090 if (args.m_package_feerates &&
1091 !CheckFeeRate(m_total_size, m_total_vsize, m_total_modified_fees,
1092 placeholder_state)) {
1094 "transaction failed");
1096 package_state, {{workspaces.back().m_ptx->GetId(),
1099 CFeeRate(m_total_modified_fees, m_total_vsize),
1100 all_package_txids)}});
1103 for (Workspace &ws : workspaces) {
1104 ws.m_package_feerate = package_feerate;
1105 const TxId &ws_txid = ws.m_ptx->GetId();
1106 if (args.m_test_accept &&
1107 std::find(valid_txids.begin(), valid_txids.end(), ws_txid) !=
1108 valid_txids.end()) {
1109 const auto effective_feerate =
1110 args.m_package_feerates
1111 ? ws.m_package_feerate
1113 static_cast<uint32_t
>(ws.m_vsize)};
1114 const auto effective_feerate_txids =
1115 args.m_package_feerates ? all_package_txids
1116 : std::vector<TxId>{ws.m_ptx->GetId()};
1120 results.emplace(ws_txid,
1122 ws.m_vsize, ws.m_base_fees, effective_feerate,
1123 effective_feerate_txids));
1127 if (args.m_test_accept) {
1131 if (!SubmitPackage(args, workspaces, package_state, results)) {
1140MemPoolAccept::AcceptSubPackage(
const std::vector<CTransactionRef> &subpackage,
1146 if (subpackage.size() > 1) {
1147 return AcceptMultipleTransactions(subpackage, args);
1149 const auto &tx = subpackage.front();
1150 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(args);
1151 const auto single_res = AcceptSingleTransaction(tx, single_args);
1153 if (single_res.m_result_type !=
1156 "transaction failed");
1159 {{tx->GetId(), single_res}});
1189 for (
const auto &outpoint : m_viewmempool.GetNonBaseCoins()) {
1193 m_view.Uncache(outpoint);
1196 m_viewmempool.Reset();
1212 if (!
CheckPackage(package, package_state_quit_early)) {
1221 "package-not-child-with-parents");
1226 assert(package.size() > 1);
1229 const auto &child = package.back();
1230 std::unordered_set<TxId, SaltedTxIdHasher> unconfirmed_parent_txids;
1232 package.cbegin(), package.cend() - 1,
1233 std::inserter(unconfirmed_parent_txids, unconfirmed_parent_txids.end()),
1234 [](
const auto &tx) { return tx->GetId(); });
1243 const CCoinsViewCache &coins_tip_cache = m_active_chainstate.CoinsTip();
1244 for (
const auto &input : child->vin) {
1246 args.m_coins_to_uncache.push_back(input.prevout);
1253 m_view.
SetBackend(m_active_chainstate.CoinsTip());
1254 const auto package_or_confirmed = [
this, &unconfirmed_parent_txids](
1255 const auto &input) {
1256 return unconfirmed_parent_txids.count(input.prevout.GetTxId()) > 0 ||
1257 m_view.HaveCoin(input.prevout);
1259 if (!std::all_of(child->vin.cbegin(), child->vin.cend(),
1260 package_or_confirmed)) {
1261 package_state_quit_early.
Invalid(
1263 "package-not-child-with-unconfirmed-parents");
1269 m_view.SetBackend(m_dummy);
1275 std::map<TxId, MempoolAcceptResult> results_final;
1280 std::map<TxId, MempoolAcceptResult> individual_results_nonfinal;
1281 bool quit_early{
false};
1282 std::vector<CTransactionRef> txns_package_eval;
1283 for (
const auto &tx : package) {
1284 const auto &txid = tx->GetId();
1287 if (m_pool.exists(txid)) {
1301 auto iter = m_pool.GetIter(txid);
1302 assert(iter != std::nullopt);
1304 (*iter.value())->GetTxSize(),
1305 (*iter.value())->GetFee()));
1309 const auto single_package_res = AcceptSubPackage({tx}, args);
1310 const auto &single_res = single_package_res.m_tx_results.at(txid);
1311 if (single_res.m_result_type ==
1316 assert(m_pool.exists(txid));
1317 results_final.emplace(txid, single_res);
1318 }
else if (single_res.m_state.GetResult() !=
1320 single_res.m_state.GetResult() !=
1335 package_state_quit_early.
Invalid(
1337 individual_results_nonfinal.emplace(txid, single_res);
1339 individual_results_nonfinal.emplace(txid, single_res);
1340 txns_package_eval.push_back(tx);
1345 auto multi_submission_result =
1346 quit_early || txns_package_eval.empty()
1348 : AcceptSubPackage(txns_package_eval, args);
1350 multi_submission_result.m_state;
1355 m_pool.LimitSize(m_active_chainstate.CoinsTip());
1357 for (
const auto &tx : package) {
1358 const auto &txid = tx->GetId();
1359 if (multi_submission_result.m_tx_results.count(txid) > 0) {
1362 Assume(results_final.count(txid) == 0);
1366 const auto &txresult =
1367 multi_submission_result.m_tx_results.at(txid);
1368 if (txresult.m_result_type ==
1370 !m_pool.exists(txid)) {
1372 "transaction failed");
1376 results_final.emplace(
1379 results_final.emplace(txid, txresult);
1381 }
else if (
const auto final_it{results_final.find(txid)};
1382 final_it != results_final.end()) {
1385 Assume(final_it->second.m_result_type !=
1387 Assume(individual_results_nonfinal.count(txid) == 0);
1388 if (!m_pool.exists(tx->GetId())) {
1390 "transaction failed");
1395 results_final.erase(txid);
1396 results_final.emplace(
1399 }
else if (
const auto non_final_it{
1400 individual_results_nonfinal.find(txid)};
1401 non_final_it != individual_results_nonfinal.end()) {
1402 Assume(non_final_it->second.m_result_type ==
1405 results_final.emplace(txid, non_final_it->second);
1408 Assume(results_final.size() == package.size());
1410 std::move(results_final));
1416 int64_t accept_time,
bool bypass_limits,
1418 unsigned int heightOverride) {
1423 std::vector<COutPoint> coins_to_uncache;
1424 auto args = MemPoolAccept::ATMPArgs::SingleAccept(
1426 coins_to_uncache, test_accept, heightOverride);
1428 .AcceptSingleTransaction(tx, args);
1436 for (
const COutPoint &outpoint : coins_to_uncache) {
1453 assert(!package.empty());
1454 assert(std::all_of(package.cbegin(), package.cend(),
1455 [](
const auto &tx) { return tx != nullptr; }));
1459 std::vector<COutPoint> coins_to_uncache;
1463 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(
1464 config,
GetTime(), coins_to_uncache);
1465 return MemPoolAccept(pool, active_chainstate)
1466 .AcceptMultipleTransactions(package, args);
1468 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(
1469 config,
GetTime(), coins_to_uncache);
1470 return MemPoolAccept(pool, active_chainstate)
1471 .AcceptPackage(package, args);
1477 if (test_accept || result.m_state.IsInvalid()) {
1478 for (
const COutPoint &hashTx : coins_to_uncache) {
1491 if (halvings >= 64) {
1502 : m_dbview{
std::move(db_params),
std::move(options)},
1503 m_catcherview(&m_dbview) {}
1505void CoinsViews::InitCache() {
1507 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1512 std::optional<BlockHash> from_snapshot_blockhash)
1513 : m_mempool(mempool), m_blockman(blockman), m_chainman(chainman),
1514 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1520 if (!m_cached_snapshot_base) {
1521 m_cached_snapshot_base =
Assert(
1524 return m_cached_snapshot_base;
1528 bool should_wipe, std::string leveldb_name) {
1535 .cache_bytes = cache_size_bytes,
1536 .memory_only = in_memory,
1537 .wipe_data = should_wipe,
1543void Chainstate::InitCoinsCache(
size_t cache_size_bytes) {
1569 if (chain.Tip() ==
nullptr) {
1578 LogPrintf(
"Leaving InitialBlockDownload (latching to false)\n");
1605 std::string warning =
1606 std::string(
"'Warning: Large-work fork detected, forking after "
1613 LogPrintf(
"%s: Warning: Large fork found\n forking the "
1614 "chain at height %d (%s)\n lasting to height %d "
1615 "(%s).\nChain state database corruption likely.\n",
1622 LogPrintf(
"%s: Warning: Found invalid chain at least ~6 blocks "
1623 "longer than our best chain.\nChain state database "
1624 "corruption likely.\n",
1669 SetBlockFailureFlags(pindexNew);
1680 m_avalancheFinalizedBlockIndex = pindexNew->
pprev;
1683 LogPrintf(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n",
1690 LogPrintf(
"%s: current best=%s height=%d log2_work=%f date=%s\n",
1702 pindex->nStatus = pindex->nStatus.withFailed();
1712 if (tx.IsCoinBase()) {
1716 txundo.
vprevout.reserve(tx.vin.size());
1717 for (
const CTxIn &txin : tx.vin) {
1732 std::vector<Coin> spent_coins;
1733 spent_coins.reserve(ptx->vin.size());
1734 for (
const CTxIn &input : ptx->vin) {
1736 const bool coinFound = coins_view.
GetCoin(input.prevout, coin);
1738 spent_coins.push_back(std::move(coin));
1744 const CScript &scriptSig =
ptxTo->vin[
nIn].scriptSig;
1746 auto debug_str =
strprintf(
"input %i of %s, spending %s:%i",
nIn,
1747 ptxTo->GetId().ToString(),
1748 ptxTo->vin[
nIn].prevout.GetTxId().ToString(),
1755 return std::make_pair(error, std::move(debug_str));
1765 std::move(debug_str));
1767 return std::nullopt;
1771 const size_t signature_cache_bytes)
1772 : m_signature_cache{signature_cache_bytes} {
1781 const auto [num_elems, approx_size_bytes] =
1783 LogPrintf(
"Using %zu MiB out of %zu MiB requested for script execution "
1784 "cache, able to store %zu elements\n",
1785 approx_size_bytes >> 20, script_execution_cache_bytes >> 20,
1791 bool sigCacheStore,
bool scriptCacheStore,
1796 std::vector<CScriptCheck> *pvChecks) {
1798 assert(!tx.IsCoinBase());
1801 pvChecks->reserve(tx.vin.size());
1812 elem, !scriptCacheStore);
1814 if (found_in_cache) {
1816 (pBlockLimitSigChecks &&
1819 "too-many-sigchecks");
1824 int nSigChecksTotal = 0;
1826 for (
size_t i = 0; i < tx.vin.size(); i++) {
1827 const COutPoint &prevout = tx.vin[i].prevout;
1840 sigCacheStore, txdata, &txLimitSigChecks, pBlockLimitSigChecks);
1844 pvChecks->push_back(std::move(check));
1848 if (
auto result = check(); result.has_value()) {
1853 uint32_t mandatoryFlags =
1854 flags & ~STANDARD_NOT_MANDATORY_VERIFY_FLAGS;
1855 if (
flags != mandatoryFlags) {
1863 mandatoryFlags, sigCacheStore, txdata);
1864 auto mandatory_result = check2();
1865 if (!mandatory_result.has_value()) {
1868 strprintf(
"non-mandatory-script-verify-flag (%s)",
1879 result = mandatory_result;
1891 strprintf(
"mandatory-script-verify-flag-failed (%s)",
1899 nSigChecksOut = nSigChecksTotal;
1901 if (scriptCacheStore && !pvChecks) {
1912 const std::string &strMessage,
1914 notifications.
fatalError(strMessage, userMessage);
1915 return state.
Error(strMessage);
1920 const COutPoint &out) {
1928 if (undo.GetHeight() == 0) {
1951 view.
AddCoin(out, std::move(undo), !fClean);
1966 LogError(
"DisconnectBlock(): failure reading undo data\n");
1970 return ApplyBlockUndo(std::move(blockUndo), block, pindex, view);
1978 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
1979 LogError(
"DisconnectBlock(): block and undo data inconsistent\n");
1984 for (
size_t i = 1; i < block.
vtx.size(); i++) {
1985 const CTransaction &tx = *(block.
vtx[i]);
1987 if (txundo.
vprevout.size() != tx.vin.size()) {
1989 "DisconnectBlock(): transaction and undo data inconsistent\n");
1993 for (
size_t j = 0; j < tx.vin.size(); j++) {
1994 const COutPoint &out = tx.vin[j].prevout;
2006 for (
const auto &ptx : block.
vtx) {
2007 const CTransaction &tx = *ptx;
2008 const TxId &txid = tx.GetId();
2009 const bool is_coinbase = tx.IsCoinBase();
2013 for (
size_t o = 0; o < tx.vout.size(); o++) {
2014 if (tx.vout[o].scriptPubKey.IsUnspendable()) {
2018 COutPoint out(txid, o);
2020 bool is_spent = view.
SpendCoin(out, &coin);
2021 if (!is_spent || tx.vout[o] != coin.
GetTxOut() ||
2110 consensusparams, pindex,
2142 const auto time_start{SteadyClock::now()};
2159 if (!
CheckBlock(block, state, consensusParams,
2167 "Corrupt block found indicating potential "
2168 "hardware failure; shutting down");
2191 bool fScriptChecks =
true;
2200 BlockMap::const_iterator it{
2203 if (it->second.GetAncestor(pindex->
nHeight) == pindex &&
2231 consensusParams) <= 60 * 60 * 24 * 7 * 2);
2236 const auto time_1{SteadyClock::now()};
2239 Ticks<MillisecondsDouble>(time_1 - time_start),
2254 bool fEnforceBIP30 = !((pindex->
nHeight == 91842 &&
2256 uint256S(
"0x00000000000a4d0a398161ffc163c503763"
2257 "b1f4360639393e0e4c8e300e0caec")) ||
2260 uint256S(
"0x00000000000743f190a18c5577a3c2d2a1f"
2261 "610ae9601ac046a38084ccb7cd721")));
2292 static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2329 (!pindexBIP34height ||
2335 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2336 for (
const auto &tx : block.
vtx) {
2337 for (
size_t o = 0; o < tx->vout.size(); o++) {
2338 if (view.
HaveCoin(COutPoint(tx->GetId(), o))) {
2341 "tried to overwrite transaction");
2348 int nLockTimeFlags = 0;
2356 const auto time_2{SteadyClock::now()};
2359 Ticks<MillisecondsDouble>(time_2 - time_1),
2363 std::vector<int> prevheights;
2375 std::vector<TxSigCheckLimiter> nSigChecksTxLimiters;
2376 nSigChecksTxLimiters.resize(block.
vtx.size() - 1);
2379 blockundo.
vtxundo.resize(block.
vtx.size() - 1);
2386 for (
const auto &ptx : block.
vtx) {
2389 }
catch (
const std::logic_error &e) {
2399 "tx-duplicate",
"tried to overwrite transaction");
2406 for (
const auto &ptx : block.
vtx) {
2407 const CTransaction &tx = *ptx;
2408 const bool isCoinBase = tx.IsCoinBase();
2409 nInputs += tx.vin.size();
2422 tx.GetId().ToString());
2430 "bad-txns-accumulated-fee-outofrange",
2431 "accumulated fee in the block out of range");
2443 prevheights.resize(tx.vin.size());
2444 for (
size_t j = 0; j < tx.vin.size(); j++) {
2448 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2450 "bad-txns-nonfinal",
2451 "contains a non-BIP68-final transaction " +
2452 tx.GetHash().ToString());
2458 bool fCacheResults = fJustCheck;
2461 if (!fEnforceSigCheck) {
2468 std::vector<CScriptCheck> vChecks;
2470 if (fScriptChecks &&
2474 nSigChecksTxLimiters[txIndex],
2475 &nSigChecksBlockLimiter, &vChecks)) {
2484 control.
Add(std::move(vChecks));
2494 const auto time_3{SteadyClock::now()};
2497 " - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) "
2498 "[%.2fs (%.2fms/blk)]\n",
2499 (
unsigned)block.
vtx.size(),
2500 Ticks<MillisecondsDouble>(time_3 - time_2),
2501 Ticks<MillisecondsDouble>(time_3 - time_2) / block.
vtx.size(),
2504 : Ticks<MillisecondsDouble>(time_3 - time_2) / (nInputs - 1),
2508 const Amount blockReward =
2510 if (block.
vtx[0]->GetValueOut() > blockReward && state.
IsValid()) {
2513 strprintf(
"coinbase pays too much (actual=%d vs limit=%d)",
2514 block.
vtx[0]->GetValueOut(), blockReward));
2521 auto parallel_result = control.
Complete();
2522 if (parallel_result.has_value() && state.
IsValid()) {
2524 strprintf(
"mandatory-script-verify-flag-failed (%s)",
2526 parallel_result->second);
2532 const auto time_4{SteadyClock::now()};
2536 " - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n",
2537 nInputs - 1, Ticks<MillisecondsDouble>(time_4 - time_2),
2540 : Ticks<MillisecondsDouble>(time_4 - time_2) / (nInputs - 1),
2548 if (!
m_blockman.WriteUndoDataForBlock(blockundo, state, *pindex)) {
2560 const auto time_5{SteadyClock::now()};
2563 Ticks<MillisecondsDouble>(time_5 - time_4),
2567 TRACE6(validation, block_connected, block_hash.data(), pindex->
nHeight,
2568 block.
vtx.size(), nInputs, nSigChecksRet,
2570 time_5 - time_start);
2583Chainstate::GetCoinsCacheSizeState(
size_t max_coins_cache_size_bytes,
2584 size_t max_mempool_size_bytes) {
2588 int64_t nTotalSpace =
2589 max_coins_cache_size_bytes +
2590 std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2593 static constexpr int64_t MAX_BLOCK_COINSDB_USAGE_BYTES =
2595 int64_t large_threshold = std::max(
2596 (9 * nTotalSpace) / 10, nTotalSpace - MAX_BLOCK_COINSDB_USAGE_BYTES);
2598 if (cacheSize > nTotalSpace) {
2599 LogPrintf(
"Cache size (%s) exceeds total space (%s)\n", cacheSize,
2602 }
else if (cacheSize > large_threshold) {
2612 std::set<int> setFilesToPrune;
2613 bool full_flush_completed =
false;
2620 bool fFlushForPrune =
false;
2632 std::optional<std::string> limiting_lock;
2634 for (
const auto &prune_lock :
m_blockman.m_prune_locks) {
2635 if (prune_lock.second.height_first ==
2636 std::numeric_limits<int>::max()) {
2641 const int lock_height{prune_lock.second.height_first -
2643 last_prune = std::max(1, std::min(last_prune, lock_height));
2644 if (last_prune == lock_height) {
2645 limiting_lock = prune_lock.first;
2649 if (limiting_lock) {
2651 limiting_lock.value(), last_prune);
2654 if (nManualPruneHeight > 0) {
2659 std::min(last_prune, nManualPruneHeight), *
this,
2668 if (!setFilesToPrune.empty()) {
2669 fFlushForPrune =
true;
2672 "prunedblockfiles",
true);
2688 bool fPeriodicWrite =
2692 fCacheLarge || fCacheCritical ||
2693 fPeriodicWrite || fFlushForPrune;
2699 "Disk space is too low!",
2700 _(
"Disk space is too low!"));
2714 "%s: Failed to flush block file.\n",
2727 "Failed to write to block index database");
2732 if (fFlushForPrune) {
2739 if (!
CoinsTip().GetBestBlock().IsNull()) {
2741 LogWarning(
"Flushing large (%d GiB) UTXO set to disk, "
2742 "it may take several minutes\n",
2743 coins_mem_usage >> 30);
2747 "write coins cache to disk (%d coins, %.2fKiB)",
2748 coins_count, coins_mem_usage >> 10),
2761 "Disk space is too low!",
2762 _(
"Disk space is too low!"));
2768 fCacheLarge || fCacheCritical};
2772 "Failed to write to coin database");
2774 full_flush_completed =
true;
2776 int64_t{Ticks<std::chrono::microseconds>(
2777 SteadyClock::now() - nNow)},
2778 uint32_t(mode), coins_count,
2779 uint64_t(coins_mem_usage), fFlushForPrune);
2783 if (should_write ||
m_next_write == NodeClock::time_point::max()) {
2791 if (full_flush_completed) {
2796 }
catch (
const std::runtime_error &e) {
2798 std::string(
"System error while flushing: ") +
2807 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
2816 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
2823 const std::string &func_name,
2824 const std::string &
prefix)
2833 "%s%s: new best=%s height=%d version=0x%08x log2_work=%f tx=%ld "
2834 "date='%s' progress=%f cache=%.1fMiB(%utxo)\n",
2843void Chainstate::UpdateTip(
const CBlockIndex *pindexNew) {
2845 const auto &coins_tip =
CoinsTip();
2854 constexpr int BACKGROUND_LOG_INTERVAL = 2000;
2855 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2857 "[background validation] ");
2873 UpdateTipLog(coins_tip, pindexNew, params, __func__,
"");
2900 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2903 LogError(
"DisconnectTip(): Failed to read block\n");
2908 const auto time_start{SteadyClock::now()};
2912 if (DisconnectBlock(block, pindexDelete, view) !=
2914 LogError(
"DisconnectTip(): DisconnectBlock %s failed\n",
2919 bool flushed = view.
Flush();
2923 Ticks<MillisecondsDouble>(SteadyClock::now() - time_start));
2928 const int max_height_first{pindexDelete->
nHeight - 1};
2929 for (
auto &prune_lock :
m_blockman.m_prune_locks) {
2930 if (prune_lock.second.height_first <= max_height_first) {
2934 prune_lock.second.height_first = max_height_first;
2936 prune_lock.first, max_height_first);
2949 if (pindexDelete->
pprev !=
nullptr &&
2953 "Disconnecting mempool due to rewind of upgrade block\n");
2954 if (disconnectpool) {
2960 if (disconnectpool) {
2967 UpdateTip(pindexDelete->
pprev);
2987 const std::shared_ptr<const CBlock> &pblock,
3000 const auto time_1{SteadyClock::now()};
3001 std::shared_ptr<const CBlock> pthisBlock;
3003 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
3006 "Failed to read block");
3008 pthisBlock = pblockNew;
3010 pthisBlock = pblock;
3013 const CBlock &blockConnecting = *pthisBlock;
3016 const auto time_2{SteadyClock::now()};
3018 SteadyClock::time_point time_3;
3020 " - Load block from disk: %.2fms [%.2fs (%.2fms/blk)]\n",
3021 Ticks<MillisecondsDouble>(time_2 - time_1),
3028 bool rv =
ConnectBlock(blockConnecting, state, pindexNew, view,
3037 LogError(
"%s: ConnectBlock %s failed, %s\n", __func__,
3058 const Amount blockReward =
3062 std::vector<std::unique_ptr<ParkingPolicy>> parkingPolicies;
3063 parkingPolicies.emplace_back(std::make_unique<MinerFundPolicy>(
3064 consensusParams, *pindexNew, blockConnecting, blockReward));
3080 parkingPolicies.emplace_back(
3081 std::make_unique<RTTPolicy>(consensusParams,
3086 parkingPolicies.emplace_back(
3087 std::make_unique<StakingRewardsPolicy>(
3088 *
avalanche, consensusParams, *pindexNew,
3089 blockConnecting, blockReward));
3092 parkingPolicies.emplace_back(
3093 std::make_unique<PreConsensusPolicy>(
3094 *
avalanche, *pindexNew, blockConnecting,
3100 if (std::find_if_not(parkingPolicies.begin(), parkingPolicies.end(),
3101 [&](
const auto &policy) {
3102 bool ret = (*policy)(blockPolicyState);
3105 "Park block because it "
3106 "violated a block policy: %s\n",
3107 blockPolicyState.ToString());
3110 }) != parkingPolicies.end()) {
3111 pindexNew->nStatus = pindexNew->nStatus.withParked();
3117 time_3 = SteadyClock::now();
3121 BCLog::BENCH,
" - Connect total: %.2fms [%.2fs (%.2fms/blk)]\n",
3122 Ticks<MillisecondsDouble>(time_3 - time_2),
3125 bool flushed = view.
Flush();
3129 const auto time_4{SteadyClock::now()};
3132 Ticks<MillisecondsDouble>(time_4 - time_3),
3139 const auto time_5{SteadyClock::now()};
3142 " - Writing chainstate: %.2fms [%.2fs (%.2fms/blk)]\n",
3143 Ticks<MillisecondsDouble>(time_5 - time_4),
3148 disconnectpool.removeForBlock(blockConnecting.vtx, *m_mempool);
3153 if (pindexNew->
pprev !=
nullptr &&
3158 "Disconnecting mempool due to acceptance of upgrade block\n");
3159 disconnectpool.importMempool(*m_mempool);
3164 m_chain.SetTip(*pindexNew);
3165 UpdateTip(pindexNew);
3167 const auto time_6{SteadyClock::now()};
3171 " - Connect postprocess: %.2fms [%.2fs (%.2fms/blk)]\n",
3172 Ticks<MillisecondsDouble>(time_6 - time_5),
3176 Ticks<MillisecondsDouble>(time_6 - time_1),
3183 if (
this != &m_chainman.ActiveChainstate()) {
3187 m_chainman.MaybeCompleteSnapshotValidation();
3199 std::vector<const CBlockIndex *> &blocksToReconcile,
bool fAutoUnpark) {
3206 std::set<CBlockIndex *, CBlockIndexWorkComparator>::reverse_iterator
3218 if (m_avalancheFinalizedBlockIndex &&
3220 LogPrintf(
"Park block %s because it forks prior to the "
3221 "avalanche finalized chaintip.\n",
3223 pindexNew->nStatus = pindexNew->nStatus.withParked();
3234 bool hasValidAncestor =
true;
3235 while (hasValidAncestor && pindexTest && pindexTest != pindexFork) {
3240 bool fParkedChain = pindexTest->nStatus.isOnParkedChain();
3241 if (fAutoUnpark && fParkedChain) {
3247 if (!pindexTip || !pindexFork) {
3261 pindexExtraPow = pindexExtraPow->
pprev;
3266 requiredWork += (deltaWork >> 1);
3277 LogPrintf(
"Unpark chain up to block %s as it has "
3278 "accumulated enough PoW.\n",
3280 fParkedChain =
false;
3289 bool fInvalidChain = pindexTest->nStatus.isInvalid();
3290 bool fMissingData = !pindexTest->nStatus.hasData();
3291 if (!(fInvalidChain || fParkedChain || fMissingData)) {
3294 pindexTest = pindexTest->
pprev;
3300 hasValidAncestor =
false;
3303 if (fInvalidChain && (
m_chainman.m_best_invalid ==
nullptr ||
3309 if (fParkedChain && (
m_chainman.m_best_parked ==
nullptr ||
3315 LogPrintf(
"Considered switching to better tip %s but that chain "
3316 "contains a%s%s%s block.\n",
3318 fInvalidChain ?
"n invalid" :
"",
3319 fParkedChain ?
" parked" :
"",
3320 fMissingData ?
" missing-data" :
"");
3324 while (pindexTest != pindexFailed) {
3325 if (fInvalidChain || fParkedChain) {
3326 pindexFailed->nStatus =
3327 pindexFailed->nStatus.withFailedParent(fInvalidChain)
3328 .withParkedParent(fParkedChain);
3329 }
else if (fMissingData) {
3335 std::make_pair(pindexFailed->
pprev, pindexFailed));
3338 pindexFailed = pindexFailed->
pprev;
3341 if (fInvalidChain || fParkedChain) {
3348 blocksToReconcile.push_back(pindexNew);
3351 if (hasValidAncestor) {
3384 const std::shared_ptr<const CBlock> &pblock,
bool &fInvalidFound,
3396 bool fBlocksDisconnected =
false;
3399 if (
m_mempool && !fBlocksDisconnected) {
3418 "Failed to disconnect block; see debug.log for details");
3422 fBlocksDisconnected =
true;
3426 std::vector<CBlockIndex *> vpindexToConnect;
3427 bool fContinue =
true;
3432 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3433 vpindexToConnect.clear();
3434 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3437 vpindexToConnect.push_back(pindexIter);
3438 pindexIter = pindexIter->
pprev;
3446 if (!
ConnectTip(state, blockPolicyState, pindexConnect,
3447 pindexConnect == pindexMostWork
3449 : std::shared_ptr<const CBlock>(),
3450 disconnectpool,
avalanche, chainstate_role)) {
3458 fInvalidFound =
true;
3479 if (!pindexOldTip ||
3491 if (fBlocksDisconnected || !disconnectpool.
isEmpty()) {
3497 "Updating mempool due to reorganization or "
3498 "rules upgrade/downgrade\n");
3506 if (fInvalidFound) {
3527 bool fNotify =
false;
3528 bool fInitialBlockDownload =
false;
3533 pindexHeader = chainman.m_best_header;
3535 if (pindexHeader != pindexHeaderOld) {
3537 fInitialBlockDownload = chainman.IsInitialBlockDownload();
3538 pindexHeaderOld = pindexHeader;
3544 chainman.GetNotifications().headerTip(
3560 std::shared_ptr<const CBlock> pblock,
3581 LogPrintf(
"m_disabled is set - this chainstate should not be in "
3582 "operation. Please report this as a bug. %s\n",
3589 bool exited_ibd{
false};
3599 std::vector<const CBlockIndex *> blocksToReconcile;
3600 bool blocks_connected =
false;
3614 if (pindexMostWork ==
nullptr) {
3621 if (pindexMostWork ==
nullptr ||
3626 bool fInvalidFound =
false;
3627 std::shared_ptr<const CBlock> nullBlockPtr;
3634 state, pindexMostWork,
3635 pblock && pblock->GetHash() ==
3639 fInvalidFound,
avalanche, chainstate_role)) {
3643 blocks_connected =
true;
3645 if (fInvalidFound ||
3646 (pindexMostWork && pindexMostWork->nStatus.isParked())) {
3648 pindexMostWork =
nullptr;
3673 if (blocks_connected) {
3677 if (was_in_ibd && !still_in_ibd) {
3686 pindexFork != pindexNewTip) {
3702 return m_avalancheFinalizedBlockIndex);
3703 for (
const CBlockIndex *pindex : blocksToReconcile) {
3709 if (blocks_connected) {
3711 while (pindexTest && pindexTest != pfinalized) {
3716 avalanche->computeStakingReward(pindexTest);
3717 pindexTest = pindexTest->
pprev;
3723 if (!blocks_connected) {
3765 }
while (pindexNewTip != pindexMostWork);
3796 std::numeric_limits<int32_t>::min()) {
3818template <
typename Func>
struct Defer {
3820 Defer(
Func &&f) : func(
std::move(f)) {}
3821 ~Defer() { func(); }
3834 bool pindex_was_in_chain =
false;
3835 int disconnected = 0;
3851 std::multimap<const arith_uint256, CBlockIndex *> candidate_blocks_by_work;
3855 for (
auto &entry :
m_blockman.m_block_index) {
3866 candidate_blocks_by_work.insert(
3867 std::make_pair(candidate->
nChainWork, candidate));
3879 constexpr int maxDisconnectPoolBlocks = 10;
3931 pindex_was_in_chain =
true;
3940 if (optDisconnectPool && disconnected > maxDisconnectPoolBlocks) {
3945 optDisconnectPool =
nullptr;
3961 invalid_walk_tip->nStatus =
3962 invalidate ? invalid_walk_tip->nStatus.withFailed()
3963 : invalid_walk_tip->nStatus.withParked();
3968 if (invalid_walk_tip == to_mark_failed_or_parked->
pprev &&
3969 (invalidate ? to_mark_failed_or_parked->nStatus.hasFailed()
3970 : to_mark_failed_or_parked->nStatus.isParked())) {
3974 to_mark_failed_or_parked->nStatus =
3976 ? to_mark_failed_or_parked->nStatus.withFailed(
false)
3978 : to_mark_failed_or_parked->nStatus.withParked(
false)
3979 .withParkedParent());
3985 auto candidate_it = candidate_blocks_by_work.lower_bound(
3987 while (candidate_it != candidate_blocks_by_work.end()) {
3989 invalid_walk_tip->
pprev)) {
3991 candidate_it = candidate_blocks_by_work.erase(candidate_it);
4000 to_mark_failed_or_parked = invalid_walk_tip;
4016 to_mark_failed_or_parked->nStatus =
4017 invalidate ? to_mark_failed_or_parked->nStatus.withFailed()
4018 : to_mark_failed_or_parked->nStatus.withParked();
4031 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
4033 block_index.HaveNumChainTxs() &&
4046 if (pindex_was_in_chain) {
4049 *to_mark_failed_or_parked->
pprev);
4073template <
typename F>
4077 if (pindex->nStatus != newStatus &&
4080 pindex->nStatus = newStatus;
4096template <
typename F,
typename C,
typename AC>
4098 F f, C fChild, AC fAncestorWasChanged) {
4104 for (
auto pindexAncestor = pindex; pindexAncestor !=
nullptr;
4105 pindexAncestor = pindexAncestor->
pprev) {
4107 pindexDeepestChanged = pindexAncestor;
4113 pindexDeepestChanged) {
4115 pindexReset =
nullptr;
4119 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
4122 fAncestorWasChanged);
4126void Chainstate::SetBlockFailureFlags(
CBlockIndex *invalid_block) {
4129 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
4130 if (block_index.GetAncestor(invalid_block->
nHeight) == invalid_block &&
4131 !block_index.nStatus.isInvalid()) {
4132 block_index.nStatus = block_index.nStatus.withFailedParent();
4143 return status.withClearedFailureFlags();
4146 return status.withClearedFailureFlags();
4149 return status.withFailedParent(false);
4163 if (is_active_chainstate) {
4167 }
else if (!m_disabled) {
4173 if (snapshot_base->GetAncestor(pindex->
nHeight) == pindex) {
4185 return status.withClearedParkedFlags();
4188 return fClearChildren ? status.withClearedParkedFlags()
4189 : status.withParkedParent(false);
4192 return status.withParkedParent(false);
4204bool Chainstate::AvalancheFinalizeBlock(
CBlockIndex *pindex,
4215 "The block to mark finalized by avalanche is not on the "
4216 "active chain: %s\n",
4227 m_avalancheFinalizedBlockIndex = pindex;
4237 m_avalancheFinalizedBlockIndex =
nullptr;
4242 return pindex && m_avalancheFinalizedBlockIndex &&
4254 pindexNew->
nTx = block.
vtx.size();
4261 return block.nTx + (block.pprev ? block.pprev->nChainTx : 0);
4264 pindexNew->
nChainTx == prev_tx_sum(*pindexNew) ||
4265 pindexNew == GetSnapshotBaseBlock())) {
4266 LogPrintf(
"Internal bug detected: block %d has unexpected nChainTx %i "
4267 "that should be %i. Please report this issue here: %s\n",
4269 prev_tx_sum(*pindexNew), PACKAGE_BUGREPORT);
4273 pindexNew->nFile = pos.
nFile;
4274 pindexNew->nDataPos = pos.
nPos;
4275 pindexNew->nUndoPos = 0;
4276 pindexNew->nStatus = pindexNew->nStatus.withData();
4283 std::deque<CBlockIndex *> queue;
4284 queue.push_back(pindexNew);
4288 while (!queue.empty()) {
4296 pindex->
nChainTx == prev_tx_sum(*pindex))) {
4298 "Internal bug detected: block %d has unexpected nChainTx "
4299 "%i that should be %i. Please report this issue here: %s\n",
4303 pindex->
nChainTx = prev_tx_sum(*pindex);
4313 c->TryAddBlockIndexCandidate(pindex);
4316 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
4317 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
4319 while (range.first != range.second) {
4320 std::multimap<CBlockIndex *, CBlockIndex *>::iterator it =
4322 queue.push_back(it->second);
4327 }
else if (pindexNew->
pprev &&
4330 std::make_pair(pindexNew->
pprev, pindexNew));
4350 "high-hash",
"proof of work failed");
4367 "hashMerkleRoot mismatch");
4376 "bad-txns-duplicate",
4377 "duplicate transaction");
4409 if (block.
vtx.empty()) {
4411 "bad-cb-missing",
"first tx is not coinbase");
4420 "bad-blk-length",
"size limits failed");
4424 if (currentBlockSize > nMaxBlockSize) {
4426 "bad-blk-length",
"size limits failed");
4434 strprintf(
"Coinbase check failed (txid %s) %s",
4435 block.
vtx[0]->GetId().ToString(),
4441 for (
size_t i = 1; i < block.
vtx.size(); i++) {
4442 auto *tx = block.
vtx[i].get();
4447 strprintf(
"Transaction check failed (txid %s) %s",
4462 return std::all_of(headers.cbegin(), headers.cend(),
4463 [&](
const auto &header) {
4464 return CheckProofOfWork(
4465 header.GetHash(), header.nBits, consensusParams);
4473 "Block mutated: %s\n", state.
ToString());
4477 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase()) {
4485 return std::any_of(block.
vtx.begin(), block.
vtx.end(), [](
auto &tx) {
4486 return GetSerializeSize(tx, PROTOCOL_VERSION) == 64;
4520 const std::optional<CCheckpointData> &test_checkpoints = std::nullopt)
4523 assert(pindexPrev !=
nullptr);
4524 const int nHeight = pindexPrev->nHeight + 1;
4530 LogPrintf(
"bad bits after height: %d\n", pindexPrev->nHeight);
4532 "bad-diffbits",
"incorrect proof of work");
4536 if (chainman.m_options.checkpoints_enabled) {
4538 test_checkpoints ? test_checkpoints.value() : params.
Checkpoints();
4544 "ERROR: %s: rejected by checkpoint lock-in at %d\n",
4547 "checkpoint mismatch");
4555 blockman.GetLastCheckpoint(checkpoints);
4556 if (pcheckpoint && nHeight < pcheckpoint->
nHeight) {
4558 "ERROR: %s: forked chain older than last checkpoint "
4562 "bad-fork-prior-to-checkpoint");
4567 if (block.
GetBlockTime() <= pindexPrev->GetMedianTimePast()) {
4569 "time-too-old",
"block's timestamp is too early");
4573 if (block.
Time() > now + std::chrono::seconds{MAX_FUTURE_BLOCK_TIME}) {
4576 "block timestamp too far in the future");
4609 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4612 bool enforce_locktime_median_time_past{
false};
4615 assert(pindexPrev !=
nullptr);
4616 enforce_locktime_median_time_past =
true;
4619 const int64_t nMedianTimePast =
4622 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past
4627 const bool fIsMagneticAnomalyEnabled =
4634 const CTransaction *prevTx =
nullptr;
4635 for (
const auto &ptx : block.
vtx) {
4636 const CTransaction &tx = *ptx;
4637 if (fIsMagneticAnomalyEnabled) {
4638 if (prevTx && (tx.GetId() <= prevTx->GetId())) {
4639 if (tx.GetId() == prevTx->GetId()) {
4643 tx.GetId().ToString()));
4648 strprintf(
"Transaction order is invalid (%s < %s)",
4649 tx.GetId().ToString(),
4650 prevTx->GetId().ToString()));
4653 if (prevTx || !tx.IsCoinBase()) {
4671 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4673 block.
vtx[0]->vin[0].scriptSig.begin())) {
4676 "block height mismatch in coinbase");
4691 const std::optional<CCheckpointData> &test_checkpoints) {
4698 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4700 if (miSelf !=
m_blockman.m_block_index.end()) {
4707 if (pindex->nStatus.isInvalid()) {
4720 "%s: Consensus::CheckBlockHeader: %s, %s\n", __func__,
4726 BlockMap::iterator mi{
4730 "header %s has prev block not found: %s\n",
4733 "prev-blk-not-found");
4738 if (pindexPrev->nStatus.isInvalid()) {
4740 "header %s has prev block invalid: %s\n", hash.
ToString(),
4750 "%s: Consensus::ContextualCheckBlockHeader: %s, %s\n",
4781 if (pindexPrev->
GetAncestor(failedit->nHeight) == failedit) {
4782 assert(failedit->nStatus.hasFailed());
4784 while (invalid_walk != failedit) {
4785 invalid_walk->nStatus =
4786 invalid_walk->nStatus.withFailedParent();
4788 invalid_walk = invalid_walk->
pprev;
4791 "header %s has prev block invalid: %s\n",
4800 if (!min_pow_checked) {
4802 "%s: not adding new block header %s, missing anti-dos "
4803 "proof-of-work validation\n",
4806 "too-little-chainwork");
4822 const auto msg =
strprintf(
"Saw new header hash=%s height=%d",
4836 const std::vector<CBlockHeader> &headers,
bool min_pow_checked,
4838 const std::optional<CCheckpointData> &test_checkpoints) {
4846 header, state, &pindex, min_pow_checked, test_checkpoints);
4862 const int64_t blocks_left{
4863 (
GetTime() - last_accepted.GetBlockTime()) /
4865 const double progress{100.0 * last_accepted.nHeight /
4866 (last_accepted.nHeight + blocks_left)};
4867 LogPrintf(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n",
4868 last_accepted.nHeight, progress);
4876 int64_t timestamp) {
4885 if (m_best_header->nChainWork >=
4891 auto now = Now<SteadyMilliseconds>();
4892 if (now < m_last_presync_update + 250ms) {
4895 m_last_presync_update = now;
4899 height, timestamp,
true);
4900 if (initial_download) {
4901 const int64_t blocks_left{(
GetTime() - timestamp) /
4903 const double progress{100.0 * height / (height + blocks_left)};
4904 LogPrintf(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n",
4912 bool *fNewBlock,
bool min_pow_checked) {
4915 const CBlock &block = *pblock;
4922 bool accepted_header{
4926 if (!accepted_header) {
4934 bool fAlreadyHave = pindex->nStatus.hasData();
4946 int64_t chainTipTimeDiff =
4952 LogPrintf(
"Chain tip timestamp-to-received-time difference: hash=%s, "
4955 LogPrintf(
"New block timestamp-to-received-time difference: hash=%s, "
4960 bool fHasMoreOrSameWork =
4968 bool fTooFarAhead{pindex->
nHeight >
4981 if (pindex->
nTx != 0) {
4986 if (!fHasMoreOrSameWork) {
5010 pindex->nStatus = pindex->nStatus.withFailed();
5029 std::optional<int> snapshot_base_height = GetSnapshotBaseHeight();
5030 const bool is_background_block =
5032 pindex->
nHeight <= snapshot_base_height;
5034 if (!is_background_block && pindexFork &&
5036 LogPrintf(
"Park block %s as it would cause a deep reorg.\n",
5038 pindex->nStatus = pindex->nStatus.withParked();
5061 if (blockPos.IsNull()) {
5063 "%s: Failed to find position to write new block to disk",
5069 }
catch (
const std::runtime_error &e) {
5071 std::string(
"System error: ") + e.what());
5089 const std::shared_ptr<const CBlock> &block,
bool force_processing,
5090 bool min_pow_checked,
bool *new_block,
5119 ret =
AcceptBlock(block, state, force_processing,
nullptr,
5120 new_block, min_pow_checked);
5125 LogError(
"%s: AcceptBlock FAILED (%s)\n", __func__,
5136 LogError(
"%s: ActivateBestChain failed (%s)\n", __func__,
5142 ? m_ibd_chainstate.get()
5145 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
5146 LogError(
"%s: [background] ActivateBestChain failed (%s)\n", __func__,
5165 false, test_accept);
5181 indexDummy.
pprev = pindexPrev;
5188 adjusted_time_callback())) {
5189 LogError(
"%s: Consensus::ContextualCheckBlockHeader: %s\n", __func__,
5201 LogError(
"%s: Consensus::ContextualCheckBlock: %s\n", __func__,
5206 if (!chainstate.
ConnectBlock(block, state, &indexDummy, viewNew,
5207 validationOptions,
nullptr,
true)) {
5217 int nManualPruneHeight) {
5220 nManualPruneHeight)) {
5221 LogPrintf(
"%s: failed to flush state (%s)\n", __func__,
5227 FopenFn mockable_fopen_function) {
5257 "Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f\n",
5288 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
5292 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
5293 LogPrintf(
"Verifying last %i blocks at level %i\n", nCheckDepth,
5299 int nGoodTransactions = 0;
5302 bool skipped_no_block_data{
false};
5303 bool skipped_l3_checks{
false};
5304 LogPrintf(
"Verification progress: 0%%\n");
5309 pindex = pindex->
pprev) {
5310 const int percentageDone = std::max(
5311 1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() -
5313 (
double)nCheckDepth *
5314 (nCheckLevel >= 4 ? 50 : 100))));
5315 if (reportDone < percentageDone / 10) {
5317 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
5318 reportDone = percentageDone / 10;
5327 !pindex->nStatus.hasData()) {
5330 LogPrintf(
"VerifyDB(): block verification stopping at height %d "
5331 "(no data). This could be due to pruning or use of an "
5332 "assumeutxo snapshot.\n",
5334 skipped_no_block_data =
true;
5343 "Verification error: ReadBlockFromDisk failed at %d, hash=%s\n",
5349 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensusParams,
5352 "Verification error: found bad block at %d, hash=%s (%s)\n",
5359 if (nCheckLevel >= 2 && pindex) {
5363 LogPrintf(
"Verification error: found bad undo data at %d, "
5376 if (nCheckLevel >= 3) {
5380 chainstate.DisconnectBlock(block, pindex, coins);
5382 LogPrintf(
"Verification error: irrecoverable inconsistency "
5383 "in block data at %d, hash=%s\n",
5389 nGoodTransactions = 0;
5390 pindexFailure = pindex;
5392 nGoodTransactions += block.
vtx.size();
5395 skipped_l3_checks =
true;
5404 if (pindexFailure) {
5405 LogPrintf(
"Verification error: coin database inconsistencies found "
5406 "(last %i blocks, %i good transactions before that)\n",
5411 if (skipped_l3_checks) {
5412 LogPrintf(
"Skipped verification of level >=3 (insufficient database "
5413 "cache size). Consider increasing -dbcache.\n");
5420 if (nCheckLevel >= 4 && !skipped_l3_checks) {
5422 const int percentageDone = std::max(
5423 1, std::min(99, 100 -
int(
double(chainstate.
m_chain.
Height() -
5425 double(nCheckDepth) * 50)));
5426 if (reportDone < percentageDone / 10) {
5428 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
5429 reportDone = percentageDone / 10;
5436 LogPrintf(
"Verification error: ReadBlockFromDisk failed at %d, "
5441 if (!chainstate.
ConnectBlock(block, state, pindex, coins,
5443 LogPrintf(
"Verification error: found unconnectable block at "
5444 "%d, hash=%s (%s)\n",
5455 LogPrintf(
"Verification: No coin database inconsistencies in last %i "
5456 "blocks (%i transactions)\n",
5457 block_count, nGoodTransactions);
5459 if (skipped_l3_checks) {
5462 if (skipped_no_block_data) {
5478 LogError(
"ReplayBlock(): ReadBlockFromDisk failed at %d, hash=%s\n",
5489 if (tx->IsCoinBase()) {
5493 for (
const CTxIn &txin : tx->vin) {
5507 std::vector<BlockHash> hashHeads =
db.GetHeadBlocks();
5508 if (hashHeads.empty()) {
5512 if (hashHeads.size() != 2) {
5513 LogError(
"ReplayBlocks(): unknown inconsistent state\n");
5527 if (
m_blockman.m_block_index.count(hashHeads[0]) == 0) {
5528 LogError(
"ReplayBlocks(): reorganization to unknown block requested\n");
5532 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
5534 if (!hashHeads[1].IsNull()) {
5536 if (
m_blockman.m_block_index.count(hashHeads[1]) == 0) {
5537 LogError(
"ReplayBlocks(): reorganization from unknown block "
5542 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
5544 assert(pindexFork !=
nullptr);
5548 while (pindexOld != pindexFork) {
5553 LogError(
"RollbackBlock(): ReadBlockFromDisk() failed at "
5565 "RollbackBlock(): DisconnectBlock failed at %d, hash=%s\n",
5578 pindexOld = pindexOld->
pprev;
5582 int nForkHeight = pindexFork ? pindexFork->
nHeight : 0;
5589 _(
"Replaying blocks…"),
5590 (
int)((
nHeight - nForkHeight) * 100.0 /
5591 (pindexNew->
nHeight - nForkHeight)),
5606void Chainstate::ClearBlockIndexCandidates() {
5622 const uint64_t numHeaders{20};
5625 const fs::path filePathTmp = filePath +
".new";
5636 bool missingIndex{
false};
5637 for (uint64_t i = 0; i < numHeaders; i++) {
5639 LogPrintf(
"Missing block index, stopping the headers time "
5640 "dumping after %d blocks.\n",
5642 missingIndex =
true;
5649 index = index->
pprev;
5653 throw std::runtime_error(
strprintf(
"Failed to commit to file %s",
5659 fs::remove(filePathTmp);
5664 throw std::runtime_error(
strprintf(
"Rename failed from %s to %s",
5668 }
catch (
const std::exception &e) {
5669 LogPrintf(
"Failed to dump the headers time: %s.\n", e.what());
5673 LogPrintf(
"Successfully dumped the last %d headers time to %s.\n",
5689 LogPrintf(
"Failed to open header times from disk, skipping.\n");
5698 LogPrintf(
"Unsupported header times file version, skipping.\n");
5705 for (uint64_t i = 0; i < numBlocks; i++) {
5707 int64_t receiveTime;
5710 file >> receiveTime;
5714 LogPrintf(
"Missing index for block %s, stopping the headers "
5715 "time loading after %d blocks.\n",
5722 }
catch (
const std::exception &e) {
5723 LogPrintf(
"Failed to read the headers time file data on disk: %s.\n",
5741 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
5743 std::vector<CBlockIndex *> vSortedByHeight{
5745 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
5757 if (pindex == GetSnapshotBaseBlock() ||
5765 if (pindex->nStatus.isInvalid() &&
5767 pindex->
nChainWork > m_best_invalid->nChainWork)) {
5768 m_best_invalid = pindex;
5771 if (pindex->nStatus.isOnParkedChain() &&
5773 pindex->
nChainWork > m_best_parked->nChainWork)) {
5774 m_best_parked = pindex;
5778 (m_best_header ==
nullptr ||
5780 m_best_header = pindex;
5784 needs_init =
m_blockman.m_block_index.empty();
5794 LogPrintf(
"Initializing databases...\n");
5815 if (blockPos.IsNull()) {
5816 LogError(
"%s: writing genesis block to disk failed\n", __func__);
5822 }
catch (
const std::runtime_error &e) {
5823 LogError(
"%s: failed to write genesis block: %s\n", __func__, e.what());
5832 std::multimap<BlockHash, FlatFilePos> *blocks_with_unknown_parent,
5835 assert(!dbp == !blocks_with_unknown_parent);
5849 uint64_t nRewind = blkdat.
GetPos();
5850 while (!blkdat.
eof()) {
5860 unsigned int nSize = 0;
5865 nRewind = blkdat.
GetPos() + 1;
5867 if (memcmp(buf, params.
DiskMagic().data(),
5877 }
catch (
const std::exception &) {
5885 const uint64_t nBlockPos{blkdat.
GetPos()};
5887 dbp->
nPos = nBlockPos;
5889 blkdat.
SetLimit(nBlockPos + nSize);
5897 nRewind = nBlockPos + nSize;
5902 std::shared_ptr<CBlock> pblock{};
5911 "%s: Out of order block %s, parent %s not known\n",
5912 __func__, hash.ToString(),
5914 if (dbp && blocks_with_unknown_parent) {
5915 blocks_with_unknown_parent->emplace(
5924 if (!pindex || !pindex->nStatus.hasData()) {
5927 blkdat.
SetPos(nBlockPos);
5928 pblock = std::make_shared<CBlock>();
5930 nRewind = blkdat.
GetPos();
5933 if (
AcceptBlock(pblock, state,
true, dbp,
nullptr,
5941 pindex->
nHeight % 1000 == 0) {
5944 "Block Import: already had block %s at height %d\n",
5945 hash.ToString(), pindex->
nHeight);
5952 bool genesis_activation_failure =
false;
5953 for (
auto c :
GetAll()) {
5955 if (!c->ActivateBestChain(state,
nullptr,
avalanche)) {
5956 genesis_activation_failure =
true;
5960 if (genesis_activation_failure) {
5974 bool activation_failure =
false;
5975 for (
auto c :
GetAll()) {
5977 if (!c->ActivateBestChain(state, pblock,
avalanche)) {
5979 "failed to activate chain (%s)\n",
5981 activation_failure =
true;
5985 if (activation_failure) {
5992 if (!blocks_with_unknown_parent) {
5998 std::deque<BlockHash> queue;
5999 queue.push_back(hash);
6000 while (!queue.empty()) {
6003 auto range = blocks_with_unknown_parent->equal_range(head);
6004 while (range.first != range.second) {
6005 std::multimap<BlockHash, FlatFilePos>::iterator it =
6007 std::shared_ptr<CBlock> pblockrecursive =
6008 std::make_shared<CBlock>();
6013 "%s: Processing out of order child %s of %s\n",
6014 __func__, pblockrecursive->GetHash().ToString(),
6019 &it->second,
nullptr,
true)) {
6021 queue.push_back(pblockrecursive->GetHash());
6025 blocks_with_unknown_parent->erase(it);
6029 }
catch (
const std::exception &e) {
6050 "%s: unexpected data at file offset 0x%x - %s. "
6052 __func__, (nRewind - 1), e.what());
6055 }
catch (
const std::runtime_error &e) {
6059 LogPrintf(
"Loaded %i blocks from external file in %dms\n", nLoaded,
6080 std::multimap<CBlockIndex *, CBlockIndex *> forward;
6081 for (
auto &[
_, block_index] :
m_blockman.m_block_index) {
6082 forward.emplace(block_index.pprev, &block_index);
6087 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6088 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6089 rangeGenesis = forward.equal_range(
nullptr);
6091 rangeGenesis.first++;
6093 assert(rangeGenesis.first == rangeGenesis.second);
6115 CBlockIndex *pindexFirstNotTransactionsValid =
nullptr;
6121 CBlockIndex *pindexFirstNotScriptsValid =
nullptr;
6127 const CBlockIndex *snap_base{GetSnapshotBaseBlock()};
6128 CBlockIndex *snap_first_missing{}, *snap_first_notx{}, *snap_first_notv{},
6129 *snap_first_nocv{}, *snap_first_nosv{};
6130 auto snap_update_firsts = [&] {
6131 if (pindex == snap_base) {
6132 std::swap(snap_first_missing, pindexFirstMissing);
6133 std::swap(snap_first_notx, pindexFirstNeverProcessed);
6134 std::swap(snap_first_notv, pindexFirstNotTransactionsValid);
6135 std::swap(snap_first_nocv, pindexFirstNotChainValid);
6136 std::swap(snap_first_nosv, pindexFirstNotScriptsValid);
6140 while (pindex !=
nullptr) {
6142 if (pindexFirstInvalid ==
nullptr && pindex->nStatus.hasFailed()) {
6143 pindexFirstInvalid = pindex;
6145 if (pindexFirstParked ==
nullptr && pindex->nStatus.isParked()) {
6146 pindexFirstParked = pindex;
6148 if (pindexFirstMissing ==
nullptr && !pindex->nStatus.hasData()) {
6149 pindexFirstMissing = pindex;
6151 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) {
6152 pindexFirstNeverProcessed = pindex;
6154 if (pindex->
pprev !=
nullptr && pindexFirstNotTreeValid ==
nullptr &&
6156 pindexFirstNotTreeValid = pindex;
6158 if (pindex->
pprev !=
nullptr) {
6159 if (pindexFirstNotTransactionsValid ==
nullptr &&
6161 pindexFirstNotTransactionsValid = pindex;
6163 if (pindexFirstNotChainValid ==
nullptr &&
6165 pindexFirstNotChainValid = pindex;
6167 if (pindexFirstNotScriptsValid ==
nullptr &&
6169 pindexFirstNotScriptsValid = pindex;
6174 if (pindex->
pprev ==
nullptr) {
6178 for (
auto c :
GetAll()) {
6179 if (c->m_chain.Genesis() !=
nullptr) {
6181 assert(pindex == c->m_chain.Genesis());
6196 assert(pindex->nStatus.hasData() == (pindex->
nTx > 0));
6197 assert(pindexFirstMissing == pindexFirstNeverProcessed);
6198 }
else if (pindex->nStatus.hasData()) {
6203 if (pindex->nStatus.hasUndo()) {
6204 assert(pindex->nStatus.hasData());
6206 if (snap_base && snap_base->GetAncestor(pindex->
nHeight) == pindex) {
6218 assert((pindexFirstNeverProcessed ==
nullptr || pindex == snap_base) ==
6220 assert((pindexFirstNotTransactionsValid ==
nullptr ||
6232 assert(pindexFirstNotTreeValid ==
nullptr);
6235 assert(pindexFirstNotTreeValid ==
nullptr);
6239 assert(pindexFirstNotChainValid ==
nullptr);
6243 assert(pindexFirstNotScriptsValid ==
nullptr);
6245 if (pindexFirstInvalid ==
nullptr) {
6248 assert(!pindex->nStatus.isInvalid());
6250 if (pindexFirstParked ==
nullptr) {
6254 assert(!pindex->nStatus.isOnParkedChain());
6257 if (!pindex->
pprev) {
6271 for (
auto c :
GetAll()) {
6272 if (c->m_chain.Tip() ==
nullptr) {
6289 (pindexFirstNeverProcessed ==
nullptr || pindex == snap_base)) {
6293 if (pindexFirstInvalid ==
nullptr) {
6300 GetSnapshotBaseBlock()->GetAncestor(pindex->
nHeight) ==
6314 if (pindexFirstMissing ==
nullptr) {
6315 assert(pindex->nStatus.isOnParkedChain() ||
6316 c->setBlockIndexCandidates.count(pindex));
6324 if (pindex == c->m_chain.Tip() ||
6325 pindex == c->SnapshotBase()) {
6326 assert(c->setBlockIndexCandidates.count(pindex));
6338 assert(c->setBlockIndexCandidates.count(pindex) == 0);
6342 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6343 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6346 bool foundInUnlinked =
false;
6347 while (rangeUnlinked.first != rangeUnlinked.second) {
6348 assert(rangeUnlinked.first->first == pindex->
pprev);
6349 if (rangeUnlinked.first->second == pindex) {
6350 foundInUnlinked =
true;
6353 rangeUnlinked.first++;
6355 if (pindex->
pprev && pindex->nStatus.hasData() &&
6356 pindexFirstNeverProcessed !=
nullptr &&
6357 pindexFirstInvalid ==
nullptr) {
6363 if (!pindex->nStatus.hasData()) {
6365 assert(!foundInUnlinked);
6367 if (pindexFirstMissing ==
nullptr) {
6370 assert(!foundInUnlinked);
6372 if (pindex->
pprev && pindex->nStatus.hasData() &&
6373 pindexFirstNeverProcessed ==
nullptr &&
6374 pindexFirstMissing !=
nullptr) {
6387 for (
auto c :
GetAll()) {
6390 c->setBlockIndexCandidates.count(pindex) == 0) {
6391 if (pindexFirstInvalid ==
nullptr) {
6393 snap_base->GetAncestor(pindex->
nHeight) == pindex) {
6405 snap_update_firsts();
6406 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6407 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6408 range = forward.equal_range(pindex);
6409 if (range.first != range.second) {
6411 pindex = range.first->second;
6419 snap_update_firsts();
6422 if (pindex == pindexFirstInvalid) {
6423 pindexFirstInvalid =
nullptr;
6425 if (pindex == pindexFirstParked) {
6426 pindexFirstParked =
nullptr;
6428 if (pindex == pindexFirstMissing) {
6429 pindexFirstMissing =
nullptr;
6431 if (pindex == pindexFirstNeverProcessed) {
6432 pindexFirstNeverProcessed =
nullptr;
6434 if (pindex == pindexFirstNotTreeValid) {
6435 pindexFirstNotTreeValid =
nullptr;
6437 if (pindex == pindexFirstNotTransactionsValid) {
6438 pindexFirstNotTransactionsValid =
nullptr;
6440 if (pindex == pindexFirstNotChainValid) {
6441 pindexFirstNotChainValid =
nullptr;
6443 if (pindex == pindexFirstNotScriptsValid) {
6444 pindexFirstNotScriptsValid =
nullptr;
6449 std::pair<std::multimap<CBlockIndex *, CBlockIndex *>::iterator,
6450 std::multimap<CBlockIndex *, CBlockIndex *>::iterator>
6451 rangePar = forward.equal_range(pindexPar);
6452 while (rangePar.first->second != pindex) {
6455 assert(rangePar.first != rangePar.second);
6460 if (rangePar.first != rangePar.second) {
6462 pindex = rangePar.first->second;
6474 assert(nNodes == forward.size());
6480 return strprintf(
"Chainstate [%s] @ height %d (%s)",
6486bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size) {
6499 coinsdb_size * (1.0 / 1024 / 1024));
6501 coinstip_size * (1.0 / 1024 / 1024));
6506 if (coinstip_size > old_coinstip_size) {
6523 if (pindex ==
nullptr) {
6528 "Block %d has unset m_chain_tx_count. Unable to "
6529 "estimate verification progress.\n",
6534 int64_t nNow = time(
nullptr);
6549 if (m_active_chainstate && m_active_chainstate->m_from_snapshot_blockhash) {
6551 return m_active_chainstate->m_from_snapshot_blockhash;
6553 return std::nullopt;
6558 std::vector<Chainstate *> out;
6561 {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
6563 out.push_back(pchainstate);
6572 assert(!m_ibd_chainstate);
6573 assert(!m_active_chainstate);
6575 m_ibd_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman, *
this);
6576 m_active_chainstate = m_ibd_chainstate.get();
6577 return *m_active_chainstate;
6590 const bool existed{fs::remove(base_blockhash_path)};
6592 LogPrintf(
"[snapshot] snapshot chainstate dir being removed "
6596 }
catch (
const fs::filesystem_error &e) {
6597 LogPrintf(
"[snapshot] failed to remove file %s: %s\n",
6604 LogPrintf(
"Removing leveldb dir at %s\n", path_str);
6608 const bool destroyed = dbwrapper::DestroyDB(path_str, {}).ok();
6611 LogPrintf(
"error: leveldb DestroyDB call failed on %s\n", path_str);
6629 "Can't activate a snapshot-based chainstate more than once")};
6637 if (!
GetParams().AssumeutxoForBlockhash(base_blockhash).has_value()) {
6639 std::string heights_formatted =
6640 Join(available_heights,
", ",
6641 [&](
const auto &i) {
return ToString(i); });
6643 Untranslated(
"assumeutxo block hash in snapshot metadata not "
6644 "recognized (hash: %s). The following "
6645 "snapshot heights are available: %s."),
6646 base_blockhash.
ToString(), heights_formatted)};
6650 if (!snapshot_start_block) {
6652 Untranslated(
"The base block header (%s) must appear in the "
6653 "headers chain. Make sure all headers are "
6654 "syncing, and call loadtxoutset again."),
6658 if (snapshot_start_block->nStatus.isInvalid()) {
6661 "The base block header (%s) is part of an invalid chain"),
6665 if (!m_best_header ||
6666 m_best_header->GetAncestor(snapshot_start_block->nHeight) !=
6667 snapshot_start_block) {
6669 "A forked headers-chain with more work than the chain with the "
6670 "snapshot base block header exists. Please proceed to sync "
6671 "without AssumeUtxo.")};
6674 if (
Assert(m_active_chainstate->GetMempool())->size() > 0) {
6676 "Can't activate a snapshot when mempool not empty.")};
6680 int64_t current_coinsdb_cache_size{0};
6681 int64_t current_coinstip_cache_size{0};
6690 static constexpr double IBD_CACHE_PERC = 0.01;
6691 static constexpr double SNAPSHOT_CACHE_PERC = 0.99;
6704 current_coinsdb_cache_size =
6706 current_coinstip_cache_size =
6712 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
6713 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
6716 auto snapshot_chainstate =
6723 snapshot_chainstate->InitCoinsDB(
6724 static_cast<size_t>(current_coinsdb_cache_size *
6725 SNAPSHOT_CACHE_PERC),
6726 in_memory,
false,
"chainstate");
6727 snapshot_chainstate->InitCoinsCache(
static_cast<size_t>(
6728 current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
6731 auto cleanup_bad_snapshot =
6733 this->MaybeRebalanceCaches();
6738 if (
auto snapshot_datadir =
6744 snapshot_chainstate.reset();
6749 "Failed to remove snapshot chainstate dir (%s). "
6750 "Manually remove it before restarting.\n",
6760 return cleanup_bad_snapshot(
Untranslated(
"population failed"));
6771 snapshot_chainstate->m_chain.Tip())) {
6772 return cleanup_bad_snapshot(
6773 Untranslated(
"work does not exceed active chainstate"));
6779 return cleanup_bad_snapshot(
6784 assert(!m_snapshot_chainstate);
6785 m_snapshot_chainstate.swap(snapshot_chainstate);
6786 const bool chaintip_loaded = m_snapshot_chainstate->LoadChainTip();
6791 Assert(m_active_chainstate->m_mempool->size() == 0);
6792 Assert(!m_snapshot_chainstate->m_mempool);
6793 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
6794 m_active_chainstate->m_mempool =
nullptr;
6795 m_active_chainstate = m_snapshot_chainstate.get();
6798 LogPrintf(
"[snapshot] successfully activated snapshot %s\n",
6801 m_snapshot_chainstate->CoinsTip().DynamicMemoryUsage() /
6804 this->MaybeRebalanceCaches();
6805 return snapshot_start_block;
6809 bool snapshot_loaded) {
6812 snapshot_loaded ?
"saving snapshot chainstate"
6813 :
"flushing coins cache",
6815 BCLog::LogFlags::ALL);
6817 coins_cache.
Flush();
6821 const char *
what()
const throw()
override {
6822 return "ComputeUTXOStats interrupted by shutdown.";
6846 if (!snapshot_start_block) {
6849 LogPrintf(
"[snapshot] Did not find snapshot start blockheader %s\n",
6854 int base_height = snapshot_start_block->
nHeight;
6857 if (!maybe_au_data) {
6858 LogPrintf(
"[snapshot] assumeutxo height in snapshot metadata not "
6859 "recognized (%d) - refusing to load snapshot\n",
6871 LogPrintf(
"[snapshot] activation failed - work does not exceed active "
6879 LogPrintf(
"[snapshot] loading %d coins from snapshot %s\n", coins_left,
6881 int64_t coins_processed{0};
6883 while (coins_left > 0) {
6887 size_t coins_per_txid{0};
6890 if (coins_per_txid > coins_left) {
6891 LogPrintf(
"[snapshot] mismatch in coins count in snapshot "
6892 "metadata and actual snapshot data\n");
6896 for (
size_t i = 0; i < coins_per_txid; i++) {
6902 if (coin.
GetHeight() > uint32_t(base_height) ||
6904 std::numeric_limits<
decltype(outpoint.GetN())>::max()) {
6905 LogPrintf(
"[snapshot] bad snapshot data after "
6906 "deserializing %d coins\n",
6907 coins_count - coins_left);
6911 LogPrintf(
"[snapshot] bad snapshot data after "
6912 "deserializing %d coins - bad tx out value\n",
6913 coins_count - coins_left);
6922 if (coins_processed % 1000000 == 0) {
6923 LogPrintf(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)\n",
6925 static_cast<float>(coins_processed) * 100 /
6926 static_cast<float>(coins_count),
6934 if (coins_processed % 120000 == 0) {
6939 const auto snapshot_cache_state =
WITH_LOCK(
6941 return snapshot_chainstate.GetCoinsCacheSizeState());
6958 }
catch (
const std::ios_base::failure &) {
6959 LogPrintf(
"[snapshot] bad snapshot format or truncated snapshot "
6960 "after deserializing %d coins\n",
6973 bool out_of_coins{
false};
6977 }
catch (
const std::ios_base::failure &) {
6979 out_of_coins =
true;
6981 if (!out_of_coins) {
6982 LogPrintf(
"[snapshot] bad snapshot - coins left over after "
6983 "deserializing %d coins\n",
6988 LogPrintf(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s\n",
7002 std::optional<CCoinsStats> maybe_stats;
7006 CoinStatsHashType::HASH_SERIALIZED, snapshot_coinsdb,
m_blockman,
7013 if (!maybe_stats.has_value()) {
7014 LogPrintf(
"[snapshot] failed to generate coins stats\n");
7022 LogPrintf(
"[snapshot] bad snapshot content hash: expected %s, got %s\n",
7024 maybe_stats->hashSerialized.ToString());
7040 constexpr int AFTER_GENESIS_START{1};
7042 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height();
7044 index = snapshot_chainstate.
m_chain[i];
7055 assert(index == snapshot_start_block);
7059 LogPrintf(
"[snapshot] validated snapshot (%.2f MB)\n",
7081 if (m_ibd_chainstate.get() == &this->ActiveChainstate() ||
7082 !this->IsUsable(m_snapshot_chainstate.get()) ||
7083 !this->IsUsable(m_ibd_chainstate.get()) ||
7084 !m_ibd_chainstate->m_chain.Tip()) {
7090 const int snapshot_base_height = *
Assert(this->GetSnapshotBaseHeight());
7093 if (index_new.
nHeight < snapshot_base_height) {
7103 _(
"%s failed to validate the -assumeutxo snapshot state. "
7104 "This indicates a hardware problem, or a bug in the software, or "
7105 "a bad software modification that allowed an invalid snapshot to "
7106 "be loaded. As a result of this, the node will shut down and "
7107 "stop using any state that was built on the snapshot, resetting "
7108 "the chain height from %d to %d. On the next restart, the node "
7109 "will resume syncing from %d without using any snapshot data. "
7110 "Please report this incident to %s, including how you obtained "
7111 "the snapshot. The invalid snapshot chainstate will be left on "
7112 "disk in case it is helpful in diagnosing the issue that caused "
7114 PACKAGE_NAME, snapshot_tip_height, snapshot_base_height,
7115 snapshot_base_height, PACKAGE_BUGREPORT);
7118 LogPrintf(
"[snapshot] deleting snapshot, reverting to validated chain, "
7119 "and stopping node\n");
7121 m_active_chainstate = m_ibd_chainstate.get();
7122 m_snapshot_chainstate->m_disabled =
true;
7126 auto rename_result = m_snapshot_chainstate->InvalidateCoinsDBOnDisk();
7127 if (!rename_result) {
7137 "[snapshot] supposed base block %s does not match the "
7138 "snapshot base block %s (height %d). Snapshot is not valid.\n",
7140 snapshot_base_height);
7141 handle_invalid_snapshot();
7147 int curr_height = m_ibd_chainstate->m_chain.Height();
7149 assert(snapshot_base_height == curr_height);
7154 CCoinsViewDB &ibd_coins_db = m_ibd_chainstate->CoinsDB();
7155 m_ibd_chainstate->ForceFlushStateToDisk();
7157 const auto &maybe_au_data =
7159 if (!maybe_au_data) {
7160 LogPrintf(
"[snapshot] assumeutxo data not found for height "
7161 "(%d) - refusing to validate snapshot\n",
7163 handle_invalid_snapshot();
7168 std::optional<CCoinsStats> maybe_ibd_stats;
7170 "[snapshot] computing UTXO stats for background chainstate to validate "
7171 "snapshot - this could take a few minutes\n");
7182 if (!maybe_ibd_stats) {
7184 "[snapshot] failed to generate stats for validation coins db\n");
7188 handle_invalid_snapshot();
7191 const auto &ibd_stats = *maybe_ibd_stats;
7200 LogPrintf(
"[snapshot] hash mismatch: actual=%s, expected=%s\n",
7201 ibd_stats.hashSerialized.ToString(),
7203 handle_invalid_snapshot();
7207 LogPrintf(
"[snapshot] snapshot beginning at %s has been fully validated\n",
7210 m_ibd_chainstate->m_disabled =
true;
7211 this->MaybeRebalanceCaches();
7218 assert(m_active_chainstate);
7219 return *m_active_chainstate;
7224 LOCK(active_chainstate.cs_avalancheFinalizedBlockIndex);
7225 return active_chainstate.m_avalancheFinalizedBlockIndex;
7230 return m_snapshot_chainstate &&
7231 m_active_chainstate == m_snapshot_chainstate.get();
7233void ChainstateManager::MaybeRebalanceCaches() {
7235 bool ibd_usable = this->
IsUsable(m_ibd_chainstate.get());
7236 bool snapshot_usable = this->
IsUsable(m_snapshot_chainstate.get());
7237 assert(ibd_usable || snapshot_usable);
7239 if (ibd_usable && !snapshot_usable) {
7240 LogPrintf(
"[snapshot] allocating all cache to the IBD chainstate\n");
7244 }
else if (snapshot_usable && !ibd_usable) {
7248 "[snapshot] allocating all cache to the snapshot chainstate\n");
7252 }
else if (ibd_usable && snapshot_usable) {
7261 m_snapshot_chainstate->ResizeCoinsCaches(
7264 m_snapshot_chainstate->ResizeCoinsCaches(
7272void ChainstateManager::ResetChainstates() {
7273 m_ibd_chainstate.reset();
7274 m_snapshot_chainstate.reset();
7275 m_active_chainstate =
nullptr;
7284 if (!opts.check_block_index.has_value()) {
7286 opts.config.GetChainParams().DefaultConsistencyChecks();
7289 if (!opts.minimum_chain_work.has_value()) {
7291 opts.config.GetChainParams().GetConsensus().nMinimumChainWork);
7293 if (!opts.assumed_valid_block.has_value()) {
7294 opts.assumed_valid_block =
7295 opts.config.GetChainParams().GetConsensus().defaultAssumeValid;
7297 Assert(opts.adjusted_time_callback);
7298 return std::move(opts);
7304 : m_interrupt{interrupt}, m_options{
Flatten(
std::move(options))},
7305 m_blockman{interrupt,
std::move(blockman_options)},
7306 m_validation_cache{m_options.script_execution_cache_bytes,
7307 m_options.signature_cache_bytes} {}
7309bool ChainstateManager::DetectSnapshotChainstate(
CTxMemPool *mempool) {
7310 assert(!m_snapshot_chainstate);
7311 std::optional<fs::path> path =
7316 std::optional<BlockHash> base_blockhash =
7318 if (!base_blockhash) {
7321 LogPrintf(
"[snapshot] detected active snapshot chainstate (%s) - loading\n",
7324 this->ActivateExistingSnapshot(*base_blockhash);
7329ChainstateManager::ActivateExistingSnapshot(
BlockHash base_blockhash) {
7330 assert(!m_snapshot_chainstate);
7331 m_snapshot_chainstate = std::make_unique<Chainstate>(
nullptr,
m_blockman,
7332 *
this, base_blockhash);
7333 LogPrintf(
"[snapshot] switching active chainstate to %s\n",
7334 m_snapshot_chainstate->ToString());
7337 Assert(m_active_chainstate->m_mempool->size() == 0);
7338 Assert(!m_snapshot_chainstate->m_mempool);
7339 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
7340 m_active_chainstate->m_mempool =
nullptr;
7341 m_active_chainstate = m_snapshot_chainstate.get();
7342 return *m_snapshot_chainstate;
7349 assert(
cs.m_from_snapshot_blockhash);
7350 auto storage_path_maybe =
cs.CoinsDB().StoragePath();
7352 assert(storage_path_maybe);
7353 return *storage_path_maybe;
7362 auto invalid_path = snapshot_datadir +
"_INVALID";
7365 LogPrintf(
"[snapshot] renaming snapshot datadir %s to %s\n", dbpath,
7372 fs::rename(snapshot_datadir, invalid_path);
7373 }
catch (
const fs::filesystem_error &e) {
7377 LogPrintf(
"%s: error renaming file '%s' -> '%s': %s\n", __func__,
7378 src_str, dest_str, e.what());
7380 "You should resolve this by manually "
7381 "moving or deleting the invalid "
7382 "snapshot directory %s, otherwise you "
7383 "will encounter the same error again "
7384 "on the next startup."),
7385 src_str, dest_str, src_str)};
7390bool ChainstateManager::DeleteSnapshotChainstate() {
7392 Assert(m_snapshot_chainstate);
7393 Assert(m_ibd_chainstate);
7398 LogPrintf(
"Deletion of %s failed. Please remove it manually to "
7399 "continue reindexing.\n",
7403 m_active_chainstate = m_ibd_chainstate.get();
7404 m_active_chainstate->m_mempool = m_snapshot_chainstate->m_mempool;
7405 m_snapshot_chainstate.reset();
7417const CBlockIndex *ChainstateManager::GetSnapshotBaseBlock()
const {
7418 return m_active_chainstate ? m_active_chainstate->SnapshotBase() :
nullptr;
7421std::optional<int> ChainstateManager::GetSnapshotBaseHeight()
const {
7422 const CBlockIndex *base = this->GetSnapshotBaseBlock();
7423 return base ? std::make_optional(base->
nHeight) :
std::nullopt;
7426void ChainstateManager::RecalculateBestHeader() {
7429 for (
auto &entry :
m_blockman.m_block_index) {
7430 if (!(entry.second.nStatus.isInvalid()) &&
7431 m_best_header->nChainWork < entry.second.nChainWork) {
7432 m_best_header = &entry.second;
7437bool ChainstateManager::ValidatedSnapshotCleanup() {
7446 std::optional<fs::path> ibd_chainstate_path_maybe =
7447 get_storage_path(m_ibd_chainstate);
7448 std::optional<fs::path> snapshot_chainstate_path_maybe =
7449 get_storage_path(m_snapshot_chainstate);
7458 if (!ibd_chainstate_path_maybe || !snapshot_chainstate_path_maybe) {
7459 LogPrintf(
"[snapshot] snapshot chainstate cleanup cannot happen with "
7460 "in-memory chainstates. You are testing, right?\n");
7464 const auto &snapshot_chainstate_path = *snapshot_chainstate_path_maybe;
7465 const auto &ibd_chainstate_path = *ibd_chainstate_path_maybe;
7473 this->ResetChainstates();
7478 LogPrintf(
"[snapshot] deleting background chainstate directory (now "
7479 "unnecessary) (%s)\n",
7482 fs::path tmp_old{ibd_chainstate_path +
"_todelete"};
7485 const fs::filesystem_error &err) {
7489 "Rename of '%s' -> '%s' failed. "
7490 "Cannot clean up the background chainstate leveldb directory.",
7495 fs::rename(ibd_chainstate_path, tmp_old);
7496 }
catch (
const fs::filesystem_error &e) {
7497 rename_failed_abort(ibd_chainstate_path, tmp_old, e);
7501 LogPrintf(
"[snapshot] moving snapshot chainstate (%s) to "
7502 "default chainstate directory (%s)\n",
7507 fs::rename(snapshot_chainstate_path, ibd_chainstate_path);
7508 }
catch (
const fs::filesystem_error &e) {
7509 rename_failed_abort(snapshot_chainstate_path, ibd_chainstate_path, e);
7516 LogPrintf(
"Deletion of %s failed. Please remove it manually, as the "
7517 "directory is now unnecessary.\n",
7520 LogPrintf(
"[snapshot] deleted background chainstate directory (%s)\n",
7526Chainstate &ChainstateManager::GetChainstateForIndexing() {
7530 return (this->
GetAll().size() > 1) ? *m_ibd_chainstate
7531 : *m_active_chainstate;
7535ChainstateManager::GetPruneRange(
const Chainstate &chainstate,
7536 int last_height_can_prune) {
7542 if (this->
GetAll().size() > 1 &&
7543 m_snapshot_chainstate.get() == &chainstate) {
7546 prune_start = *
Assert(GetSnapshotBaseHeight()) + 1;
7549 int max_prune = std::max<int>(0, chainstate.
m_chain.
Height() -
7559 int prune_end = std::min(last_height_can_prune, max_prune);
7561 return {prune_start, prune_end};
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.
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
bool m_checked_merkle_root
std::vector< CTransactionRef > vtx
The block chain is a tree shaped structure starting with the genesis block at the root,...
bool IsValid(enum BlockValidity nUpTo=BlockValidity::TRANSACTIONS) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
std::string ToString() const
CBlockIndex * pprev
pointer to the index of the predecessor of this block
int64_t GetHeaderReceivedTime() const
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
const BlockHash * phashBlock
pointer to the hash of the block, if any.
int64_t GetChainTxCount() const
Get the number of transaction in the chain so far.
bool HaveNumChainTxs() const
Check whether this block and all previous blocks back to the genesis block or an assumeutxo snapshot ...
int32_t nSequenceId
(memory only) Sequential id assigned to distinguish order in which blocks are received.
int64_t GetReceivedTimeDiff() const
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
FlatFilePos GetUndoPos() const EXCLUSIVE_LOCKS_REQUIRED(
CBlockIndex * pskip
pointer to the index of some further predecessor of this block
unsigned int nTx
Number of transactions in this block.
bool RaiseValidity(enum BlockValidity nUpTo) EXCLUSIVE_LOCKS_REQUIRED(
Raise the validity level of this block index entry.
int32_t nVersion
block header
int64_t nTimeReceived
(memory only) block header metadata
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
BlockHash GetBlockHash() const
unsigned int nSize
Size of this block.
int nHeight
height of the entry in the chain. The genesis block has height 0
unsigned int nChainTx
(memory only) Number of transactions in the chain up to and including this block.
Undo information for a CBlock.
std::vector< CTxUndo > vtxundo
Non-refcounted RAII wrapper around a FILE* that implements a ring buffer to deserialize from.
bool SetLimit(uint64_t nPos=std::numeric_limits< uint64_t >::max())
Prevent reading beyond a certain position.
uint64_t GetPos() const
return the current reading position
void FindByte(std::byte byte)
search for a given byte in the stream, and remain positioned on it
void SkipTo(const uint64_t file_pos)
Move the read position ahead in the stream to the given position.
bool SetPos(uint64_t nPos)
rewind to a given reading position
bool eof() const
check whether we're at the end of the source file
An in-memory indexed chain of blocks.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
void SetTip(CBlockIndex &block)
Set/initialize a chain with a given tip.
CBlockIndex * Genesis() const
Returns the index entry for the genesis block of this chain, or nullptr if none.
CBlockIndex * Next(const CBlockIndex *pindex) const
Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip...
int Height() const
Return the maximal height in the chain.
const CBlockIndex * FindFork(const CBlockIndex *pindex) const
Find the last common block between this chain and a block index entry.
bool Contains(const CBlockIndex *pindex) const
Efficiently check whether a block is present in this chain.
CBlockLocator GetLocator() const
Return a CBlockLocator that refers to the tip of this chain.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
const CBlock & GenesisBlock() const
std::vector< int > GetAvailableSnapshotHeights() const
const CMessageHeader::MessageMagic & DiskMagic() const
const ChainTxData & TxData() const
const Consensus::Params & GetConsensus() const
std::optional< AssumeutxoData > AssumeutxoForHeight(int height) const
const CCheckpointData & Checkpoints() const
RAII-style controller object for a CCheckQueue that guarantees the passed queue is finished before co...
std::optional< R > Complete()
void Add(std::vector< T > &&vChecks)
The verifications are represented by a type T, which must provide an operator(), returning an std::op...
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)
bool Sync()
Push the modifications applied to this cache to its base while retaining the contents of this cache (...
void EmplaceCoinInternalDANGER(COutPoint &&outpoint, Coin &&coin)
Emplace a coin into cacheCoins without performing any checks, marking the emplaced coin as dirty.
bool HaveCoin(const COutPoint &outpoint) const override
Just check whether a given outpoint is unspent.
const Coin & AccessCoin(const COutPoint &output) const
Return a reference to Coin in the cache, or coinEmpty if not found.
CCoinsView backed by the coin database (chainstate/)
std::optional< fs::path > StoragePath()
void ResizeCache(size_t new_cache_size) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Dynamically alter the underlying leveldb cache size.
Abstract view on the open txout dataset.
virtual bool GetCoin(const COutPoint &outpoint, Coin &coin) const
Retrieve the Coin (unspent transaction output) for a given outpoint.
CCoinsView that brings transactions from a mempool into view.
Fee rate in satoshis per kilobyte: Amount / kB.
void TransactionAddedToMempool(const CTransactionRef &, std::shared_ptr< const std::vector< Coin > >, uint64_t mempool_sequence)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void BlockConnected(ChainstateRole, const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockChecked(const CBlock &, const BlockValidationState &)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void ChainStateFlushed(ChainstateRole, const CBlockLocator &)
void BlockFinalized(const CBlockIndex *pindex)
void insert(Span< const uint8_t > vKey)
bool contains(Span< const uint8_t > vKey) const
CSHA256 & Write(const uint8_t *data, size_t len)
Closure representing one script verification.
SignatureCache * m_signature_cache
ScriptExecutionMetrics GetScriptExecutionMetrics() const
TxSigCheckLimiter * pTxLimitSigChecks
ScriptExecutionMetrics metrics
std::optional< std::pair< ScriptError, std::string > > operator()()
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
CTransactionRef get(const TxId &txid) const
void clear(bool include_finalized_txs=false)
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.
bool ConnectTip(BlockValidationState &state, BlockPolicyValidationState &blockPolicyState, CBlockIndex *pindexNew, const std::shared_ptr< const CBlock > &pblock, DisconnectedBlockTransactions &disconnectpool, const avalanche::Processor *const avalanche=nullptr, ChainstateRole chainstate_role=ChainstateRole::NORMAL) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Connect a new block to m_chain.
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?
void UpdateTip(const CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(NodeClock::time_poin m_next_write)
Check warning conditions and do some notifications on new chain tip set.
CTxMemPool * GetMempool()
bool RollforwardBlock(const CBlockIndex *pindex, CCoinsViewCache &inputs) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Apply the effects of a block on the utxo cache, ignoring that it may already have been applied.
size_t m_coinstip_cache_size_bytes
The cache size of the in-memory coins view.
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
bool 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 SetBlockFailureFlags(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(voi ResetBlockFailureFlags)(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Set invalidity status to all descendants of a block.
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
bool ActivateBestChainStep(BlockValidationState &state, CBlockIndex *pindexMostWork, const std::shared_ptr< const CBlock > &pblock, bool &fInvalidFound, const avalanche::Processor *const avalanche=nullptr, ChainstateRole=ChainstateRole::NORMAL) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Try to make some progress towards making pindexMostWork the active block.
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 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(
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.
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
void TryAddBlockIndexCandidate(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool AvalancheFinalizeBlock(CBlockIndex *pindex, avalanche::Processor &avalanche) EXCLUSIVE_LOCKS_REQUIRED(voi ClearAvalancheFinalizedBlock)() EXCLUSIVE_LOCKS_REQUIRED(!cs_avalancheFinalizedBlockIndex)
Mark a block as finalized by avalanche.
void PruneAndFlush()
Prune blockfiles from the disk if necessary and then flush chainstate changes if we pruned.
bool ResizeCoinsCaches(size_t coinstip_size, size_t coinsdb_size) EXCLUSIVE_LOCKS_REQUIRED(bool FlushStateToDisk(BlockValidationState &state, FlushStateMode mode, int nManualPruneHeight=0)
Resize the CoinsViews caches dynamically and flush state to disk.
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances.
ChainstateRole GetRole() const EXCLUSIVE_LOCKS_REQUIRED(void InitCoinsDB(size_t cache_size_bytes, bool in_memory, bool should_wipe, std::string leveldb_name="chainstate")
Return the current role of the chainstate.
CBlockIndex const * m_best_fork_base
void InvalidChainFound(CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(cs_main
void UnparkBlock(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove parked status from a block.
bool PreciousBlock(BlockValidationState &state, CBlockIndex *pindex, avalanche::Processor *const avalanche=nullptr) EXCLUSIVE_LOCKS_REQUIRED(!m_chainstate_mutex
Mark a block as precious and reorganize.
void ClearBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex * FindForkInGlobalIndex(const CBlockLocator &locator) const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Find the last common block of this chain and a locator.
CBlockIndex * FindMostWorkChain(std::vector< const CBlockIndex * > &blocksToReconcile, bool fAutoUnpark) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Return the tip of the chain with the most work in it, that isn't known to be invalid (it's however fa...
bool UpdateFlagsForBlock(CBlockIndex *pindexBase, CBlockIndex *pindex, F f) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ParkBlock(BlockValidationState &state, CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(!m_chainstate_mutex
Park a block.
CoinsCacheSizeState GetCoinsCacheSizeState() EXCLUSIVE_LOCKS_REQUIRED(CoinsCacheSizeState GetCoinsCacheSizeState(size_t max_coins_cache_size_bytes, size_t max_mempool_size_bytes) EXCLUSIVE_LOCKS_REQUIRED(std::string ToString() EXCLUSIVE_LOCKS_REQUIRED(RecursiveMutex * MempoolMutex() const LOCK_RETURNED(m_mempool -> cs)
Dictates whether we need to flush the cache to disk or not.
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
SnapshotCompletionResult MaybeCompleteSnapshotValidation() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetSnapshotBaseBlock() const EXCLUSIVE_LOCKS_REQUIRED(Chainstate ActiveChainstate)() const
Once the background validation chainstate has reached the height which is the base of the UTXO snapsh...
ValidationCache m_validation_cache
std::atomic< int32_t > nBlockSequenceId
Every received block is assigned a unique and increasing identifier, so we know which one to give pri...
bool DetectSnapshotChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(void ResetChainstates() EXCLUSIVE_LOCKS_REQUIRED(bool DeleteSnapshotChainstate() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &ActivateExistingSnapshot(BlockHash base_blockhash) EXCLUSIVE_LOCKS_REQUIRED(bool ValidatedSnapshotCleanup() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &GetChainstateForIndexing() EXCLUSIVE_LOCKS_REQUIRED(std::pair< int, int > GetPruneRange(const Chainstate &chainstate, int last_height_can_prune) EXCLUSIVE_LOCKS_REQUIRED(std::optional< int > GetSnapshotBaseHeight() const EXCLUSIVE_LOCKS_REQUIRED(void RecalculateBestHeader() 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
size_t m_total_coinstip_cache
The total number of bytes available for us to use across all in-memory coins caches.
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.
bool AcceptBlockHeader(const CBlockHeader &block, BlockValidationState &state, CBlockIndex **ppindex, bool min_pow_checked, const std::optional< CCheckpointData > &test_checkpoints=std::nullopt) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
If a block header hasn't already been seen, call CheckBlockHeader on it, ensure that it doesn't desce...
kernel::Notifications & GetNotifications() const
bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network)
void ReceivedBlockTransactions(const CBlock &block, CBlockIndex *pindexNew, const FlatFilePos &pos) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as having its data received and checked (up to BLOCK_VALID_TRANSACTIONS).
bool ShouldCheckBlockIndex() const
bool ProcessNewBlock(const std::shared_ptr< const CBlock > &block, bool force_processing, bool min_pow_checked, bool *new_block, avalanche::Processor *const avalanche=nullptr) LOCKS_EXCLUDED(cs_main)
Process an incoming block.
bool LoadRecentHeadersTime(const fs::path &filePath) EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Load the recent block headers reception time from a file.
void LoadExternalBlockFile(FILE *fileIn, FlatFilePos *dbp=nullptr, std::multimap< BlockHash, FlatFilePos > *blocks_with_unknown_parent=nullptr, avalanche::Processor *const avalanche=nullptr)
Import blocks from an external file.
std::optional< BlockHash > SnapshotBlockhash() const
bool IsSnapshotValidated() const EXCLUSIVE_LOCKS_REQUIRED(
Is there a snapshot in use and has it been fully validated?
bool IsUsable(const Chainstate *const pchainstate) const EXCLUSIVE_LOCKS_REQUIRED(
Return true if a chainstate is considered usable.
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
bool BackgroundSyncInProgress() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
The state of a background sync (for net processing)
size_t m_total_coinsdb_cache
The total number of bytes available for us to use across all leveldb coins databases.
std::atomic< bool > m_cached_finished_ibd
Whether initial block download has ended and IsInitialBlockDownload should return false from now on.
bool PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
const util::SignalInterrupt & m_interrupt
int ActiveHeight() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
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).
std::function< void()> snapshot_download_completed
Function to restart active indexes; set dynamically to avoid a circular dependency on base/index....
const CChainParams & GetParams() const
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
ChainstateManager(const util::SignalInterrupt &interrupt, Options options, node::BlockManager::Options blockman_options)
const arith_uint256 & MinimumChainWork() const
void CheckBlockIndex()
Make various assertions about the state of the block index.
const CBlockIndex * GetAvalancheFinalizedTip() const
util::Result< CBlockIndex * > ActivateSnapshot(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Construct and activate a Chainstate on the basis of UTXO snapshot data.
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
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
std::pair< uint32_t, size_t > setup_bytes(size_t bytes)
setup_bytes is a convenience function which accounts for internal memory usage when deciding how many...
bool get(Element &e, const bool erase) const
get is almost identical to contains(), with the difference that it obtains the found element (for Ele...
void insert(Element e, bool replace=false)
insert loops at most depth_limit times trying to insert a hash at various locations in the table via ...
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.
Tp rand_uniform_delay(const Tp &time, typename Tp::duration range) noexcept
Return the time point advanced by a uniform random duration.
The script cache is a map using a key/value element, that caches the success of executing a specific ...
static TxSigCheckLimiter getDisabled()
Convenience class for initializing and passing the script execution cache and signature cache.
CuckooCache::cache< ScriptCacheElement, ScriptCacheHasher > m_script_execution_cache
ValidationCache(size_t script_execution_cache_bytes, size_t signature_cache_bytes)
CSHA256 ScriptExecutionCacheHasher() const
Return a copy of the pre-initialized hasher.
CSHA256 m_script_execution_cache_hasher
Pre-initialized hasher to avoid having to recreate it for every hash calculation.
SignatureCache m_signature_cache
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 fatalError(const std::string &debug_message, const bilingual_str &user_message={})
The fatal error notification is sent to notify the user when an error occurs in kernel code that can'...
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...
const kernel::BlockManagerOpts m_opts
bool ReadBlockFromDisk(CBlock &block, const FlatFilePos &pos) const
Functions for disk access for blocks.
RecursiveMutex cs_LastBlockFile
bool CheckBlockDataAvailability(const CBlockIndex &upper_block LIFETIMEBOUND, const CBlockIndex &lower_block LIFETIMEBOUND) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetFirstBlock(const CBlockIndex &upper_block LIFETIMEBOUND, std::function< bool(BlockStatus)> status_test, const CBlockIndex *lower_block=nullptr) const EXCLUSIVE_LOCKS_REQUIRED(boo m_have_pruned)
Check if all blocks in the [upper_block, lower_block] range have data available.
bool FlushChainstateBlockFile(int tip_height)
void FindFilesToPrune(std::set< int > &setFilesToPrune, int last_prune, const Chainstate &chain, ChainstateManager &chainman)
Prune block and undo files (blk???.dat and undo???.dat) so that the disk space used is less than a us...
void UpdateBlockInfo(const CBlock &block, unsigned int nHeight, const FlatFilePos &pos)
Update blockfile info while processing a block during reindex.
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 WriteUndoDataForBlock(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos SaveBlockToDisk(const CBlock &block, int nHeight)
Store block on disk and update block file statistics.
bool WriteBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(bool LoadBlockIndexDB(const std::optional< BlockHash > &snapshot_blockhash) EXCLUSIVE_LOCKS_REQUIRED(void ScanAndUnlinkAlreadyPrunedFiles() EXCLUSIVE_LOCKS_REQUIRED(CBlockIndex * AddToBlockIndex(const CBlockHeader &block, CBlockIndex *&best_header) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove any pruned block & undo files that are still on disk.
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
bool 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.
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, const Chainstate &chain, ChainstateManager &chainman)
Calculate the block/rev files to delete based on height specified by user with RPC command pruneblock...
std::optional< int > m_snapshot_height
The height of the base block of an assumeutxo snapshot, if one is in use.
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.
Helper class that manages an interrupt flag, and allows a thread or signal to interrupt another threa...
static constexpr int CLIENT_VERSION
bitcoind-res.rc includes this file, but it cannot cope with real c++ code.
const Coin & AccessByTxid(const CCoinsViewCache &view, const TxId &txid)
Utility function to find any unspent output with a given txid.
void AddCoins(CCoinsViewCache &cache, const CTransaction &tx, int nHeight, bool check_for_overwrite)
Utility function to add all of a transaction's outputs to a cache.
@ BLOCK_CHECKPOINT
the block failed to meet one of our checkpoints
@ BLOCK_HEADER_LOW_WORK
the block header may be on a too-little-work chain
@ BLOCK_INVALID_HEADER
invalid proof of work or time too old
@ BLOCK_CACHED_INVALID
this block was cached as being invalid and we didn't store the reason why
@ BLOCK_CONSENSUS
invalid by consensus rules (excluding any below reasons)
@ BLOCK_MISSING_PREV
We don't have the previous block the checked one is built on.
@ BLOCK_INVALID_PREV
A block this one builds on is invalid.
@ BLOCK_MUTATED
the block's data didn't match the data committed to by the PoW
@ BLOCK_TIME_FUTURE
block timestamp was > 2 hours in the future (or our clock is bad)
@ TX_MISSING_INPUTS
transaction was missing some of its inputs
@ TX_CHILD_BEFORE_PARENT
This tx outputs are already spent in the mempool.
@ TX_MEMPOOL_POLICY
violated mempool's fee/size/descendant/etc limits
@ TX_PACKAGE_RECONSIDERABLE
fails some policy, but might be acceptable if submitted in a (different) package
@ TX_PREMATURE_SPEND
transaction spends a coinbase too early, or violates locktime/sequence locks
@ TX_DUPLICATE
Tx already in mempool or in the chain.
@ TX_INPUTS_NOT_STANDARD
inputs failed policy rules
@ TX_CONFLICT
Tx conflicts with a finalized tx, i.e.
@ TX_NOT_STANDARD
otherwise didn't meet our local policy rules
@ TX_AVALANCHE_RECONSIDERABLE
fails some policy, but might be reconsidered by avalanche voting
@ TX_NO_MEMPOOL
this node does not have a mempool so can't validate the transaction
@ TX_CONSENSUS
invalid by consensus rules
static constexpr unsigned int LOCKTIME_VERIFY_SEQUENCE
Flags for nSequence and nLockTime locks.
static const uint64_t MAX_TX_SIZE
The maximum allowed size for a transaction, in bytes.
uint64_t GetMaxBlockSigChecksCount(uint64_t maxBlockSize)
Compute the maximum number of sigchecks that can be contained in a block given the MAXIMUM block size...
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
bool DeploymentActiveAfter(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep)
Determine if a deployment is active for the next block.
bool DeploymentActiveAt(const CBlockIndex &index, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep)
Determine if a deployment is active for this block.
bool RenameOver(fs::path src, fs::path dest)
bool CheckDiskSpace(const fs::path &dir, uint64_t additional_bytes)
bool FileCommit(FILE *file)
Ensure file contents are fully committed to disk, using a platform-specific feature analogous to fsyn...
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, uint32_t flags, const BaseSignatureChecker &checker, ScriptExecutionMetrics &metricsOut, ScriptError *serror)
Execute an unlocking and locking script together.
ChainstateRole
This enum describes the various roles a specific Chainstate instance can take.
#define LogPrintLevel(category, level,...)
#define LogPrintLevel_(category, level, should_ratelimit,...)
#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)
const fs::path SNAPSHOT_BLOCKHASH_FILENAME
The file in the snapshot chainstate dir which stores the base blockhash.
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate)
std::unordered_map< BlockHash, CBlockIndex, BlockHasher > BlockMap
std::optional< BlockHash > ReadSnapshotBaseBlockhash(const fs::path &chaindir)
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::optional< fs::path > FindSnapshotChainstateDir(const fs::path &data_dir)
Return a path to the snapshot-based chainstate dir, if one exists.
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
========== CONVENIENCE FUNCTIONS FOR COMMONLY USED RANDOMNESS ==========
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
uint64_t ReadCompactSize(Stream &is, bool range_check=true)
Decode a CompactSize-encoded variable-length integer.
size_t GetSerializeSize(const T &t, int nVersion=0)
void StartShutdown()
Request shutdown of the application.
auto Join(const std::vector< T > &list, const BaseType &separator, UnaryOp unary_op) -> decltype(unary_op(list.at(0)))
Join a list of items.
std::string ToString(const T &t)
Locale-independent version of std::to_string.
static constexpr Amount zero() noexcept
Holds configuration for use during UTXO snapshot load and validation.
AssumeutxoHash hash_serialized
The expected hash of the deserialized UTXO set.
unsigned int nChainTx
Used to populate the nChainTx value, which is used during BlockManager::LoadBlockIndex().
A BlockHash is a unqiue identifier for a block.
bool isValid(enum BlockValidity nUpTo=BlockValidity::TRANSACTIONS) const
Check whether this block index entry is valid up to the passed validity level.
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
std::vector< BlockHash > vHave
Holds various statistics on transactions within a chain.
User-controlled performance and debug options.
Parameters that influence chain consensus.
int BIP34Height
Block height and hash at which BIP34 becomes active.
int nSubsidyHalvingInterval
int obolenskyActivationTime
Unix time used for MTP activation of 15 May 2026 12:00:00 UTC upgrade.
BlockHash hashGenesisBlock
int64_t nPowTargetSpacing
bool fPowAllowMinDifficultyBlocks
Application-specific storage settings.
fs::path path
Location in the filesystem where leveldb data will be stored.
Validation result for a transaction evaluated by MemPoolAccept (single or package).
const ResultType m_result_type
Result type.
@ VALID
Fully validated, valid.
static MempoolAcceptResult Failure(TxValidationState state)
static MempoolAcceptResult FeeFailure(TxValidationState state, CFeeRate effective_feerate, const std::vector< TxId > &txids_fee_calculations)
static MempoolAcceptResult Success(int64_t vsize, Amount fees, CFeeRate effective_feerate, const std::vector< TxId > &txids_fee_calculations)
Constructor for success case.
static MempoolAcceptResult MempoolTx(int64_t vsize, Amount fees)
Constructor for already-in-mempool case.
static time_point now() noexcept
Return current system time or mocked time, if set.
std::chrono::time_point< NodeClock > time_point
Validation result for package mempool acceptance.
Precompute sighash midstate to avoid quadratic hashing.
In future if many more values are added, it should be considered to expand the element size to 64 byt...
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...
const fs::path blocks_dir
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
std::optional< int64_t > replay_protection_activation_time
If set, this overwrites the timestamp at which replay protection activates.
#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 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 ContextualCheckTransactionForCurrentBlock(const CBlockIndex &active_chain_tip, const Consensus::Params ¶ms, const CTransaction &tx, TxValidationState &state)
bool EvaluateSequenceLocks(const CBlockIndex &block, std::pair< int, int64_t > lockPair)
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.
bool ContextualCheckTransactionForCurrentBlock(const CBlockIndex &active_chain_tip, const Consensus::Params ¶ms, const CTransaction &tx, TxValidationState &state) EXCLUSIVE_LOCKS_REQUIRED(std::pair< int, int64_t > CalculateSequenceLocks(const CTransaction &tx, int flags, std::vector< int > &prevHeights, const CBlockIndex &block)
This is a variant of ContextualCheckTransaction which computes the contextual check for a transaction...
static const uint32_t MEMPOOL_HEIGHT
Fake height value used in Coins to signify they are only in the memory pool(since 0....
uint256 uint256S(const char *str)
uint256 from const char *.
static bool DeleteCoinsDBFromDisk(const fs::path &db_path, bool is_snapshot) EXCLUSIVE_LOCKS_REQUIRED(
static bool NotifyHeaderTip(ChainstateManager &chainman) LOCKS_EXCLUDED(cs_main)
void StartScriptCheckWorkerThreads(int threads_num)
Run instances of script checking worker threads.
static int64_t num_blocks_total
bool FatalError(Notifications ¬ifications, BlockValidationState &state, const std::string &strMessage, const bilingual_str &userMessage)
GlobalMutex g_best_block_mutex
static SteadyClock::duration time_connect_total
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...
return CheckInputScripts(tx, state, view, flags, true, true, txdata, validation_cache, nSigChecksOut)
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.
static constexpr size_t WARN_FLUSH_COINS_SIZE
Size threshold for warning about slow UTXO set flush to disk.
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 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 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 SteadyClock::duration time_post_connect
static SteadyClock::duration time_chainstate
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.
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept)
Validate (and maybe submit) a package to the mempool.
static SteadyClock::duration time_forks
static CCheckQueue< CScriptCheck > scriptcheckqueue(128)
static ChainstateManager::Options && Flatten(ChainstateManager::Options &&opts)
Apply default chain params to nullopt members.
static constexpr auto DATABASE_WRITE_INTERVAL_MAX
static SteadyClock::duration time_verify
static bool CheckMerkleRoot(const CBlock &block, BlockValidationState &state)
static SteadyClock::duration time_check
static constexpr int PRUNE_LOCK_BUFFER
The number of blocks to keep below the deepest prune lock.
static SteadyClock::duration time_index
void StopScriptCheckWorkerThreads()
Stop all of the script checking worker threads.
static void LimitValidationInterfaceQueue() LOCKS_EXCLUDED(cs_main)
void SpendCoins(CCoinsViewCache &view, const CTransaction &tx, CTxUndo &txundo, int nHeight)
Mark all the coins corresponding to a given transaction inputs as spent.
bool CheckBlock(const CBlock &block, BlockValidationState &state, const Consensus::Params ¶ms, BlockValidationOptions validationOptions)
Functions for validating blocks and updating the block tree.
static SteadyClock::duration time_connect
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.
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)
Check a block is completely valid from start to finish (only works on top of our current best block)
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 IsBlockMutated(const CBlock &block)
Check if a block has been mutated (with respect to its merkle root).
std::vector< Coin > GetSpentCoins(const CTransactionRef &ptx, const CCoinsViewCache &coins_view)
Get the coins spent by ptx from the coins_view.
static constexpr auto DATABASE_WRITE_INTERVAL_MIN
Time window to wait between writing blocks/block index and chainstate to disk.
static SteadyClock::duration time_total
static bool CheckInputsFromMempoolAndCache(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &view, const CTxMemPool &pool, const uint32_t flags, PrecomputedTransactionData &txdata, ValidationCache &validation_cache, 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 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 SteadyClock::duration time_flush
static SteadyClock::duration time_read_from_disk_total
static void SnapshotUTXOHashBreakpoint(const util::SignalInterrupt &interrupt)
static constexpr uint64_t HEADERS_TIME_VERSION
static fs::path GetSnapshotCoinsDBPath(Chainstate &cs) EXCLUSIVE_LOCKS_REQUIRED(
static bool IsReplayProtectionEnabled(const Consensus::Params ¶ms, const CBlockIndex *pindexPrev, const std::optional< int64_t > activation_time)
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(
#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.
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()