94 "level 0 reads the blocks from disk",
95 "level 1 verifies block validity",
96 "level 2 verifies undo data",
97 "level 3 checks disconnection of tip blocks",
98 "level 4 tries to reconnect the blocks",
99 "each level includes the checks of the previous levels",
135 std::vector<CScriptCheck>* pvChecks =
nullptr)
148 const int nBlockHeight = active_chain_tip.nHeight + 1;
155 const int64_t nBlockTime{active_chain_tip.GetMedianTimePast()};
157 return IsFinalTx(tx, nBlockHeight, nBlockTime);
171 std::optional<std::vector<int>> CalculatePrevHeights(
176 std::vector<int> prev_heights;
177 prev_heights.resize(tx.
vin.size());
178 for (
size_t i = 0; i < tx.
vin.size(); ++i) {
187 prev_heights[i] = tip.
nHeight + 1;
189 prev_heights[i] = coin.
nHeight;
203 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
204 if (!prev_heights.has_value())
return std::nullopt;
207 next_tip.
pprev = tip;
225 int max_input_height{0};
226 for (
const int height : prev_heights.value()) {
228 if (height != next_tip.
nHeight) {
229 max_input_height = std::max(max_input_height, height);
267 int expired = pool.Expire(GetTime<std::chrono::seconds>() - pool.m_expiry);
272 std::vector<COutPoint> vNoSpendsRemaining;
273 pool.TrimToSize(pool.m_max_size_bytes, &vNoSpendsRemaining);
274 for (
const COutPoint& removed : vNoSpendsRemaining)
275 coins_cache.Uncache(removed);
281 if (active_chainstate.m_chainman.IsInitialBlockDownload()) {
286 if (active_chainstate.m_chain.Height() < active_chainstate.m_chainman.m_best_header->nHeight - 1) {
300 std::vector<uint256> vHashUpdate;
307 const auto queuedTx = disconnectpool.
take();
308 auto it = queuedTx.rbegin();
309 while (it != queuedTx.rend()) {
311 if (!fAddToMempool || (*it)->IsCoinBase() ||
319 vHashUpdate.push_back((*it)->GetHash());
365 if (it->GetSpendsCoinbase()) {
408 if (coin.
IsSpent())
return false;
420 const Coin& coinFromUTXOSet = coins_tip.AccessCoin(txin.
prevout);
438 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
439 m_active_chainstate(active_chainstate)
447 const int64_t m_accept_time;
448 const bool m_bypass_limits;
456 std::vector<COutPoint>& m_coins_to_uncache;
457 const bool m_test_accept;
461 const bool m_allow_replacement;
466 const bool m_package_submission;
470 const bool m_package_feerates;
473 static ATMPArgs SingleAccept(
const CChainParams& chainparams, int64_t accept_time,
474 bool bypass_limits, std::vector<COutPoint>& coins_to_uncache,
476 return ATMPArgs{ chainparams,
488 static ATMPArgs PackageTestAccept(
const CChainParams& chainparams, int64_t accept_time,
489 std::vector<COutPoint>& coins_to_uncache) {
490 return ATMPArgs{ chainparams,
502 static ATMPArgs PackageChildWithParents(
const CChainParams& chainparams, int64_t accept_time,
503 std::vector<COutPoint>& coins_to_uncache) {
504 return ATMPArgs{ chainparams,
516 static ATMPArgs SingleInPackageAccept(
const ATMPArgs& package_args) {
517 return ATMPArgs{ package_args.m_chainparams,
518 package_args.m_accept_time,
520 package_args.m_coins_to_uncache,
521 package_args.m_test_accept,
534 std::vector<COutPoint>& coins_to_uncache,
536 bool allow_replacement,
537 bool package_submission,
538 bool package_feerates)
539 : m_chainparams{chainparams},
540 m_accept_time{accept_time},
541 m_bypass_limits{bypass_limits},
542 m_coins_to_uncache{coins_to_uncache},
543 m_test_accept{test_accept},
544 m_allow_replacement{allow_replacement},
545 m_package_submission{package_submission},
546 m_package_feerates{package_feerates}
584 explicit Workspace(
const CTransactionRef& ptx) : m_ptx(ptx), m_hash(ptx->GetHash()) {}
586 std::set<uint256> m_conflicts;
596 std::unique_ptr<CTxMemPoolEntry> m_entry;
600 std::list<CTransactionRef> m_replaced_transactions;
612 size_t m_conflicting_size{0};
639 bool PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
662 std::map<uint256, MempoolAcceptResult>& results)
670 CAmount mempoolRejectFee = m_pool.GetMinFee().GetFee(package_size);
671 if (mempoolRejectFee > 0 && package_fee < mempoolRejectFee) {
675 if (package_fee < m_pool.m_min_relay_feerate.GetFee(package_size)) {
677 strprintf(
"%d < %d", package_fee, m_pool.m_min_relay_feerate.GetFee(package_size)));
694 bool MemPoolAccept::PreChecks(ATMPArgs&
args, Workspace& ws)
700 const uint256& hash = ws.m_hash;
703 const int64_t nAcceptTime =
args.m_accept_time;
704 const bool bypass_limits =
args.m_bypass_limits;
705 std::vector<COutPoint>& coins_to_uncache =
args.m_coins_to_uncache;
709 std::unique_ptr<CTxMemPoolEntry>& entry = ws.m_entry;
721 if (m_pool.m_require_standard && !
IsStandardTx(tx, m_pool.m_max_datacarrier_bytes, m_pool.m_permit_bare_multisig, m_pool.m_dust_relay_feerate, reason)) {
749 if (ptxConflicting) {
750 if (!
args.m_allow_replacement) {
754 if (!ws.m_conflicts.count(ptxConflicting->
GetHash()))
770 ws.m_conflicts.insert(ptxConflicting->
GetHash());
775 m_view.SetBackend(m_viewmempool);
781 coins_to_uncache.push_back(txin.
prevout);
787 if (!m_view.HaveCoin(txin.
prevout)) {
802 m_view.GetBestBlock();
807 m_view.SetBackend(m_dummy);
809 assert(m_active_chainstate.m_blockman.LookupBlockIndex(m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
816 const std::optional<LockPoints> lock_points{
CalculateLockPointsAtTip(m_active_chainstate.m_chain.Tip(), m_view, tx)};
838 ws.m_modified_fees = ws.m_base_fees;
839 m_pool.ApplyDelta(hash, ws.m_modified_fees);
843 bool fSpendsCoinbase =
false;
845 const Coin &coin = m_view.AccessCoin(txin.
prevout);
847 fSpendsCoinbase =
true;
854 const uint64_t entry_sequence = bypass_limits ? 0 : m_pool.GetSequence();
855 entry.reset(
new CTxMemPoolEntry(ptx, ws.m_base_fees, nAcceptTime, m_active_chainstate.m_chain.Height(), entry_sequence,
856 fSpendsCoinbase, nSigOpsCost, lock_points.value()));
857 ws.m_vsize = entry->GetTxSize();
868 if (!bypass_limits && ws.m_modified_fees < m_pool.m_min_relay_feerate.GetFee(ws.m_vsize)) {
870 strprintf(
"%d < %d", ws.m_modified_fees, m_pool.m_min_relay_feerate.GetFee(ws.m_vsize)));
875 if (!bypass_limits && !
args.m_package_feerates && !
CheckFeeRate(ws.m_vsize, ws.m_modified_fees, state))
return false;
877 ws.m_iters_conflicting = m_pool.GetIterSet(ws.m_conflicts);
884 if (ws.m_conflicts.size() == 1) {
912 assert(ws.m_iters_conflicting.size() == 1);
919 auto ancestors{m_pool.CalculateMemPoolAncestors(*entry, maybe_rbf_limits)};
943 ancestors = m_pool.CalculateMemPoolAncestors(*entry, cpfp_carve_out_limits);
947 ws.m_ancestors = *ancestors;
959 m_rbf = !ws.m_conflicts.empty();
963 bool MemPoolAccept::ReplacementChecks(Workspace& ws)
969 const uint256& hash = ws.m_hash;
972 CFeeRate newFeeRate(ws.m_modified_fees, ws.m_vsize);
987 if (
const auto err_string{
GetEntriesForConflicts(tx, m_pool, ws.m_iters_conflicting, ws.m_all_conflicting)}) {
989 "too many potential replacements", *err_string);
994 "replacement-adds-unconfirmed", *err_string);
999 ws.m_conflicting_fees += it->GetModifiedFee();
1000 ws.m_conflicting_size += it->GetTxSize();
1002 if (
const auto err_string{
PaysForRBF(ws.m_conflicting_fees, ws.m_modified_fees, ws.m_vsize,
1003 m_pool.m_incremental_relay_feerate, hash)}) {
1009 bool MemPoolAccept::PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
1010 const int64_t total_vsize,
1017 assert(std::all_of(txns.cbegin(), txns.cend(), [
this](
const auto& tx)
1020 std::string err_string;
1021 if (!m_pool.CheckPackageLimits(txns, total_vsize, err_string)) {
1028 bool MemPoolAccept::PolicyScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1039 if (!
CheckInputScripts(tx, state, m_view, scriptVerifyFlags,
true,
false, ws.m_precomputed_txdata)) {
1056 bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1061 const uint256& hash = ws.m_hash;
1079 unsigned int currentBlockScriptVerifyFlags{
GetBlockScriptFlags(*m_active_chainstate.m_chain.Tip(), m_active_chainstate.m_chainman)};
1081 ws.m_precomputed_txdata, m_active_chainstate.CoinsTip())) {
1082 LogPrintf(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against latest-block but not STANDARD flags %s, %s\n", hash.
ToString(), state.
ToString());
1089 bool MemPoolAccept::Finalize(
const ATMPArgs&
args, Workspace& ws)
1094 const uint256& hash = ws.m_hash;
1096 const bool bypass_limits =
args.m_bypass_limits;
1098 std::unique_ptr<CTxMemPoolEntry>& entry = ws.m_entry;
1103 LogPrint(
BCLog::MEMPOOL,
"replacing tx %s (wtxid=%s) with %s (wtxid=%s) for %s additional fees, %d delta bytes\n",
1104 it->GetTx().GetHash().ToString(),
1105 it->GetTx().GetWitnessHash().ToString(),
1108 FormatMoney(ws.m_modified_fees - ws.m_conflicting_fees),
1109 (int)entry->GetTxSize() - (int)ws.m_conflicting_size);
1110 TRACE7(mempool, replaced,
1111 it->GetTx().GetHash().data(),
1114 std::chrono::duration_cast<std::chrono::duration<std::uint64_t>>(it->GetTime()).
count(),
1119 ws.m_replaced_transactions.push_back(it->GetSharedTx());
1129 bool validForFeeEstimation = !bypass_limits && !
args.m_package_submission &&
IsCurrentForFeeEstimation(m_active_chainstate) && m_pool.HasNoInputsOf(tx);
1132 m_pool.addUnchecked(*entry, ws.m_ancestors, validForFeeEstimation);
1138 if (!
args.m_package_submission && !bypass_limits) {
1146 bool MemPoolAccept::SubmitPackage(
const ATMPArgs&
args, std::vector<Workspace>& workspaces,
1148 std::map<uint256, MempoolAcceptResult>& results)
1154 assert(std::all_of(workspaces.cbegin(), workspaces.cend(), [
this](
const auto& ws){
1155 return !m_pool.exists(
GenTxid::Txid(ws.m_ptx->GetHash())); }));
1157 bool all_submitted =
true;
1162 for (Workspace& ws : workspaces) {
1163 if (!ConsensusScriptChecks(
args, ws)) {
1167 all_submitted =
false;
1169 strprintf(
"BUG! PolicyScriptChecks succeeded but ConsensusScriptChecks failed: %s",
1170 ws.m_ptx->GetHash().ToString()));
1176 auto ancestors{m_pool.CalculateMemPoolAncestors(*ws.m_entry, m_pool.m_limits)};
1181 all_submitted =
false;
1183 strprintf(
"BUG! Mempool ancestors or descendants were underestimated: %s",
1184 ws.m_ptx->GetHash().ToString()));
1186 ws.m_ancestors = std::move(ancestors).value_or(ws.m_ancestors);
1193 if (!Finalize(
args, ws)) {
1197 all_submitted =
false;
1199 strprintf(
"BUG! Adding to mempool failed: %s", ws.m_ptx->GetHash().ToString()));
1203 std::vector<uint256> all_package_wtxids;
1204 all_package_wtxids.reserve(workspaces.size());
1205 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1206 [](
const auto& ws) {
return ws.m_ptx->GetWitnessHash(); });
1209 for (Workspace& ws : workspaces) {
1210 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1211 CFeeRate{ws.m_modified_fees,
static_cast<uint32_t
>(ws.m_vsize)};
1212 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1213 std::vector<uint256>({ws.m_ptx->GetWitnessHash()});
1214 results.emplace(ws.m_ptx->GetWitnessHash(),
1216 ws.m_base_fees, effective_feerate, effective_feerate_wtxids));
1219 return all_submitted;
1239 const CFeeRate effective_feerate{ws.m_modified_fees,
static_cast<uint32_t
>(ws.m_vsize)};
1240 const std::vector<uint256> single_wtxid{ws.m_ptx->GetWitnessHash()};
1242 if (
args.m_test_accept) {
1244 ws.m_base_fees, effective_feerate, single_wtxid);
1252 effective_feerate, single_wtxid);
1263 std::vector<Workspace> workspaces{};
1264 workspaces.reserve(txns.size());
1265 std::transform(txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1266 [](
const auto& tx) {
return Workspace(tx); });
1267 std::map<uint256, MempoolAcceptResult> results;
1272 for (Workspace& ws : workspaces) {
1273 if (!PreChecks(
args, ws)) {
1284 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1296 const auto m_total_vsize = std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1297 [](int64_t
sum,
auto& ws) {
return sum + ws.m_vsize; });
1298 const auto m_total_modified_fees = std::accumulate(workspaces.cbegin(), workspaces.cend(),
CAmount{0},
1299 [](
CAmount sum,
auto& ws) {
return sum + ws.m_modified_fees; });
1300 const CFeeRate package_feerate(m_total_modified_fees, m_total_vsize);
1302 if (
args.m_package_feerates &&
1303 !
CheckFeeRate(m_total_vsize, m_total_modified_fees, placeholder_state)) {
1311 std::string err_string;
1312 if (txns.size() > 1 && !PackageMempoolChecks(txns, m_total_vsize, package_state)) {
1316 std::vector<uint256> all_package_wtxids;
1317 all_package_wtxids.reserve(workspaces.size());
1318 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1319 [](
const auto& ws) {
return ws.m_ptx->GetWitnessHash(); });
1320 for (Workspace& ws : workspaces) {
1321 ws.m_package_feerate = package_feerate;
1322 if (!PolicyScriptChecks(
args, ws)) {
1328 if (
args.m_test_accept) {
1329 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1330 CFeeRate{ws.m_modified_fees,
static_cast<uint32_t
>(ws.m_vsize)};
1331 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1332 std::vector<uint256>{ws.m_ptx->GetWitnessHash()};
1333 results.emplace(ws.m_ptx->GetWitnessHash(),
1335 ws.m_vsize, ws.m_base_fees, effective_feerate,
1336 effective_feerate_wtxids));
1342 if (!SubmitPackage(
args, workspaces, package_state, results)) {
1355 if (subpackage.size() > 1) {
1356 return AcceptMultipleTransactions(subpackage,
args);
1358 const auto& tx = subpackage.front();
1359 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(
args);
1360 const auto single_res = AcceptSingleTransaction(tx, single_args);
1388 for (
const auto& outpoint : m_viewmempool.GetNonBaseCoins()) {
1391 m_view.Uncache(outpoint);
1394 m_viewmempool.Reset();
1418 assert(package.size() > 1);
1421 const auto& child = package.back();
1422 std::unordered_set<uint256, SaltedTxidHasher> unconfirmed_parent_txids;
1423 std::transform(package.cbegin(), package.cend() - 1,
1424 std::inserter(unconfirmed_parent_txids, unconfirmed_parent_txids.end()),
1425 [](
const auto& tx) {
return tx->
GetHash(); });
1432 const CCoinsViewCache& coins_tip_cache = m_active_chainstate.CoinsTip();
1433 for (
const auto& input : child->vin) {
1434 if (!coins_tip_cache.HaveCoinInCache(input.prevout)) {
1435 args.m_coins_to_uncache.push_back(input.prevout);
1441 m_view.SetBackend(m_active_chainstate.CoinsTip());
1442 const auto package_or_confirmed = [
this, &unconfirmed_parent_txids](
const auto& input) {
1443 return unconfirmed_parent_txids.count(input.prevout.hash) > 0 || m_view.HaveCoin(input.prevout);
1445 if (!std::all_of(child->vin.cbegin(), child->vin.cend(), package_or_confirmed)) {
1451 m_view.SetBackend(m_dummy);
1456 std::map<uint256, MempoolAcceptResult> results_final;
1460 std::map<uint256, MempoolAcceptResult> individual_results_nonfinal;
1461 bool quit_early{
false};
1462 std::vector<CTransactionRef> txns_package_eval;
1463 for (
const auto& tx : package) {
1465 const auto& txid = tx->
GetHash();
1479 auto iter = m_pool.GetIter(txid);
1480 assert(iter != std::nullopt);
1490 auto iter = m_pool.GetIter(txid);
1491 assert(iter != std::nullopt);
1497 const auto single_package_res = AcceptSubPackage({tx},
args);
1498 const auto& single_res = single_package_res.m_tx_results.at(wtxid);
1503 results_final.emplace(wtxid, single_res);
1517 individual_results_nonfinal.emplace(wtxid, single_res);
1519 individual_results_nonfinal.emplace(wtxid, single_res);
1520 txns_package_eval.push_back(tx);
1525 auto multi_submission_result = quit_early || txns_package_eval.empty() ?
PackageMempoolAcceptResult(package_state_quit_early, {}) :
1526 AcceptSubPackage(txns_package_eval,
args);
1533 for (
const auto& tx : package) {
1535 if (multi_submission_result.m_tx_results.count(wtxid) > 0) {
1537 Assume(results_final.count(wtxid) == 0);
1540 const auto& txresult = multi_submission_result.m_tx_results.at(wtxid);
1547 results_final.emplace(wtxid, txresult);
1549 }
else if (
const auto it{results_final.find(wtxid)}; it != results_final.end()) {
1553 Assume(individual_results_nonfinal.count(wtxid) == 0);
1560 results_final.erase(wtxid);
1563 }
else if (
const auto it{individual_results_nonfinal.find(wtxid)}; it != individual_results_nonfinal.end()) {
1566 results_final.emplace(wtxid, it->second);
1569 Assume(results_final.size() == package.size());
1576 int64_t accept_time,
bool bypass_limits,
bool test_accept)
1580 const CChainParams& chainparams{active_chainstate.m_chainman.GetParams()};
1581 assert(active_chainstate.GetMempool() !=
nullptr);
1582 CTxMemPool& pool{*active_chainstate.GetMempool()};
1584 std::vector<COutPoint> coins_to_uncache;
1585 auto args = MemPoolAccept::ATMPArgs::SingleAccept(chainparams, accept_time, bypass_limits, coins_to_uncache, test_accept);
1593 for (
const COutPoint& hashTx : coins_to_uncache)
1594 active_chainstate.CoinsTip().Uncache(hashTx);
1595 TRACE2(mempool, rejected,
1607 const Package& package,
bool test_accept)
1610 assert(!package.empty());
1611 assert(std::all_of(package.cbegin(), package.cend(), [](
const auto& tx){
return tx !=
nullptr;}));
1613 std::vector<COutPoint> coins_to_uncache;
1618 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(chainparams,
GetTime(), coins_to_uncache);
1619 return MemPoolAccept(pool, active_chainstate).AcceptMultipleTransactions(package,
args);
1621 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(chainparams,
GetTime(), coins_to_uncache);
1622 return MemPoolAccept(pool, active_chainstate).AcceptPackage(package,
args);
1627 if (test_accept || result.m_state.IsInvalid()) {
1628 for (
const COutPoint& hashTx : coins_to_uncache) {
1647 nSubsidy >>= halvings;
1652 : m_dbview{std::move(db_params), std::move(options)},
1653 m_catcherview(&m_dbview) {}
1655 void CoinsViews::InitCache()
1658 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1665 std::optional<uint256> from_snapshot_blockhash)
1666 : m_mempool(mempool),
1667 m_blockman(blockman),
1668 m_chainman(chainman),
1669 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1675 return m_cached_snapshot_base;
1678 void Chainstate::InitCoinsDB(
1679 size_t cache_size_bytes,
1691 .cache_bytes = cache_size_bytes,
1692 .memory_only = in_memory,
1693 .wipe_data = should_wipe,
1699 void Chainstate::InitCoinsCache(
size_t cache_size_bytes)
1725 if (chain.Tip() ==
nullptr) {
1734 LogPrintf(
"Leaving InitialBlockDownload (latching to false)\n");
1750 LogPrintf(
"%s: Warning: Found invalid chain at least ~6 blocks longer than our best chain.\nChain state database corruption likely.\n", __func__);
1768 LogPrintf(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n", __func__,
1773 LogPrintf(
"%s: current best=%s height=%d log2_work=%f date=%s\n", __func__,
1828 if (!setup_results)
return false;
1830 const auto [num_elems, approx_size_bytes] = *setup_results;
1831 LogPrintf(
"Using %zu MiB out of %zu MiB requested for script execution cache, able to store %zu elements\n",
1832 approx_size_bytes >> 20, max_size_bytes >> 20, num_elems);
1858 std::vector<CScriptCheck>* pvChecks)
1863 pvChecks->reserve(tx.
vin.size());
1880 std::vector<CTxOut> spent_outputs;
1881 spent_outputs.reserve(tx.
vin.size());
1883 for (
const auto& txin : tx.
vin) {
1887 spent_outputs.emplace_back(coin.
out);
1889 txdata.
Init(tx, std::move(spent_outputs));
1893 for (
unsigned int i = 0; i < tx.
vin.size(); i++) {
1904 pvChecks->emplace_back(std::move(check));
1905 }
else if (!check()) {
1933 if (cacheFullScriptStore && !pvChecks) {
1944 notifications.
fatalError(strMessage, userMessage);
1945 return state.
Error(strMessage);
1961 if (undo.nHeight == 0) {
1967 undo.nHeight = alternate.
nHeight;
1992 error(
"DisconnectBlock(): failure reading undo data");
1996 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
1997 error(
"DisconnectBlock(): block and undo data inconsistent");
2007 bool fEnforceBIP30 = !((pindex->
nHeight==91722 && pindex->
GetBlockHash() ==
uint256S(
"0x00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e")) ||
2008 (pindex->
nHeight==91812 && pindex->
GetBlockHash() ==
uint256S(
"0x00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f")));
2011 for (
int i = block.
vtx.size() - 1; i >= 0; i--) {
2015 bool is_bip30_exception = (is_coinbase && !fEnforceBIP30);
2019 for (
size_t o = 0; o < tx.
vout.size(); o++) {
2020 if (!tx.
vout[o].scriptPubKey.IsUnspendable()) {
2025 if (!is_bip30_exception) {
2036 error(
"DisconnectBlock(): transaction and undo data inconsistent");
2039 for (
unsigned int j = tx.
vin.size(); j > 0;) {
2158 const auto time_start{SteadyClock::now()};
2174 if (!
CheckBlock(block, state, params.GetConsensus(), !fJustCheck, !fJustCheck)) {
2181 return error(
"%s: Consensus::CheckBlock: %s", __func__, state.
ToString());
2192 if (block_hash == params.GetConsensus().hashGenesisBlock) {
2198 bool fScriptChecks =
true;
2207 if (it->second.GetAncestor(pindex->
nHeight) == pindex &&
2229 const auto time_1{SteadyClock::now()};
2232 Ticks<MillisecondsDouble>(time_1 - time_start),
2274 static constexpr
int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2306 fEnforceBIP30 = fEnforceBIP30 && (!pindexBIP34height || !(pindexBIP34height->
GetBlockHash() == params.GetConsensus().BIP34Hash));
2311 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2312 for (
const auto& tx : block.
vtx) {
2313 for (
size_t o = 0; o < tx->
vout.size(); o++) {
2315 LogPrintf(
"ERROR: ConnectBlock(): tried to overwrite transaction\n");
2323 int nLockTimeFlags = 0;
2331 const auto time_2{SteadyClock::now()};
2334 Ticks<MillisecondsDouble>(time_2 - time_1),
2346 std::vector<PrecomputedTransactionData> txsdata(block.
vtx.size());
2348 std::vector<int> prevheights;
2351 int64_t nSigOpsCost = 0;
2352 blockundo.
vtxundo.reserve(block.
vtx.size() - 1);
2353 for (
unsigned int i = 0; i < block.
vtx.size(); i++)
2357 nInputs += tx.
vin.size();
2371 LogPrintf(
"ERROR: %s: accumulated fee in the block out of range.\n", __func__);
2378 prevheights.resize(tx.
vin.size());
2379 for (
size_t j = 0; j < tx.
vin.size(); j++) {
2383 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2384 LogPrintf(
"ERROR: %s: contains a non-BIP68-final transaction\n", __func__);
2395 LogPrintf(
"ERROR: ConnectBlock(): too many sigops\n");
2401 std::vector<CScriptCheck> vChecks;
2402 bool fCacheResults = fJustCheck;
2404 if (fScriptChecks && !
CheckInputScripts(tx, tx_state, view,
flags, fCacheResults, fCacheResults, txsdata[i], parallel_script_checks ? &vChecks :
nullptr)) {
2408 return error(
"ConnectBlock(): CheckInputScripts on %s failed with %s",
2411 control.
Add(std::move(vChecks));
2416 blockundo.
vtxundo.emplace_back();
2420 const auto time_3{SteadyClock::now()};
2422 LogPrint(
BCLog::BENCH,
" - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) [%.2fs (%.2fms/blk)]\n", (
unsigned)block.
vtx.size(),
2423 Ticks<MillisecondsDouble>(time_3 - time_2), Ticks<MillisecondsDouble>(time_3 - time_2) / block.
vtx.size(),
2424 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_3 - time_2) / (nInputs - 1),
2429 if (block.
vtx[0]->GetValueOut() > blockReward) {
2430 LogPrintf(
"ERROR: ConnectBlock(): coinbase pays too much (actual=%d vs limit=%d)\n", block.
vtx[0]->GetValueOut(), blockReward);
2434 if (!control.
Wait()) {
2435 LogPrintf(
"ERROR: %s: CheckQueue failed\n", __func__);
2438 const auto time_4{SteadyClock::now()};
2440 LogPrint(
BCLog::BENCH,
" - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n", nInputs - 1,
2441 Ticks<MillisecondsDouble>(time_4 - time_2),
2442 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_4 - time_2) / (nInputs - 1),
2449 if (!
m_blockman.WriteUndoDataForBlock(blockundo, state, *pindex)) {
2453 const auto time_5{SteadyClock::now()};
2456 Ticks<MillisecondsDouble>(time_5 - time_4),
2468 const auto time_6{SteadyClock::now()};
2471 Ticks<MillisecondsDouble>(time_6 - time_5),
2475 TRACE6(validation, block_connected,
2490 return this->GetCoinsCacheSizeState(
2496 size_t max_coins_cache_size_bytes,
2497 size_t max_mempool_size_bytes)
2502 int64_t nTotalSpace =
2503 max_coins_cache_size_bytes + std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2506 static constexpr int64_t MAX_BLOCK_COINSDB_USAGE_BYTES = 10 * 1024 * 1024;
2507 int64_t large_threshold =
2508 std::max((9 * nTotalSpace) / 10, nTotalSpace - MAX_BLOCK_COINSDB_USAGE_BYTES);
2510 if (cacheSize > nTotalSpace) {
2511 LogPrintf(
"Cache size (%s) exceeds total space (%s)\n", cacheSize, nTotalSpace);
2513 }
else if (cacheSize > large_threshold) {
2519 bool Chainstate::FlushStateToDisk(
2522 int nManualPruneHeight)
2525 assert(this->CanFlushToDisk());
2526 std::set<int> setFilesToPrune;
2527 bool full_flush_completed =
false;
2534 bool fFlushForPrune =
false;
2535 bool fDoFullFlush =
false;
2543 std::optional<std::string> limiting_lock;
2545 for (
const auto& prune_lock :
m_blockman.m_prune_locks) {
2546 if (prune_lock.second.height_first == std::numeric_limits<int>::max())
continue;
2549 last_prune = std::max(1, std::min(last_prune, lock_height));
2550 if (last_prune == lock_height) {
2551 limiting_lock = prune_lock.first;
2555 if (limiting_lock) {
2556 LogPrint(
BCLog::PRUNE,
"%s limited pruning to height %d\n", limiting_lock.value(), last_prune);
2559 if (nManualPruneHeight > 0) {
2564 std::min(last_prune, nManualPruneHeight),
2572 if (!setFilesToPrune.empty()) {
2573 fFlushForPrune =
true;
2575 m_blockman.m_block_tree_db->WriteFlag(
"prunedblockfiles",
true);
2580 const auto nNow{SteadyClock::now()};
2597 fDoFullFlush = (mode ==
FlushStateMode::ALWAYS) || fCacheLarge || fCacheCritical || fPeriodicFlush || fFlushForPrune;
2599 if (fDoFullFlush || fPeriodicWrite) {
2624 if (fFlushForPrune) {
2632 if (fDoFullFlush && !
CoinsTip().GetBestBlock().IsNull()) {
2648 full_flush_completed =
true;
2650 int64_t{Ticks<std::chrono::microseconds>(SteadyClock::now() - nNow)},
2652 (uint64_t)coins_count,
2653 (uint64_t)coins_mem_usage,
2654 (
bool)fFlushForPrune);
2657 if (full_flush_completed) {
2661 }
catch (
const std::runtime_error& e) {
2695 const std::string& func_name,
2696 const std::string&
prefix,
2701 LogPrintf(
"%s%s: new best=%s height=%d version=0x%08x log2_work=%f tx=%lu date='%s' progress=%f cache=%.1fMiB(%utxo)%s\n",
2707 coins_tip.DynamicMemoryUsage() * (1.0 / (1 << 20)),
2708 coins_tip.GetCacheSize(),
2709 !warning_messages.empty() ?
strprintf(
" warning='%s'", warning_messages) :
"");
2712 void Chainstate::UpdateTip(
const CBlockIndex* pindexNew)
2715 const auto& coins_tip = this->
CoinsTip();
2723 constexpr
int BACKGROUND_LOG_INTERVAL = 2000;
2724 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2725 UpdateTipLog(coins_tip, pindexNew, params, __func__,
"[background validation] ",
"");
2779 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2782 return error(
"DisconnectTip(): Failed to read block");
2785 const auto time_start{SteadyClock::now()};
2789 if (DisconnectBlock(block, pindexDelete, view) !=
DISCONNECT_OK)
2791 bool flushed = view.
Flush();
2795 Ticks<MillisecondsDouble>(SteadyClock::now() - time_start));
2799 const int max_height_first{pindexDelete->
nHeight - 1};
2800 for (
auto& prune_lock :
m_blockman.m_prune_locks) {
2801 if (prune_lock.second.height_first <= max_height_first)
continue;
2803 prune_lock.second.height_first = max_height_first;
2804 LogPrint(
BCLog::PRUNE,
"%s prune lock moved back to %d\n", prune_lock.first, max_height_first);
2823 UpdateTip(pindexDelete->
pprev);
2888 const auto time_1{SteadyClock::now()};
2889 std::shared_ptr<const CBlock> pthisBlock;
2891 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
2895 pthisBlock = pblockNew;
2898 pthisBlock = pblock;
2900 const CBlock& blockConnecting = *pthisBlock;
2902 const auto time_2{SteadyClock::now()};
2903 SteadyClock::time_point time_3;
2907 Ticks<MillisecondsDouble>(time_2 - time_1));
2910 bool rv =
ConnectBlock(blockConnecting, state, pindexNew, view);
2917 time_3 = SteadyClock::now();
2921 Ticks<MillisecondsDouble>(time_3 - time_2),
2924 bool flushed = view.
Flush();
2927 const auto time_4{SteadyClock::now()};
2930 Ticks<MillisecondsDouble>(time_4 - time_3),
2937 const auto time_5{SteadyClock::now()};
2940 Ticks<MillisecondsDouble>(time_5 - time_4),
2950 UpdateTip(pindexNew);
2952 const auto time_6{SteadyClock::now()};
2956 Ticks<MillisecondsDouble>(time_6 - time_5),
2960 Ticks<MillisecondsDouble>(time_6 - time_1),
2969 m_chainman.MaybeCompleteSnapshotValidation();
2997 bool fInvalidAncestor =
false;
3007 if (fFailedChain || fMissingData) {
3014 while (pindexTest != pindexFailed) {
3018 }
else if (fMissingData) {
3023 std::make_pair(pindexFailed->
pprev, pindexFailed));
3026 pindexFailed = pindexFailed->
pprev;
3029 fInvalidAncestor =
true;
3032 pindexTest = pindexTest->
pprev;
3034 if (!fInvalidAncestor)
3066 bool fBlocksDisconnected =
false;
3080 fBlocksDisconnected =
true;
3084 std::vector<CBlockIndex*> vpindexToConnect;
3085 bool fContinue =
true;
3090 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3091 vpindexToConnect.clear();
3092 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3095 vpindexToConnect.push_back(pindexIter);
3096 pindexIter = pindexIter->
pprev;
3102 if (!
ConnectTip(state, pindexConnect, pindexConnect == pindexMostWork ? pblock : std::shared_ptr<const CBlock>(), connectTrace, disconnectpool)) {
3109 fInvalidFound =
true;
3130 if (fBlocksDisconnected) {
3151 bool fNotify =
false;
3152 bool fInitialBlockDownload =
false;
3157 pindexHeader = chainman.m_best_header;
3159 if (pindexHeader != pindexHeaderOld) {
3161 fInitialBlockDownload = chainman.IsInitialBlockDownload();
3162 pindexHeaderOld = pindexHeader;
3180 bool Chainstate::ActivateBestChain(
BlockValidationState& state, std::shared_ptr<const CBlock> pblock)
3199 LogPrintf(
"m_disabled is set - this chainstate should not be in operation. " 3206 bool exited_ibd{
false};
3219 LOCK(MempoolMutex());
3222 bool blocks_connected =
false;
3228 if (pindexMostWork ==
nullptr) {
3233 if (pindexMostWork ==
nullptr || pindexMostWork ==
m_chain.
Tip()) {
3237 bool fInvalidFound =
false;
3238 std::shared_ptr<const CBlock> nullBlockPtr;
3239 if (!
ActivateBestChainStep(state, pindexMostWork, pblock && pblock->GetHash() == pindexMostWork->
GetBlockHash() ? pblock : nullBlockPtr, fInvalidFound, connectTrace)) {
3243 blocks_connected =
true;
3245 if (fInvalidFound) {
3247 pindexMostWork =
nullptr;
3252 assert(trace.pblock && trace.pindex);
3265 if (!blocks_connected)
return true;
3270 if (was_in_ibd && !still_in_ibd) {
3320 }
while (pindexNewTip != pindexMostWork);
3360 return ActivateBestChain(state, std::shared_ptr<const CBlock>());
3370 if (pindex->
nHeight == 0)
return false;
3373 bool pindex_was_in_chain =
false;
3374 int disconnected = 0;
3388 std::multimap<const arith_uint256, CBlockIndex *> candidate_blocks_by_work;
3392 for (
auto& entry :
m_blockman.m_block_index) {
3403 candidate_blocks_by_work.insert(std::make_pair(candidate->
nChainWork, candidate));
3418 LOCK(MempoolMutex());
3420 pindex_was_in_chain =
true;
3433 if (!
ret)
return false;
3453 auto candidate_it = candidate_blocks_by_work.lower_bound(invalid_walk_tip->
pprev->
nChainWork);
3454 while (candidate_it != candidate_blocks_by_work.end()) {
3457 candidate_it = candidate_blocks_by_work.erase(candidate_it);
3465 to_mark_failed = invalid_walk_tip;
3490 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3500 if (pindex_was_in_chain) {
3519 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3520 if (!block_index.IsValid() && block_index.GetAncestor(
nHeight) == pindex) {
3526 if (&block_index ==
m_chainman.m_best_invalid) {
3535 while (pindex !=
nullptr) {
3541 pindex = pindex->
pprev;
3555 if (is_active_chainstate) {
3559 }
else if (!m_disabled) {
3564 if (snapshot_base->GetAncestor(pindex->
nHeight) == pindex) {
3574 pindexNew->
nTx = block.
vtx.size();
3576 pindexNew->nFile = pos.
nFile;
3577 pindexNew->nDataPos = pos.
nPos;
3578 pindexNew->nUndoPos = 0;
3588 std::deque<CBlockIndex*> queue;
3589 queue.push_back(pindexNew);
3592 while (!queue.empty()) {
3598 c->TryAddBlockIndexCandidate(pindex);
3600 std::pair<std::multimap<CBlockIndex*, CBlockIndex*>::iterator, std::multimap<CBlockIndex*, CBlockIndex*>::iterator> range =
m_blockman.
m_blocks_unlinked.equal_range(pindex);
3601 while (range.first != range.second) {
3602 std::multimap<CBlockIndex*, CBlockIndex*>::iterator it = range.first;
3603 queue.push_back(it->second);
3634 "hashMerkleRoot mismatch");
3643 "bad-txns-duplicate",
3644 "duplicate transaction");
3659 if (expect_witness_commitment) {
3664 assert(!block.
vtx.empty() && !block.
vtx[0]->vin.empty());
3665 const auto& witness_stack{block.
vtx[0]->vin[0].scriptWitness.stack};
3667 if (witness_stack.size() != 1 || witness_stack[0].size() != 32) {
3670 "bad-witness-nonce-size",
3671 strprintf(
"%s : invalid witness reserved value size", __func__));
3680 if (memcmp(hash_witness.
begin(), &block.
vtx[0]->vout[commitpos].scriptPubKey[6], 32)) {
3683 "bad-witness-merkle-match",
3684 strprintf(
"%s : witness merkle commitment mismatch", __func__));
3693 for (
const auto& tx : block.
vtx) {
3697 "unexpected-witness",
3698 strprintf(
"%s : unexpected witness data found", __func__));
3738 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase())
3740 for (
unsigned int i = 1; i < block.
vtx.size(); i++)
3741 if (block.
vtx[i]->IsCoinBase())
3746 for (
const auto& tx : block.
vtx) {
3756 unsigned int nSigOps = 0;
3757 for (
const auto& tx : block.
vtx)
3764 if (fCheckPOW && fCheckMerkleRoot)
3770 void ChainstateManager::UpdateUncommittedBlockStructures(
CBlock& block,
const CBlockIndex* pindexPrev)
const 3773 static const std::vector<unsigned char>
nonce(32, 0x00);
3776 tx.
vin[0].scriptWitness.stack.resize(1);
3777 tx.
vin[0].scriptWitness.stack[0] =
nonce;
3784 std::vector<unsigned char> commitment;
3786 std::vector<unsigned char>
ret(32, 0x00);
3794 out.scriptPubKey[1] = 0x24;
3795 out.scriptPubKey[2] = 0xaa;
3796 out.scriptPubKey[3] = 0x21;
3797 out.scriptPubKey[4] = 0xa9;
3798 out.scriptPubKey[5] = 0xed;
3799 memcpy(&
out.scriptPubKey[6], witnessroot.
begin(), 32);
3800 commitment = std::vector<unsigned char>(
out.scriptPubKey.begin(),
out.scriptPubKey.end());
3802 tx.vout.push_back(
out);
3805 UpdateUncommittedBlockStructures(block, pindexPrev);
3811 return std::all_of(headers.cbegin(), headers.cend(),
3812 [&](
const auto& header) {
return CheckProofOfWork(header.GetHash(), header.nBits, consensusParams);});
3823 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase()) {
3830 return std::any_of(block.
vtx.begin(), block.
vtx.end(),
3868 assert(pindexPrev !=
nullptr);
3877 if (chainman.m_options.checkpoints_enabled) {
3881 const CBlockIndex* pcheckpoint = blockman.GetLastCheckpoint(chainman.GetParams().Checkpoints());
3882 if (pcheckpoint && nHeight < pcheckpoint->
nHeight) {
3883 LogPrintf(
"ERROR: %s: forked chain older than last checkpoint (height %d)\n", __func__,
nHeight);
3916 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
3919 bool enforce_locktime_median_time_past{
false};
3921 assert(pindexPrev !=
nullptr);
3922 enforce_locktime_median_time_past =
true;
3925 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past ?
3930 for (
const auto& tx : block.
vtx) {
3940 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
3941 !std::equal(
expect.begin(),
expect.end(), block.
vtx[0]->vin[0].scriptSig.begin())) {
3977 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
3979 if (miSelf !=
m_blockman.m_block_index.end()) {
4003 pindexPrev = &((*mi).second);
4038 if (pindexPrev->
GetAncestor(failedit->nHeight) == failedit) {
4041 while (invalid_walk != failedit) {
4044 invalid_walk = invalid_walk->
pprev;
4052 if (!min_pow_checked) {
4070 "Saw new header hash=%s height=%d", hash.
ToString(), pindex->
nHeight);
4104 const double progress{100.0 * last_accepted.nHeight / (last_accepted.nHeight + blocks_left)};
4105 LogPrintf(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n", last_accepted.nHeight, progress);
4122 auto now = std::chrono::steady_clock::now();
4123 if (now < m_last_presync_update + std::chrono::milliseconds{250})
return;
4124 m_last_presync_update = now;
4128 if (initial_download) {
4130 const double progress{100.0 * height / (height + blocks_left)};
4131 LogPrintf(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n", height, progress);
4138 const CBlock& block = *pblock;
4140 if (fNewBlock) *fNewBlock =
false;
4144 CBlockIndex *&pindex = ppindex ? *ppindex : pindexDummy;
4146 bool accepted_header{
AcceptBlockHeader(block, state, &pindex, min_pow_checked)};
4149 if (!accepted_header)
4173 if (fAlreadyHave)
return true;
4175 if (pindex->
nTx != 0)
return true;
4176 if (!fHasMoreOrSameWork)
return true;
4177 if (fTooFarAhead)
return true;
4188 if (!
CheckBlock(block, state, params.GetConsensus()) ||
4203 if (fNewBlock) *fNewBlock =
true;
4206 if (blockPos.IsNull()) {
4207 state.
Error(
strprintf(
"%s: Failed to find position to write new block to disk", __func__));
4211 }
catch (
const std::runtime_error& e) {
4235 if (new_block) *new_block =
false;
4250 ret =
AcceptBlock(block, state, &pindex, force_processing,
nullptr, new_block, min_pow_checked);
4254 return error(
"%s: AcceptBlock FAILED (%s)", __func__, state.
ToString());
4262 return error(
"%s: ActivateBestChain failed (%s)", __func__, state.
ToString());
4267 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
4268 return error(
"%s: [background] ActivateBestChain failed (%s)", __func__, bg_state.
ToString());
4295 bool fCheckMerkleRoot)
4302 indexDummy.pprev = pindexPrev;
4304 indexDummy.phashBlock = &block_hash;
4308 return error(
"%s: Consensus::ContextualCheckBlockHeader: %s", __func__, state.
ToString());
4310 return error(
"%s: Consensus::CheckBlock: %s", __func__, state.
ToString());
4312 return error(
"%s: Consensus::ContextualCheckBlock: %s", __func__, state.
ToString());
4313 if (!chainstate.
ConnectBlock(block, state, &indexDummy, viewNew,
true)) {
4325 if (!active_chainstate.FlushStateToDisk(
4338 if (tip && tip->GetBlockHash() == coins_cache.
GetBestBlock()) {
4351 LogPrintf(
"Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f\n",
4352 tip->GetBlockHash().ToString(),
4374 int nCheckLevel,
int nCheckDepth)
4383 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
4386 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
4387 LogPrintf(
"Verifying last %i blocks at level %i\n", nCheckDepth, nCheckLevel);
4391 int nGoodTransactions = 0;
4394 bool skipped_no_block_data{
false};
4395 bool skipped_l3_checks{
false};
4396 LogPrintf(
"Verification progress: 0%%\n");
4401 const int percentageDone = std::max(1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (double)nCheckDepth * (nCheckLevel >= 4 ? 50 : 100))));
4402 if (reportDone < percentageDone / 10) {
4404 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
4405 reportDone = percentageDone / 10;
4414 LogPrintf(
"VerifyDB(): block verification stopping at height %d (no data). This could be due to pruning or use of an assumeutxo snapshot.\n", pindex->
nHeight);
4415 skipped_no_block_data =
true;
4425 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensus_params)) {
4426 LogPrintf(
"Verification error: found bad block at %d, hash=%s (%s)\n",
4431 if (nCheckLevel >= 2 && pindex) {
4443 if (nCheckLevel >= 3) {
4452 nGoodTransactions = 0;
4453 pindexFailure = pindex;
4455 nGoodTransactions += block.
vtx.size();
4458 skipped_l3_checks =
true;
4463 if (pindexFailure) {
4464 LogPrintf(
"Verification error: coin database inconsistencies found (last %i blocks, %i good transactions before that)\n", chainstate.
m_chain.
Height() - pindexFailure->
nHeight + 1, nGoodTransactions);
4467 if (skipped_l3_checks) {
4468 LogPrintf(
"Skipped verification of level >=3 (insufficient database cache size). Consider increasing -dbcache.\n");
4475 if (nCheckLevel >= 4 && !skipped_l3_checks) {
4477 const int percentageDone = std::max(1, std::min(99, 100 - (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (double)nCheckDepth * 50)));
4478 if (reportDone < percentageDone / 10) {
4480 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
4481 reportDone = percentageDone / 10;
4490 if (!chainstate.
ConnectBlock(block, state, pindex, coins)) {
4498 LogPrintf(
"Verification: No coin database inconsistencies in last %i blocks (%i transactions)\n", block_count, nGoodTransactions);
4500 if (skipped_l3_checks) {
4503 if (skipped_no_block_data) {
4520 if (!tx->IsCoinBase()) {
4521 for (
const CTxIn &txin : tx->vin) {
4539 if (hashHeads.empty())
return true;
4540 if (hashHeads.size() != 2)
return error(
"ReplayBlocks(): unknown inconsistent state");
4549 if (
m_blockman.m_block_index.count(hashHeads[0]) == 0) {
4550 return error(
"ReplayBlocks(): reorganization to unknown block requested");
4552 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
4554 if (!hashHeads[1].IsNull()) {
4555 if (
m_blockman.m_block_index.count(hashHeads[1]) == 0) {
4556 return error(
"ReplayBlocks(): reorganization from unknown block requested");
4558 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
4560 assert(pindexFork !=
nullptr);
4564 while (pindexOld != pindexFork) {
4580 pindexOld = pindexOld->
pprev;
4584 int nForkHeight = pindexFork ? pindexFork->
nHeight : 0;
4611 block = block->pprev;
4617 void Chainstate::ClearBlockIndexCandidates()
4630 if (!
ret)
return false;
4632 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
4634 std::vector<CBlockIndex*> vSortedByHeight{
m_blockman.GetAllBlockIndices()};
4635 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
4645 if (pindex == GetSnapshotBaseBlock() ||
4654 m_best_invalid = pindex;
4657 m_best_header = pindex;
4660 needs_init =
m_blockman.m_block_index.empty();
4670 LogPrintf(
"Initializing databases...\n");
4685 if (
m_blockman.m_block_index.count(params.GenesisBlock().GetHash()))
4689 const CBlock& block = params.GenesisBlock();
4691 if (blockPos.IsNull()) {
4692 return error(
"%s: writing genesis block to disk failed", __func__);
4696 }
catch (
const std::runtime_error& e) {
4697 return error(
"%s: failed to write genesis block: %s", __func__, e.what());
4706 std::multimap<uint256, FlatFilePos>* blocks_with_unknown_parent)
4709 assert(!dbp == !blocks_with_unknown_parent);
4711 const auto start{SteadyClock::now()};
4719 uint64_t nRewind = blkdat.GetPos();
4720 while (!blkdat.eof()) {
4723 blkdat.SetPos(nRewind);
4726 unsigned int nSize = 0;
4730 blkdat.FindByte(std::byte(params.MessageStart()[0]));
4731 nRewind = blkdat.GetPos() + 1;
4733 if (buf != params.MessageStart()) {
4740 }
catch (
const std::exception&) {
4747 const uint64_t nBlockPos{blkdat.GetPos()};
4749 dbp->
nPos = nBlockPos;
4750 blkdat.SetLimit(nBlockPos + nSize);
4753 const uint256 hash{header.GetHash()};
4756 nRewind = nBlockPos + nSize;
4757 blkdat.SkipTo(nRewind);
4759 std::shared_ptr<CBlock> pblock{};
4766 header.hashPrevBlock.ToString());
4767 if (dbp && blocks_with_unknown_parent) {
4768 blocks_with_unknown_parent->emplace(header.hashPrevBlock, *dbp);
4777 blkdat.SetPos(nBlockPos);
4778 pblock = std::make_shared<CBlock>();
4780 nRewind = blkdat.GetPos();
4783 if (
AcceptBlock(pblock, state,
nullptr,
true, dbp,
nullptr,
true)) {
4789 }
else if (hash != params.GetConsensus().hashGenesisBlock && pindex->
nHeight % 1000 == 0) {
4795 if (hash == params.GetConsensus().hashGenesisBlock) {
4796 bool genesis_activation_failure =
false;
4797 for (
auto c :
GetAll()) {
4799 if (!c->ActivateBestChain(state,
nullptr)) {
4800 genesis_activation_failure =
true;
4804 if (genesis_activation_failure) {
4817 bool activation_failure =
false;
4818 for (
auto c :
GetAll()) {
4820 if (!c->ActivateBestChain(state, pblock)) {
4822 activation_failure =
true;
4826 if (activation_failure) {
4833 if (!blocks_with_unknown_parent)
continue;
4836 std::deque<uint256> queue;
4837 queue.push_back(hash);
4838 while (!queue.empty()) {
4841 auto range = blocks_with_unknown_parent->equal_range(head);
4842 while (range.first != range.second) {
4843 std::multimap<uint256, FlatFilePos>::iterator it = range.first;
4844 std::shared_ptr<CBlock> pblockrecursive = std::make_shared<CBlock>();
4846 LogPrint(
BCLog::REINDEX,
"%s: Processing out of order child %s of %s\n", __func__, pblockrecursive->GetHash().ToString(),
4850 if (
AcceptBlock(pblockrecursive, dummy,
nullptr,
true, &it->second,
nullptr,
true)) {
4852 queue.push_back(pblockrecursive->GetHash());
4856 blocks_with_unknown_parent->erase(it);
4860 }
catch (
const std::exception& e) {
4872 LogPrint(
BCLog::REINDEX,
"%s: unexpected data at file offset 0x%x - %s. continuing\n", __func__, (nRewind - 1), e.what());
4875 }
catch (
const std::runtime_error& e) {
4878 LogPrintf(
"Loaded %i blocks from external file in %dms\n", nLoaded, Ticks<std::chrono::milliseconds>(SteadyClock::now() - start));
4898 std::multimap<CBlockIndex*,CBlockIndex*> forward;
4899 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
4900 forward.emplace(block_index.pprev, &block_index);
4905 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> rangeGenesis = forward.equal_range(
nullptr);
4907 rangeGenesis.first++;
4908 assert(rangeGenesis.first == rangeGenesis.second);
4919 CBlockIndex* pindexFirstNotTransactionsValid =
nullptr;
4921 CBlockIndex* pindexFirstNotScriptsValid =
nullptr;
4923 while (pindex !=
nullptr) {
4931 || (pindex->
nChainTx == 0 && prev_chain_tx == 0 && pindex->
pprev)
4935 if (pindexFirstAssumeValid ==
nullptr && pindex->nStatus &
BLOCK_ASSUMED_VALID) pindexFirstAssumeValid = pindex;
4936 if (pindexFirstInvalid ==
nullptr && pindex->nStatus &
BLOCK_FAILED_VALID) pindexFirstInvalid = pindex;
4937 if (pindexFirstMissing ==
nullptr && !(pindex->nStatus &
BLOCK_HAVE_DATA)) {
4938 pindexFirstMissing = pindex;
4940 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) pindexFirstNeverProcessed = pindex;
4947 if (pindexFirstNotTransactionsValid ==
nullptr &&
4949 pindexFirstNotTransactionsValid = pindex;
4952 if (pindexFirstNotChainValid ==
nullptr &&
4954 pindexFirstNotChainValid = pindex;
4957 if (pindexFirstNotScriptsValid ==
nullptr &&
4959 pindexFirstNotScriptsValid = pindex;
4964 if (pindex->
pprev ==
nullptr) {
4967 for (
auto c :
GetAll()) {
4968 if (c->m_chain.Genesis() !=
nullptr) {
4969 assert(pindex == c->m_chain.Genesis());
4981 if (pindexFirstAssumeValid ==
nullptr) {
4986 assert(pindexFirstMissing == pindexFirstNeverProcessed);
5009 assert(pindexFirstNotTreeValid ==
nullptr);
5013 if (pindexFirstInvalid ==
nullptr) {
5018 for (
auto c :
GetAll()) {
5019 if (c->m_chain.Tip() ==
nullptr)
continue;
5021 if (pindexFirstInvalid ==
nullptr) {
5028 if ((pindexFirstMissing ==
nullptr || pindex == c->m_chain.Tip())) {
5032 if (is_active || GetSnapshotBaseBlock()->GetAncestor(pindex->
nHeight) == pindex) {
5033 assert(c->setBlockIndexCandidates.count(pindex));
5041 assert(c->setBlockIndexCandidates.count(pindex) == 0);
5045 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> rangeUnlinked =
m_blockman.
m_blocks_unlinked.equal_range(pindex->
pprev);
5046 bool foundInUnlinked =
false;
5047 while (rangeUnlinked.first != rangeUnlinked.second) {
5048 assert(rangeUnlinked.first->first == pindex->
pprev);
5049 if (rangeUnlinked.first->second == pindex) {
5050 foundInUnlinked =
true;
5053 rangeUnlinked.first++;
5055 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed !=
nullptr && pindexFirstInvalid ==
nullptr) {
5060 if (pindexFirstMissing ==
nullptr)
assert(!foundInUnlinked);
5061 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed ==
nullptr && pindexFirstMissing !=
nullptr) {
5072 for (
auto c :
GetAll()) {
5075 if (pindexFirstInvalid ==
nullptr) {
5076 if (is_active || GetSnapshotBaseBlock()->GetAncestor(pindex->
nHeight) == pindex) {
5087 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> range = forward.equal_range(pindex);
5088 if (range.first != range.second) {
5090 pindex = range.first->second;
5099 if (pindex == pindexFirstInvalid) pindexFirstInvalid =
nullptr;
5100 if (pindex == pindexFirstMissing) pindexFirstMissing =
nullptr;
5101 if (pindex == pindexFirstNeverProcessed) pindexFirstNeverProcessed =
nullptr;
5102 if (pindex == pindexFirstNotTreeValid) pindexFirstNotTreeValid =
nullptr;
5103 if (pindex == pindexFirstNotTransactionsValid) pindexFirstNotTransactionsValid =
nullptr;
5104 if (pindex == pindexFirstNotChainValid) pindexFirstNotChainValid =
nullptr;
5105 if (pindex == pindexFirstNotScriptsValid) pindexFirstNotScriptsValid =
nullptr;
5106 if (pindex == pindexFirstAssumeValid) pindexFirstAssumeValid =
nullptr;
5110 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> rangePar = forward.equal_range(pindexPar);
5111 while (rangePar.first->second != pindex) {
5112 assert(rangePar.first != rangePar.second);
5117 if (rangePar.first != rangePar.second) {
5119 pindex = rangePar.first->second;
5131 assert(nNodes == forward.size());
5138 return strprintf(
"Chainstate [%s] @ height %d (%s)",
5143 bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size)
5156 LogPrintf(
"[%s] resized coinsdb cache to %.1f MiB\n",
5157 this->
ToString(), coinsdb_size * (1.0 / 1024 / 1024));
5158 LogPrintf(
"[%s] resized coinstip cache to %.1f MiB\n",
5159 this->
ToString(), coinstip_size * (1.0 / 1024 / 1024));
5164 if (coinstip_size > old_coinstip_size) {
5177 if (pindex ==
nullptr)
5180 int64_t nNow = time(
nullptr);
5190 return std::min<double>(pindex->
nChainTx / fTxTotal, 1.0);
5196 if (m_active_chainstate && m_active_chainstate->m_from_snapshot_blockhash) {
5198 return m_active_chainstate->m_from_snapshot_blockhash;
5200 return std::nullopt;
5206 std::vector<Chainstate*>
out;
5208 for (
Chainstate*
cs : {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
5218 assert(!m_ibd_chainstate);
5219 assert(!m_active_chainstate);
5221 m_ibd_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman, *
this);
5222 m_active_chainstate = m_ibd_chainstate.get();
5223 return *m_active_chainstate;
5235 bool existed = fs::remove(base_blockhash_path);
5237 LogPrintf(
"[snapshot] snapshot chainstate dir being removed lacks %s file\n",
5240 }
catch (
const fs::filesystem_error& e) {
5241 LogPrintf(
"[snapshot] failed to remove file %s: %s\n",
5247 LogPrintf(
"Removing leveldb dir at %s\n", path_str);
5251 const bool destroyed =
DestroyDB(path_str);
5254 LogPrintf(
"error: leveldb DestroyDB call failed on %s\n", path_str);
5274 LogPrintf(
"[snapshot] can't activate a snapshot-based chainstate more than once\n");
5280 if (
Assert(m_active_chainstate->GetMempool())->size() > 0) {
5281 LogPrintf(
"[snapshot] can't activate a snapshot when mempool not empty\n");
5286 int64_t current_coinsdb_cache_size{0};
5287 int64_t current_coinstip_cache_size{0};
5295 static constexpr
double IBD_CACHE_PERC = 0.01;
5296 static constexpr
double SNAPSHOT_CACHE_PERC = 0.99;
5308 current_coinsdb_cache_size = this->
ActiveChainstate().m_coinsdb_cache_size_bytes;
5309 current_coinstip_cache_size = this->
ActiveChainstate().m_coinstip_cache_size_bytes;
5314 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
5315 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
5319 return std::make_unique<Chainstate>(
5320 nullptr,
m_blockman, *
this, base_blockhash));
5324 snapshot_chainstate->InitCoinsDB(
5325 static_cast<size_t>(current_coinsdb_cache_size * SNAPSHOT_CACHE_PERC),
5326 in_memory,
false,
"chainstate");
5327 snapshot_chainstate->InitCoinsCache(
5328 static_cast<size_t>(current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
5332 LogPrintf(
"[snapshot] activation failed - %s\n", reason);
5333 this->MaybeRebalanceCaches();
5341 snapshot_chainstate.reset();
5345 "Manually remove it before restarting.\n",
fs::PathToString(*snapshot_datadir)));
5353 return cleanup_bad_snapshot(
"population failed");
5362 return cleanup_bad_snapshot(
"work does not exceed active chainstate");
5368 return cleanup_bad_snapshot(
"could not write base blockhash");
5372 assert(!m_snapshot_chainstate);
5373 m_snapshot_chainstate.swap(snapshot_chainstate);
5374 const bool chaintip_loaded = m_snapshot_chainstate->LoadChainTip();
5379 Assert(m_active_chainstate->m_mempool->size() == 0);
5380 Assert(!m_snapshot_chainstate->m_mempool);
5381 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
5382 m_active_chainstate->m_mempool =
nullptr;
5383 m_active_chainstate = m_snapshot_chainstate.get();
5386 LogPrintf(
"[snapshot] successfully activated snapshot %s\n", base_blockhash.
ToString());
5388 m_snapshot_chainstate->CoinsTip().DynamicMemoryUsage() / (1000 * 1000));
5390 this->MaybeRebalanceCaches();
5398 snapshot_loaded ?
"saving snapshot chainstate" :
"flushing coins cache",
5402 coins_cache.
Flush();
5407 const char*
what() const noexcept
override 5409 return "ComputeUTXOStats interrupted.";
5431 if (!snapshot_start_block) {
5434 LogPrintf(
"[snapshot] Did not find snapshot start blockheader %s\n",
5435 base_blockhash.ToString());
5439 int base_height = snapshot_start_block->
nHeight;
5442 if (!maybe_au_data) {
5443 LogPrintf(
"[snapshot] assumeutxo height in snapshot metadata not recognized " 5444 "(%d) - refusing to load snapshot\n", base_height);
5454 LogPrintf(
"[snapshot] activation failed - work does not exceed active chainstate\n");
5463 LogPrintf(
"[snapshot] loading coins from snapshot %s\n", base_blockhash.ToString());
5464 int64_t coins_processed{0};
5466 while (coins_left > 0) {
5468 coins_file >> outpoint;
5470 }
catch (
const std::ios_base::failure&) {
5471 LogPrintf(
"[snapshot] bad snapshot format or truncated snapshot after deserializing %d coins\n",
5472 coins_count - coins_left);
5475 if (coin.
nHeight > base_height ||
5476 outpoint.
n >= std::numeric_limits<decltype(outpoint.
n)>::max()
5478 LogPrintf(
"[snapshot] bad snapshot data after deserializing %d coins\n",
5479 coins_count - coins_left);
5483 LogPrintf(
"[snapshot] bad snapshot data after deserializing %d coins - bad tx out value\n",
5484 coins_count - coins_left);
5493 if (coins_processed % 1000000 == 0) {
5494 LogPrintf(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)\n",
5496 static_cast<float>(coins_processed) * 100 / static_cast<float>(coins_count),
5504 if (coins_processed % 120000 == 0) {
5510 return snapshot_chainstate.GetCoinsCacheSizeState());
5531 bool out_of_coins{
false};
5533 coins_file >> outpoint;
5534 }
catch (
const std::ios_base::failure&) {
5536 out_of_coins =
true;
5538 if (!out_of_coins) {
5539 LogPrintf(
"[snapshot] bad snapshot - coins left over after deserializing %d coins\n",
5544 LogPrintf(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s\n",
5547 base_blockhash.ToString());
5558 std::optional<CCoinsStats> maybe_stats;
5566 if (!maybe_stats.has_value()) {
5567 LogPrintf(
"[snapshot] failed to generate coins stats\n");
5573 LogPrintf(
"[snapshot] bad snapshot content hash: expected %s, got %s\n",
5590 constexpr
int AFTER_GENESIS_START{1};
5592 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height(); ++i) {
5593 index = snapshot_chainstate.
m_chain[i];
5628 LogPrintf(
"[snapshot] validated snapshot (%.2f MB)\n",
5651 !this->
IsUsable(m_snapshot_chainstate.get()) ||
5652 !this->
IsUsable(m_ibd_chainstate.get()) ||
5653 !m_ibd_chainstate->m_chain.Tip()) {
5659 const int snapshot_base_height = *
Assert(this->GetSnapshotBaseHeight());
5662 if (index_new.
nHeight < snapshot_base_height) {
5672 "%s failed to validate the -assumeutxo snapshot state. " 5673 "This indicates a hardware problem, or a bug in the software, or a " 5674 "bad software modification that allowed an invalid snapshot to be " 5675 "loaded. As a result of this, the node will shut down and stop using any " 5676 "state that was built on the snapshot, resetting the chain height " 5677 "from %d to %d. On the next " 5678 "restart, the node will resume syncing from %d " 5679 "without using any snapshot data. " 5680 "Please report this incident to %s, including how you obtained the snapshot. " 5681 "The invalid snapshot chainstate will be left on disk in case it is " 5682 "helpful in diagnosing the issue that caused this error."),
5687 LogPrintf(
"[snapshot] deleting snapshot, reverting to validated chain, and stopping node\n");
5689 m_active_chainstate = m_ibd_chainstate.get();
5690 m_snapshot_chainstate->m_disabled =
true;
5694 auto rename_result = m_snapshot_chainstate->InvalidateCoinsDBOnDisk();
5695 if (!rename_result) {
5703 LogPrintf(
"[snapshot] supposed base block %s does not match the " 5704 "snapshot base block %s (height %d). Snapshot is not valid.\n",
5705 index_new.
ToString(), snapshot_blockhash.
ToString(), snapshot_base_height);
5706 handle_invalid_snapshot();
5712 int curr_height = m_ibd_chainstate->m_chain.Height();
5714 assert(snapshot_base_height == curr_height);
5719 CCoinsViewDB& ibd_coins_db = m_ibd_chainstate->CoinsDB();
5720 m_ibd_chainstate->ForceFlushStateToDisk();
5723 if (!maybe_au_data) {
5724 LogPrintf(
"[snapshot] assumeutxo data not found for height " 5725 "(%d) - refusing to validate snapshot\n", curr_height);
5726 handle_invalid_snapshot();
5731 std::optional<CCoinsStats> maybe_ibd_stats;
5732 LogPrintf(
"[snapshot] computing UTXO stats for background chainstate to validate " 5733 "snapshot - this could take a few minutes\n");
5736 CoinStatsHashType::HASH_SERIALIZED,
5745 if (!maybe_ibd_stats) {
5746 LogPrintf(
"[snapshot] failed to generate stats for validation coins db\n");
5750 handle_invalid_snapshot();
5753 const auto& ibd_stats = *maybe_ibd_stats;
5762 LogPrintf(
"[snapshot] hash mismatch: actual=%s, expected=%s\n",
5763 ibd_stats.hashSerialized.ToString(),
5765 handle_invalid_snapshot();
5769 LogPrintf(
"[snapshot] snapshot beginning at %s has been fully validated\n",
5772 m_ibd_chainstate->m_disabled =
true;
5773 this->MaybeRebalanceCaches();
5781 assert(m_active_chainstate);
5782 return *m_active_chainstate;
5788 return m_snapshot_chainstate && m_active_chainstate == m_snapshot_chainstate.get();
5791 void ChainstateManager::MaybeRebalanceCaches()
5794 bool ibd_usable = this->
IsUsable(m_ibd_chainstate.get());
5795 bool snapshot_usable = this->
IsUsable(m_snapshot_chainstate.get());
5796 assert(ibd_usable || snapshot_usable);
5798 if (ibd_usable && !snapshot_usable) {
5803 else if (snapshot_usable && !ibd_usable) {
5805 LogPrintf(
"[snapshot] allocating all cache to the snapshot chainstate\n");
5809 else if (ibd_usable && snapshot_usable) {
5814 m_ibd_chainstate->ResizeCoinsCaches(
5816 m_snapshot_chainstate->ResizeCoinsCaches(
5819 m_snapshot_chainstate->ResizeCoinsCaches(
5821 m_ibd_chainstate->ResizeCoinsCaches(
5827 void ChainstateManager::ResetChainstates()
5829 m_ibd_chainstate.reset();
5830 m_snapshot_chainstate.reset();
5831 m_active_chainstate =
nullptr;
5841 if (!opts.check_block_index.has_value()) opts.
check_block_index = opts.chainparams.DefaultConsistencyChecks();
5842 if (!opts.minimum_chain_work.has_value()) opts.minimum_chain_work =
UintToArith256(opts.chainparams.GetConsensus().nMinimumChainWork);
5843 if (!opts.assumed_valid_block.has_value()) opts.assumed_valid_block = opts.chainparams.GetConsensus().defaultAssumeValid;
5844 Assert(opts.adjusted_time_callback);
5845 return std::move(opts);
5849 : m_interrupt{interrupt},
5850 m_options{
Flatten(std::move(options))},
5851 m_blockman{interrupt, std::move(blockman_options)} {}
5860 bool ChainstateManager::DetectSnapshotChainstate()
5862 assert(!m_snapshot_chainstate);
5868 if (!base_blockhash) {
5871 LogPrintf(
"[snapshot] detected active snapshot chainstate (%s) - loading\n",
5874 this->ActivateExistingSnapshot(*base_blockhash);
5880 assert(!m_snapshot_chainstate);
5881 m_snapshot_chainstate =
5882 std::make_unique<Chainstate>(
nullptr,
m_blockman, *
this, base_blockhash);
5883 LogPrintf(
"[snapshot] switching active chainstate to %s\n", m_snapshot_chainstate->ToString());
5886 Assert(m_active_chainstate->m_mempool->size() == 0);
5887 Assert(!m_snapshot_chainstate->m_mempool);
5888 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
5889 m_active_chainstate->m_mempool =
nullptr;
5890 m_active_chainstate = m_snapshot_chainstate.get();
5891 return *m_snapshot_chainstate;
5896 return (block_index.
nHeight==91842 && block_index.
GetBlockHash() ==
uint256S(
"0x00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec")) ||
5897 (block_index.
nHeight==91880 && block_index.
GetBlockHash() ==
uint256S(
"0x00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721"));
5902 return (block_index.
nHeight==91722 && block_index.
GetBlockHash() ==
uint256S(
"0x00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e")) ||
5903 (block_index.
nHeight==91812 && block_index.
GetBlockHash() ==
uint256S(
"0x00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"));
5910 assert(
cs.m_from_snapshot_blockhash);
5911 auto storage_path_maybe =
cs.CoinsDB().StoragePath();
5913 assert(storage_path_maybe);
5914 return *storage_path_maybe;
5924 auto invalid_path = snapshot_datadir +
"_INVALID";
5927 LogPrintf(
"[snapshot] renaming snapshot datadir %s to %s\n", dbpath, target);
5933 fs::rename(snapshot_datadir, invalid_path);
5934 }
catch (
const fs::filesystem_error& e) {
5938 LogPrintf(
"%s: error renaming file '%s' -> '%s': %s\n",
5939 __func__, src_str, dest_str, e.what());
5941 "Rename of '%s' -> '%s' failed. " 5942 "You should resolve this by manually moving or deleting the invalid " 5943 "snapshot directory %s, otherwise you will encounter the same error again " 5944 "on the next startup."),
5945 src_str, dest_str, src_str)};
5950 bool ChainstateManager::DeleteSnapshotChainstate()
5953 Assert(m_snapshot_chainstate);
5954 Assert(m_ibd_chainstate);
5958 LogPrintf(
"Deletion of %s failed. Please remove it manually to continue reindexing.\n",
5962 m_active_chainstate = m_ibd_chainstate.get();
5963 m_snapshot_chainstate.reset();
5977 const CBlockIndex* ChainstateManager::GetSnapshotBaseBlock()
const 5979 return m_active_chainstate ? m_active_chainstate->SnapshotBase() :
nullptr;
5982 std::optional<int> ChainstateManager::GetSnapshotBaseHeight()
const 5984 const CBlockIndex* base = this->GetSnapshotBaseBlock();
5985 return base ? std::make_optional(base->
nHeight) :
std::nullopt;
5988 bool ChainstateManager::ValidatedSnapshotCleanup()
5997 std::optional<fs::path> ibd_chainstate_path_maybe = get_storage_path(m_ibd_chainstate);
5998 std::optional<fs::path> snapshot_chainstate_path_maybe = get_storage_path(m_snapshot_chainstate);
6007 if (!ibd_chainstate_path_maybe || !snapshot_chainstate_path_maybe) {
6008 LogPrintf(
"[snapshot] snapshot chainstate cleanup cannot happen with " 6009 "in-memory chainstates. You are testing, right?\n");
6013 const auto& snapshot_chainstate_path = *snapshot_chainstate_path_maybe;
6014 const auto& ibd_chainstate_path = *ibd_chainstate_path_maybe;
6022 this->ResetChainstates();
6027 LogPrintf(
"[snapshot] deleting background chainstate directory (now unnecessary) (%s)\n",
6030 fs::path tmp_old{ibd_chainstate_path +
"_todelete"};
6032 auto rename_failed_abort = [
this](
6035 const fs::filesystem_error& err) {
6036 LogPrintf(
"%s: error renaming file (%s): %s\n",
6039 "Rename of '%s' -> '%s' failed. " 6040 "Cannot clean up the background chainstate leveldb directory.",
6045 fs::rename(ibd_chainstate_path, tmp_old);
6046 }
catch (
const fs::filesystem_error& e) {
6047 rename_failed_abort(ibd_chainstate_path, tmp_old, e);
6051 LogPrintf(
"[snapshot] moving snapshot chainstate (%s) to " 6052 "default chainstate directory (%s)\n",
6056 fs::rename(snapshot_chainstate_path, ibd_chainstate_path);
6057 }
catch (
const fs::filesystem_error& e) {
6058 rename_failed_abort(snapshot_chainstate_path, ibd_chainstate_path, e);
6065 LogPrintf(
"Deletion of %s failed. Please remove it manually, as the " 6066 "directory is now unnecessary.\n",
6069 LogPrintf(
"[snapshot] deleted background chainstate directory (%s)\n",
6075 Chainstate& ChainstateManager::GetChainstateForIndexing()
6080 return (this->
GetAll().size() > 1) ? *m_ibd_chainstate : *m_active_chainstate;
6083 std::pair<int, int> ChainstateManager::GetPruneRange(
const Chainstate& chainstate,
int last_height_can_prune)
6090 if (this->
GetAll().size() > 1 && m_snapshot_chainstate.get() == &chainstate) {
6093 prune_start = *
Assert(GetSnapshotBaseHeight()) + 1;
6096 int max_prune = std::max<int>(
6105 int prune_end = std::min(last_height_can_prune, max_prune);
6107 return {prune_start, prune_end};
std::optional< std::string > PaysForRBF(CAmount original_fees, CAmount replacement_fees, size_t replacement_vsize, CFeeRate relay_fee, const uint256 &txid)
The replacement transaction must pay more fees than the original transactions.
std::shared_ptr< const CTransaction > CTransactionRef
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
static SteadyClock::duration time_index
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
const Coin & AccessByTxid(const CCoinsViewCache &view, const uint256 &txid)
Utility function to find any unspent output with a given txid.
CoinsViewOptions coins_view
static unsigned int GetBlockScriptFlags(const CBlockIndex &block_index, const ChainstateManager &chainman)
CSHA256 & Write(const unsigned char *data, size_t len)
static void UpdateTipLog(const CCoinsViewCache &coins_tip, const CBlockIndex *tip, const CChainParams ¶ms, const std::string &func_name, const std::string &prefix, const std::string &warning_messages) EXCLUSIVE_LOCKS_REQUIRED(
void PruneBlockFilesManual(Chainstate &active_chainstate, int nManualPruneHeight)
Prune block files up to a given height.
bool IsSpent() const
Either this coin never existed (see e.g.
node::BlockManager m_blockman
A single BlockManager instance is shared across each constructed chainstate to avoid duplicating bloc...
const std::optional< uint256 > m_from_snapshot_blockhash
The blockhash which is the base of the snapshot this chainstate was created from. ...
const uint256 & AssumedValidBlock() const
static constexpr unsigned int LOCKTIME_VERIFY_SEQUENCE
Flags for nSequence and nLockTime locks.
std::optional< int > m_snapshot_height
The height of the base block of an assumeutxo snapshot, if one is in use.
fs::path path
Location in the filesystem where leveldb data will be stored.
const std::vector< std::string > CHECKLEVEL_DOC
Documentation for argument 'checklevel'.
std::vector< Coin > vprevout
void InvalidChainFound(CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
std::string ToString() const
int64_t EndTime(const Consensus::Params ¶ms) const override
static constexpr std::chrono::hours MAX_FEE_ESTIMATION_TIP_AGE
Maximum age of our tip for us to be considered current for fee estimation.
CTxMemPool * m_mempool
Optional mempool that is kept in sync with the chain.
int32_t nSequenceId
(memory only) Sequential id assigned to distinguish order in which blocks are received.
bool DetectSnapshotChainstate() EXCLUSIVE_LOCKS_REQUIRED(void ResetChainstates() EXCLUSIVE_LOCKS_REQUIRED(bool DeleteSnapshotChainstate() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &ActivateExistingSnapshot(uint256 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( ~ChainstateManager()
When starting up, search the datadir for a chainstate based on a UTXO snapshot that is in the process...
bool ReplayBlocks()
Replay blocks that aren't fully applied to the database.
void SyncWithValidationInterfaceQueue()
This is a synonym for the following, which asserts certain locks are not held: std::promise<void> pro...
bool PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
static const int SERIALIZE_TRANSACTION_NO_WITNESS
A flag that is ORed into the protocol version to designate that a transaction should be (un)serialize...
invalid by consensus rules
bool CheckTxInputs(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, int nSpendHeight, CAmount &txfee)
Check whether all inputs of this transaction are valid (no double spends and amounts) This does not m...
static bool CheckWitnessMalleation(const CBlock &block, bool expect_witness_commitment, BlockValidationState &state)
CheckWitnessMalleation performs checks for block malleation with regard to its witnesses.
void UpdateLockPoints(const LockPoints &lp)
bool HaveNumChainTxs() const
Check whether this block's and all previous blocks' transactions have been downloaded (and stored to ...
std::chrono::time_point< NodeClock > time_point
CBlockIndex * pskip
pointer to the index of some further predecessor of this block
std::optional< std::string > PaysMoreThanConflicts(const CTxMemPool::setEntries &iters_conflicting, CFeeRate replacement_feerate, const uint256 &txid)
Check that the feerate of the replacement transaction(s) is higher than the feerate of each of the tr...
CAmount GetBlockSubsidy(int nHeight, const Consensus::Params &consensusParams)
static GenTxid Wtxid(const uint256 &hash)
static SteadyClock::duration time_chainstate
bool m_check_for_pruning
Global flag to indicate we should check to see if there are block/undo files that should be deleted...
std::condition_variable g_best_block_cv
SteadyClock::time_point m_last_flush
static void FlushSnapshotToDisk(CCoinsViewCache &coins_cache, bool snapshot_loaded)
bool Error(const std::string &reject_reason)
SynchronizationState
Current sync state passed to tip changed callbacks.
static SteadyClock::duration time_forks
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 const int WITNESS_SCALE_FACTOR
bool IsPruneMode() const
Whether running in -prune mode.
void Finalize(Span< unsigned char > output)
std::string GetDebugMessage() const
#define LogPrint(category,...)
int64_t GetBlockTime() const
indexed_transaction_set::nth_index< 0 >::type::const_iterator txiter
int Threshold(const Consensus::Params ¶ms) const override
bool LoadGenesisBlock()
Ensures we have a genesis block in the block tree, possibly writing one to disk.
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
double dTxRate
estimated number of transactions per second after that timestamp
virtual bool GetCoin(const COutPoint &outpoint, Coin &coin) const
Retrieve the Coin (unspent transaction output) for a given outpoint.
descends from failed block
CBlockIndex * pprev
pointer to the index of the predecessor of this block
const Coin & AccessCoin(const COutPoint &output) const
Return a reference to Coin in the cache, or coinEmpty if not found.
The package itself is invalid (e.g. too many transactions).
bool Flush()
Push the modifications applied to this cache to its base and wipe local state.
std::atomic_bool fReindex
std::optional< std::string > HasNoNewUnconfirmed(const CTransaction &tx, const CTxMemPool &pool, const CTxMemPool::setEntries &iters_conflicting)
The replacement transaction may only include an unconfirmed input if that input was included in one o...
void BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances...
bool exists(const GenTxid >xid) const
std::function< void()> restart_indexes
Function to restart active indexes; set dynamically to avoid a circular dependency on base/index...
void UpdateTransactionsFromBlock(const std::vector< uint256 > &vHashesToUpdate) EXCLUSIVE_LOCKS_REQUIRED(cs
UpdateTransactionsFromBlock is called when adding transactions from a disconnected block back to the ...
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...
We don't have the previous block the checked one is built on.
int64_t BeginTime(const Consensus::Params ¶ms) const override
RecursiveMutex cs_LastBlockFile
The cache is at >= 90% capacity.
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
Options struct containing limit options for a CTxMemPool.
An in-memory indexed chain of blocks.
const int64_t m_max_size_bytes
bool ShouldCheckBlockIndex() const
size_t DynamicMemoryUsage() const
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, const CScriptWitness *witness, unsigned int flags, const BaseSignatureChecker &checker, ScriptError *serror)
static const int32_t VERSIONBITS_TOP_MASK
What bitmask determines whether versionbits is in use.
reverse_range< T > reverse_iterate(T &x)
uint256 GetRandHash() noexcept
int Period(const Consensus::Params ¶ms) const override
bool DestroyDB(const std::string &path_str)
invalid proof of work or time too old
std::optional< uint256 > ReadSnapshotBaseBlockhash(fs::path chaindir)
size_t m_coinsdb_cache_size_bytes
The cache size of the on-disk coins view.
bool Condition(const CBlockIndex *pindex, const Consensus::Params ¶ms) const override
VerifyDBResult VerifyDB(Chainstate &chainstate, const Consensus::Params &consensus_params, CCoinsView &coinsview, int nCheckLevel, int nCheckDepth) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
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...
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
#define TRACE2(context, event, a, b)
std::vector< CTransactionRef > Package
A package is an ordered list of transactions.
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...
const CBlockIndex * LastCommonAncestor(const CBlockIndex *pa, const CBlockIndex *pb)
Find the last common ancestor two blocks have.
transaction was missing some of its inputs
static CSHA256 g_scriptExecutionCacheHasher
std::unique_ptr< CoinsViews > m_coins_views
Manages the UTXO set, which is a reflection of the contents of m_chain.
void LoadExternalBlockFile(CAutoFile &file_in, FlatFilePos *dbp=nullptr, std::multimap< uint256, FlatFilePos > *blocks_with_unknown_parent=nullptr)
Import blocks from an external file.
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
All parent headers found, difficulty matches, timestamp >= median previous, checkpoint.
bool MoneyRange(const CAmount &nValue)
int Height() const
Return the maximal height in the chain.
virtual void progress(const bilingual_str &title, int progress_percent, bool resume_possible)
size_t DynamicMemoryUsage() const
Calculate the size of the cache (in bytes)
std::unordered_map< uint256, CBlockIndex, BlockHasher > BlockMap
CTxOut out
unspent transaction output
const CChainParams & chainparams
bool DeploymentActiveAfter(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, [[maybe_unused]] VersionBitsCache &versionbitscache)
Determine if a deployment is active for the next block.
bool AcceptBlockHeader(const CBlockHeader &block, BlockValidationState &state, CBlockIndex **ppindex, bool min_pow_checked) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
If a block header hasn't already been seen, call CheckBlockHeader on it, ensure that it doesn't desce...
cache implements a cache with properties similar to a cuckoo-set.
stage after last reached validness failed
std::optional< bool > check_block_index
bool CheckFinalTxAtTip(const CBlockIndex &active_chain_tip, const CTransaction &tx)
static constexpr size_t MINIMUM_WITNESS_COMMITMENT
Minimum size of a witness commitment structure.
std::optional< std::string > EntriesAndTxidsDisjoint(const CTxMemPool::setEntries &ancestors, const std::set< uint256 > &direct_conflicts, const uint256 &txid)
Check the intersection between two sets of transactions (a set of mempool entries and a set of txids)...
unsigned int fCoinBase
whether containing transaction was a coinbase
The coins cache is in immediate need of a flush.
static const int COINBASE_MATURITY
Coinbase transaction outputs can only be spent after this number of new blocks (network rule) ...
bool HaveCoinInCache(const COutPoint &outpoint) const
Check if we have the given utxo already loaded in this cache.
Only first tx is coinbase, 2 <= coinbase input script length <= 100, transactions valid...
void ResizeCache(size_t new_cache_size) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Dynamically alter the underlying leveldb cache size.
An options struct for ChainstateManager, more ergonomically referred to as ChainstateManager::Options...
unsigned int nChainTx
Used to populate the nChainTx value, which is used during BlockManager::LoadBlockIndex().
bool IsInterrupted(const T &result)
uint256 BlockWitnessMerkleRoot(const CBlock &block, bool *mutated)
std::vector< CTxOut > m_spent_outputs
std::set< txiter, CompareIteratorByHash > setEntries
std::string FormatISO8601DateTime(int64_t nTime)
ISO 8601 formatting is preferred.
static constexpr unsigned int STANDARD_NOT_MANDATORY_VERIFY_FLAGS
For convenience, standard but not mandatory verify flags.
const TxValidationState m_state
Contains information about why the transaction failed.
static bool IsCurrentForFeeEstimation(Chainstate &active_chainstate) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
CBlockIndex * Genesis() const
Returns the index entry for the genesis block of this chain, or nullptr if none.
static fs::path GetSnapshotCoinsDBPath(Chainstate &cs) EXCLUSIVE_LOCKS_REQUIRED(
PerBlockConnectTrace()=default
#define TRACE7(context, event, a, b, c, d, e, f, g)
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...
Wrapper around a CAutoFile& that implements a ring buffer to deserialize from.
int ApplyTxInUndo(Coin &&undo, CCoinsViewCache &view, const COutPoint &out)
Restore the UTXO in a Coin at a given COutPoint.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system...
User-controlled performance and debug options.
bool ConnectTip(BlockValidationState &state, CBlockIndex *pindexNew, const std::shared_ptr< const CBlock > &pblock, ConnectTrace &connectTrace, DisconnectedBlockTransactions &disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Connect a new block to m_chain.
violated mempool's fee/size/descendant/RBF/etc limits
the block header may be on a too-little-work chain
void Clear() EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
inputs (covered by txid) failed policy rules
undo data available in rev*.dat
A hasher class for Bitcoin's 256-bit hash (double SHA-256).
ThresholdState
BIP 9 defines a finite-state-machine to deploy a softfork in multiple stages.
void MaybeUpdateMempoolForReorg(DisconnectedBlockTransactions &disconnectpool, bool fAddToMempool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Make mempool consistent after a reorg, by re-adding or recursively erasing disconnected block transac...
const ResultType m_result_type
Result type.
std::optional< fs::path > FindSnapshotChainstateDir(const fs::path &data_dir)
Return a path to the snapshot-based chainstate dir, if one exists.
bool ReadBlockFromDisk(CBlock &block, const FlatFilePos &pos) const
Functions for disk access for blocks.
void InvalidBlockFound(CBlockIndex *pindex, const BlockValidationState &state) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
RAII-style controller object for a CCheckQueue that guarantees the passed queue is finished before co...
void removeRecursive(const CTransaction &tx, MemPoolRemovalReason reason) EXCLUSIVE_LOCKS_REQUIRED(cs)
transaction spends a coinbase too early, or violates locktime/sequence locks
static constexpr int NO_WITNESS_COMMITMENT
Index marker for when no witness commitment is present in a coinbase transaction. ...
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 CheckPackage(const Package &txns, PackageValidationState &state)
Context-free package policy checks:
bool SignalsOptInRBF(const CTransaction &tx)
Check whether the sequence numbers on this transaction are signaling opt-in to replace-by-fee, according to BIP 125.
CChain m_chain
The current chain of blockheaders we consult and build on.
static bool CheckMerkleRoot(const CBlock &block, BlockValidationState &state)
int64_t nTime
UNIX timestamp of last known number of transactions.
static const int64_t MAX_BLOCK_SIGOPS_COST
The maximum allowed number of signature check operations in a block (network rule) ...
static SteadyClock::duration time_verify
bool DeploymentActiveAt(const CBlockIndex &index, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, [[maybe_unused]] VersionBitsCache &versionbitscache)
Determine if a deployment is active for this block.
void UnlinkPrunedFiles(const std::set< int > &setFilesToPrune) const
Actually unlink the specified files.
const char * what() const noexcept override
Non-refcounted RAII wrapper for FILE*.
static constexpr unsigned int STANDARD_SCRIPT_VERIFY_FLAGS
Standard script verification flags that standard transactions will comply with.
std::optional< fs::path > StoragePath()
void SetTip(CBlockIndex &block)
Set/initialize a chain with a given tip.
static void LimitMempoolSize(CTxMemPool &pool, CCoinsViewCache &coins_cache) EXCLUSIVE_LOCKS_REQUIRED(
unsigned int nChainTx
(memory only) Number of transactions in the chain up to and including this block. ...
bool HasCoinsViews() const
Does this chainstate have a UTXO set attached?
bool SpendCoin(const COutPoint &outpoint, Coin *moveto=nullptr)
Spend a coin.
const std::vector< CTxIn > vin
std::map< uint256, uint32_t > script_flag_exceptions
Hashes of blocks that.
const util::SignalInterrupt & m_interrupt
bool IsWitnessStandard(const CTransaction &tx, const CCoinsViewCache &mapInputs)
Check if the transaction is over standard P2WSH resources limit: 3600bytes witnessScript size...
void StartScriptCheckWorkerThreads(int threads_num)
Run instances of script checking worker threads.
void removeForReorg(CChain &chain, std::function< bool(txiter)> filter_final_and_mature) EXCLUSIVE_LOCKS_REQUIRED(cs
After reorg, filter the entries that would no longer be valid in the next block, and update the entri...
CVerifyDB(kernel::Notifications ¬ifications)
size_t GetSerializeSize(const T &t, int nVersion=0)
std::vector< CBlockIndex * > GetAllBlockIndices() EXCLUSIVE_LOCKS_REQUIRED(std::multimap< CBlockIndex *, CBlockIndex * > m_blocks_unlinked
All pairs A->B, where A (or one of its ancestors) misses transactions, but B has transactions.
void TransactionAddedToMempool(const CTransactionRef &, uint64_t mempool_sequence)
uint256 g_best_block
Used to notify getblocktemplate RPC of new tips.
arith_uint256 UintToArith256(const uint256 &a)
VersionBitsCache m_versionbitscache
Track versionbit status.
constexpr unsigned char * begin()
CoinsViews(DBParams db_params, CoinsViewOptions options)
This constructor initializes CCoinsViewDB and CCoinsViewErrorCatcher instances, but it does not creat...
bool IsUsable(const Chainstate *const cs) const EXCLUSIVE_LOCKS_REQUIRED(
Return true if a chainstate is considered usable.
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
static const unsigned int MAX_BLOCK_WEIGHT
The maximum allowed weight for a block, see BIP 141 (network rule)
int nSubsidyHalvingInterval
bool ProcessNewBlock(const std::shared_ptr< const CBlock > &block, bool force_processing, bool min_pow_checked, bool *new_block) LOCKS_EXCLUDED(cs_main)
Process an incoming block.
std::optional< std::string > GetEntriesForConflicts(const CTransaction &tx, CTxMemPool &pool, const CTxMemPool::setEntries &iters_conflicting, CTxMemPool::setEntries &all_conflicts)
Get all descendants of iters_conflicting.
int64_t nTxCount
total number of transactions between genesis and that timestamp
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate) EXCLUSIVE_LOCKS_REQUIRED(std::optional< uint256 > ReadSnapshotBaseBlockhash(fs::path chaindir) EXCLUSIVE_LOCKS_REQUIRED(constexpr std::string_view SNAPSHOT_CHAINSTATE_SUFFIX
Write out the blockhash of the snapshot base block that was used to construct this chainstate...
int64_t CAmount
Amount in satoshis (Can be negative)
void SetBestBlock(const uint256 &hashBlock)
bool WriteUndoDataForBlock(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos SaveBlockToDisk(const CBlock &block, int nHeight, const FlatFilePos *dbp)
Store block on disk.
void PruneAndFlush()
Prune blockfiles from the disk if necessary and then flush chainstate changes if we pruned...
MempoolAcceptResult AcceptToMemoryPool(Chainstate &active_chainstate, const CTransactionRef &tx, int64_t accept_time, bool bypass_limits, bool test_accept) EXCLUSIVE_LOCKS_REQUIRED(
Try to add a transaction to the mempool.
void UpdateTip(const CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(SteadyClock::time_poin m_last_write)
Check warning conditions and do some notifications on new chain tip set.
std::array< uint8_t, 4 > MessageStartChars
uint256 GetBlockHash() const
bool AreInputsStandard(const CTransaction &tx, const CCoinsViewCache &mapInputs)
Check transaction inputs to mitigate two potential denial-of-service attacks:
static CuckooCache::cache< uint256, SignatureCacheHasher > g_scriptExecutionCache
void FindFilesToPrune(std::set< int > &setFilesToPrune, int last_prune, const Chainstate &chain, ChainstateManager &chainman)
Prune block and undo files (blk???.dat and rev???.dat) so that the disk space used is less than a use...
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.
uint32_t nHeight
at which height this containing transaction was included in the active block chain ...
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, const Chainstate &chain, ChainstateManager &chainman)
const fs::path SNAPSHOT_BLOCKHASH_FILENAME
The file in the snapshot chainstate dir which stores the base blockhash.
std::list< CTransactionRef > take()
Clear all data structures and return the list of transactions.
unsigned int GetCacheSize() const
Calculate the size of the cache (in number of transaction outputs)
CBlockLocator GetLocator() const
Return a CBlockLocator that refers to the tip in of this chain.
static const uint32_t MEMPOOL_HEIGHT
Fake height value used in Coin to signify they are only in the memory pool (since 0...
std::function< FILE *(const fs::path &, const char *)> FopenFn
std::string ToString(const T &t)
Locale-independent version of std::to_string.
bool IsBlockMutated(const CBlock &block, bool check_witness_root)
Check if a block has been mutated (with respect to its merkle root and witness commitments).
AssumeutxoHash hash_serialized
The expected hash of the deserialized UTXO set.
bool IsBIP30Repeat(const CBlockIndex &block_index)
Identifies blocks that overwrote an existing coinbase output in the UTXO set (see BIP30) ...
void SetfLargeWorkInvalidChainFound(bool flag)
ChainstateManager & m_chainman
The chainstate manager that owns this chainstate.
Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends...
bool DisconnectTip(BlockValidationState &state, DisconnectedBlockTransactions *disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Disconnect m_chain's tip.
bool WriteBlockIndexDB() EXCLUSIVE_LOCKS_REQUIRED(bool LoadBlockIndexDB(const std::optional< uint256 > &snapshot_blockhash) EXCLUSIVE_LOCKS_REQUIRED(void ScanAndUnlinkAlreadyPrunedFiles() EXCLUSIVE_LOCKS_REQUIRED(CBlockIndex * AddToBlockIndex(const CBlockHeader &block, CBlockIndex *&best_header) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Remove any pruned block & undo files that are still on disk.
static void SnapshotUTXOHashBreakpoint(const util::SignalInterrupt &interrupt)
An options struct for BlockManager, more ergonomically referred to as BlockManager::Options due to th...
bool HasValidProofOfWork(const std::vector< CBlockHeader > &headers, const Consensus::Params &consensusParams)
Check with the proof of work on each blockheader matches the value in nBits.
static SteadyClock::duration time_check
void Init(const T &tx, std::vector< CTxOut > &&spent_outputs, bool force=false)
Initialize this PrecomputedTransactionData with transaction data.
virtual void headerTip(SynchronizationState state, int64_t height, int64_t timestamp, bool presync)
bool BackgroundSyncInProgress() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
The state of a background sync (for net processing)
bool CheckInputScripts(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, unsigned int flags, bool cacheSigStore, bool cacheFullScriptStore, PrecomputedTransactionData &txdata, std::vector< CScriptCheck > *pvChecks=nullptr) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Check whether all of this transaction's input scripts succeed.
void EmplaceCoinInternalDANGER(COutPoint &&outpoint, Coin &&coin)
Emplace a coin into cacheCoins without performing any checks, marking the emplaced coin as dirty...
bool IsSnapshotActive() const
static std::string PathToString(const path &path)
Convert path object to a byte string.
Chainstate stores and provides an API to update our local knowledge of the current best chain...
bool Invalid(Result result, const std::string &reject_reason="", const std::string &debug_message="")
Scripts & signatures ok. Implies all parents are either at least VALID_SCRIPTS, or are ASSUMED_VALID...
std::optional< LockPoints > CalculateLockPointsAtTip(CBlockIndex *tip, const CCoinsView &coins_view, const CTransaction &tx)
Transaction might have a witness prior to SegWit activation, or witness may have been malleated (whic...
int64_t GetTransactionSigOpCost(const CTransaction &tx, const CCoinsViewCache &inputs, uint32_t flags)
Compute total signature operation cost of a transaction.
Abstract view on the open txout dataset.
ChainstateRole
This enum describes the various roles a specific Chainstate instance can take.
this block was cached as being invalid and we didn't store the reason why
Validation result for package mempool acceptance.
static SteadyClock::duration time_undo
std::atomic< bool > m_cached_finished_ibd
Whether initial block download has ended and IsInitialBlockDownload should return false from now on...
An input of a transaction.
bool ActivateBestChain(BlockValidationState &state, std::shared_ptr< const CBlock > pblock=nullptr) LOCKS_EXCLUDED(DisconnectResult DisconnectBlock(const CBlock &block, const CBlockIndex *pindex, CCoinsViewCache &view) EXCLUSIVE_LOCKS_REQUIRED(boo ConnectBlock)(const CBlock &block, BlockValidationState &state, CBlockIndex *pindex, CCoinsViewCache &view, bool fJustCheck=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Find the best known block, and make it the tip of the block chain.
std::vector< unsigned char > GenerateCoinbaseCommitment(CBlock &block, const CBlockIndex *pindexPrev) const
Produce the necessary coinbase commitment for a block (modifies the hash, don't call for mined blocks...
static SteadyClock::duration time_connect_total
static constexpr int PRUNE_LOCK_BUFFER
The number of blocks to keep below the deepest prune lock.
const uint256 & GetWitnessHash() const
int64_t descendant_count
The maximum allowed number of transactions in a package including the entry and its descendants...
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...
Removed for reorganization.
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).
unsigned int GetNextWorkRequired(const CBlockIndex *pindexLast, const CBlockHeader *pblock, const Consensus::Params ¶ms)
const uint256 & GetHash() const
std::string ToString() const
bilingual_str _(const char *psz)
Translation function.
static CCheckQueue< CScriptCheck > scriptcheckqueue(128)
the block failed to meet one of our checkpoints
static bool DeleteCoinsDBFromDisk(const fs::path db_path, bool is_snapshot) EXCLUSIVE_LOCKS_REQUIRED(
const std::function< NodeClock::time_point()> adjusted_time_callback
bool LoadBlockIndex() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Load the block tree and coins database from disk, initializing state if we're running with -reindex...
Chainstate(CTxMemPool *mempool, node::BlockManager &blockman, ChainstateManager &chainman, std::optional< uint256 > from_snapshot_blockhash=std::nullopt)
bool Contains(const CBlockIndex *pindex) const
Efficiently check whether a block is present in this chain.
static feebumper::Result CheckFeeRate(const CWallet &wallet, const CMutableTransaction &mtx, const CFeeRate &newFeerate, const int64_t maxTxSize, CAmount old_fee, std::vector< bilingual_str > &errors)
Check if the user provided a valid feeRate.
kernel::Notifications & GetNotifications() const
const kernel::BlockManagerOpts m_opts
uint256 uint256S(const char *str)
int64_t nPowTargetSpacing
std::string ScriptErrorString(const ScriptError serror)
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...
Abstract class that implements BIP9-style threshold logic, and caches results.
bool NeedsRedownload() const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Whether the chain state needs to be redownloaded due to lack of witness data.
ChainstateManager(const util::SignalInterrupt &interrupt, Options options, node::BlockManager::Options blockman_options)
Holds various statistics on transactions within a chain.
void Finalize(unsigned char hash[OUTPUT_SIZE])
bool LoadingBlocks() const
const std::vector< CTxOut > vout
void removeForBlock(const std::vector< CTransactionRef > &vtx)
Remove any entries that are in this block.
bool signet_blocks
If true, witness commitments contain a payload equal to a Bitcoin Script solution to the signet chall...
for(const CTxIn &txin :tx.vin)
bool UndoReadFromDisk(CBlockUndo &blockundo, const CBlockIndex &index) const
bool TestLockPointValidity(CChain &active_chain, const LockPoints &lp)
Test whether the LockPoints height and time are still valid on the current chain. ...
const Consensus::Params & GetConsensus() const
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept)
Validate (and maybe submit) a package to the mempool.
std::shared_ptr< const CBlock > pblock
arith_uint256 nLastPreciousChainwork
chainwork for the last block that preciousblock has been applied to.
bool ActivateBestChainStep(BlockValidationState &state, CBlockIndex *pindexMostWork, const std::shared_ptr< const CBlock > &pblock, bool &fInvalidFound, ConnectTrace &connectTrace) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Try to make some progress towards making pindexMostWork the active block.
bool EvaluateSequenceLocks(const CBlockIndex &block, std::pair< int, int64_t > lockPair)
static void LimitValidationInterfaceQueue() LOCKS_EXCLUDED(cs_main)
CMainSignals & GetMainSignals()
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
CCoinsViewDB & CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(
#define LogPrintLevel(category, level,...)
static MempoolAcceptResult Success(std::list< CTransactionRef > &&replaced_txns, int64_t vsize, CAmount fees, CFeeRate effective_feerate, const std::vector< uint256 > &wtxids_fee_calculations)
virtual std::vector< uint256 > GetHeadBlocks() const
Retrieve the range of blocks that may have been only partially written.
bool InitScriptExecutionCache(size_t max_size_bytes)
Initializes the script-execution cache.
uint256 BlockMerkleRoot(const CBlock &block, bool *mutated)
static ChainstateManager::Options && Flatten(ChainstateManager::Options &&opts)
Apply default chain params to nullopt members.
void ForceFlushStateToDisk()
Unconditionally flush all changes to disk.
void BlockConnected(ChainstateRole, const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
An output of a transaction.
std::string ToString() const
bool LoadChainTip() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Update the chain tip based on database information, i.e.
bool CheckBlock(const CBlock &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW, bool fCheckMerkleRoot)
Functions for validating blocks and updating the block tree.
std::vector< uint256 > vHave
At least one tx is invalid.
void StopScriptCheckWorkerThreads()
Stop all of the script checking worker threads.
uint32_t nMinerConfirmationWindow
CBlockIndex * FindMostWorkChain() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Return the tip of the chain with the most work in it, that isn't known to be invalid (it's however fa...
Queue for verifications that have to be performed.
Parameters that influence chain consensus.
bool CheckProofOfWork(uint256 hash, unsigned int nBits, const Consensus::Params ¶ms)
Check whether a block hash satisfies the proof-of-work requirement specified by nBits.
An outpoint - a combination of a transaction hash and an index n into its vout.
CBlockIndex * LookupBlockIndex(const uint256 &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
A base class defining functions for notifying about certain kernel events.
void TryAddBlockIndexCandidate(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
void AddTransactionsUpdated(unsigned int n)
std::chrono::seconds max_tip_age
If the tip is older than this, the node is considered to be in initial block download.
constexpr bool IsNull() const
std::pair< int, int64_t > CalculateSequenceLocks(const CTransaction &tx, int flags, std::vector< int > &prevHeights, const CBlockIndex &block)
Calculates the block height and previous block's median time past at which the transaction will be co...
DisconnectedBlockTransactions.
Chainstate &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(std::vector< Chainstate * GetAll)()
Instantiate a new chainstate.
Validation result for a single transaction mempool acceptance.
bool CheckDiskSpace(const fs::path &dir, uint64_t additional_bytes)
static MempoolAcceptResult MempoolTx(int64_t vsize, CAmount fees)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define Assume(val)
Assume is the identity function.
256-bit unsigned big integer.
int64_t GetMedianTimePast() const
int64_t ancestor_size_vbytes
The maximum allowed size in virtual bytes of an entry and its ancestors within a package.
block data in blk*.dat was received with a witness-enforcing client
void AddCoin(const COutPoint &outpoint, Coin &&coin, bool possible_overwrite)
Add a coin.
#define TRACE5(context, event, a, b, c, d, e)
void BlockConnected(CBlockIndex *pindex, std::shared_ptr< const CBlock > pblock)
bool CheckBlockDataAvailability(const CBlockIndex &upper_block LIFETIMEBOUND, const CBlockIndex &lower_block LIFETIMEBOUND) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetFirstStoredBlock(const CBlockIndex &start_block LIFETIMEBOUND, const CBlockIndex *lower_block=nullptr) EXCLUSIVE_LOCKS_REQUIRED(boo m_have_pruned)
Check if all blocks in the [upper_block, lower_block] range have data available.
int32_t ComputeBlockVersion(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Determine what nVersion a new block should use.
int64_t m_total_coinstip_cache
The total number of bytes available for us to use across all in-memory coins caches.
FlatFilePos GetUndoPos() const EXCLUSIVE_LOCKS_REQUIRED(
constexpr int64_t count_seconds(std::chrono::seconds t)
Closure representing one script verification Note that this stores references to the spending transac...
static constexpr unsigned int MAX_STANDARD_TX_SIGOPS_COST
The maximum number of sigops we're willing to relay/mine in a single tx.
static CTransactionRef MakeTransactionRef(Tx &&txIn)
static bool CheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW=true)
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate)
bool IsBIP30Unspendable(const CBlockIndex &block_index)
Identifies blocks which coinbase output was subsequently overwritten in the UTXO set (see BIP30) ...
const CBlockIndex *SnapshotBase() EXCLUSIVE_LOCKS_REQUIRED(std::set< CBlockIndex *, node::CBlockIndexWorkComparator > setBlockIndexCandidates
The base of the snapshot this chainstate was created from.
uint256 GetBestBlock() const override
Retrieve the block hash whose state this CCoinsView currently represents.
Transaction is missing a witness.
Helper class that manages an interrupt flag, and allows a thread or signal to interrupt another threa...
bool IsValid(enum BlockStatus nUpTo=BLOCK_VALID_TRANSACTIONS) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
void CheckBlockIndex()
Make various assertions about the state of the block index.
int32_t nVersion
block header
static bool ComputeUTXOStats(CCoinsView *view, CCoinsStats &stats, T hash_obj, const std::function< void()> &interruption_point)
Calculate statistics about the unspent transaction output set.
const CChainParams & GetParams() const
static SteadyClock::duration time_flush
std::string FormatMoney(const CAmount n)
Money parsing/formatting utilities.
std::string get_filesystem_error_message(const fs::filesystem_error &e)
invalid by consensus rules (excluding any below reasons)
const CTransaction * ptxTo
const arith_uint256 & MinimumChainWork() const
#define EXCLUSIVE_LOCKS_REQUIRED(...)
std::vector< CTransactionRef > vtx
const ChainTxData & TxData() const
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'...
kernel::Notifications & m_notifications
static bool ContextualCheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, BlockManager &blockman, const ChainstateManager &chainman, const CBlockIndex *pindexPrev, NodeClock::time_point now) EXCLUSIVE_LOCKS_REQUIRED(
Context-dependent validity checks.
the block's data didn't match the data committed to by the PoW
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
static SteadyClock::duration time_connect
constexpr const unsigned char * data() const
#define LOCKS_EXCLUDED(...)
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
bool ActivateSnapshot(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Construct and activate a Chainstate on the basis of UTXO snapshot data.
unsigned int GetLegacySigOpCount(const CTransaction &tx)
Auxiliary functions for transaction validation (ideally should not be exposed)
virtual void warning(const bilingual_str &warning)
bool error(const char *fmt, const Args &... args)
The block chain is a tree shaped structure starting with the genesis block at the root...
Undo information for a CBlock.
Serialized script, used inside transaction inputs and outputs.
int64_t descendant_size_vbytes
The maximum allowed size in virtual bytes of an entry and its descendants within a package...
Undo information for a CTransaction.
std::vector< PerBlockConnectTrace > blocksConnected
CCoinsView backed by the coin database (chainstate/)
static constexpr unsigned int MIN_STANDARD_TX_NONWITNESS_SIZE
The minimum non-witness size for transactions we're willing to relay/mine: one larger than 64...
const ChainstateManager & m_chainman
void PruneBlockIndexCandidates()
Delete all entries in setBlockIndexCandidates that are worse than the current tip.
static const int PROTOCOL_VERSION
network protocol versioning
const fs::path blocks_dir
static const int32_t VERSIONBITS_TOP_BITS
What bits to set in version for versionbits blocks.
bool m_spent_outputs_ready
Whether m_spent_outputs is initialized.
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...
void Uncache(const COutPoint &outpoint)
Removes the UTXO with the given outpoint from the cache, if it is not modified.
A block this one builds on is invalid.
std::string ToString() const
#define TRACE6(context, event, a, b, c, d, e, f)
bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network) ...
static void AppendWarning(bilingual_str &res, const bilingual_str &warn)
Private helper function that concatenates warning messages.
std::vector< CTransactionRef > AddTransactionsFromBlock(const std::vector< CTransactionRef > &vtx)
Add transactions from the block, iterating through vtx in reverse order.
void ChainStateFlushed(ChainstateRole, const CBlockLocator &)
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
static constexpr std::chrono::hours DATABASE_WRITE_INTERVAL
Time to wait between writing blocks/block index to disk.
int ActiveHeight() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
std::string GetHex() const
bool RaiseValidity(enum BlockStatus nUpTo) EXCLUSIVE_LOCKS_REQUIRED(
Raise the validity level of this block index entry.
Application-specific storage settings.
Fee rate in satoshis per kilovirtualbyte: CAmount / kvB.
bool IsAssumedValid() const EXCLUSIVE_LOCKS_REQUIRED(
static bool NotifyHeaderTip(ChainstateManager &chainman) LOCKS_EXCLUDED(cs_main)
bool m_checked_witness_commitment
Holds configuration for use during UTXO snapshot load and validation.
#define AssertLockNotHeld(cs)
bilingual_str ErrorString(const Result< T > &result)
bool AcceptBlock(const std::shared_ptr< const CBlock > &pblock, BlockValidationState &state, CBlockIndex **ppindex, bool fRequested, const FlatFilePos *dbp, bool *fNewBlock, bool min_pow_checked) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Sufficiently validate a block for disk storage (and store on disk).
arith_uint256 CalculateHeadersWork(const std::vector< CBlockHeader > &headers)
Return the sum of the work on a given set of headers.
int MinBIP9WarningHeight
Don't warn about unknown BIP 9 activations below this height.
bool FlushChainstateBlockFile(int tip_height)
this node does not have a mempool so can't validate the transaction
A mutable version of CTransaction.
uint32_t nRuleChangeActivationThreshold
Minimum blocks including miner confirmation of the total of 2016 blocks in a retargeting period...
block timestamp was > 2 hours in the future (or our clock is bad)
size_t m_coinstip_cache_size_bytes
The cache size of the in-memory coins view.
bool IsFinalTx(const CTransaction &tx, int nBlockHeight, int64_t nBlockTime)
Check if transaction is final and can be included in a block with the specified height and time...
bool 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 const unsigned int MAX_BLOCK_SERIALIZED_SIZE
The maximum allowed size for a serialized block, in bytes (only for buffer size limits) ...
static SteadyClock::duration time_post_connect
Mutex m_chainstate_mutex
The ChainState Mutex A lock that must be held when modifying this ChainState - held in ActivateBestCh...
arith_uint256 GetBlockProof(const CBlockIndex &block)
static MempoolAcceptResult Failure(TxValidationState state)
static bool CheckInputsFromMempoolAndCache(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &view, const CTxMemPool &pool, unsigned int flags, PrecomputedTransactionData &txdata, CCoinsViewCache &coins_tip) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Checks to avoid mempool polluting consensus critical paths since cached signature and script validity...
static constexpr int64_t MAX_FUTURE_BLOCK_TIME
Maximum amount of time that a block timestamp is allowed to exceed the current network-adjusted time ...
The basic transaction that is broadcasted on the network and contained in blocks. ...
Different type to mark Mutex at global scope.
void CheckForkWarningConditions() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
int nHeight
height of the entry in the chain. The genesis block has height 0
bool TestBlockValidity(BlockValidationState &state, const CChainParams &chainparams, Chainstate &chainstate, const CBlock &block, CBlockIndex *pindexPrev, const std::function< NodeClock::time_point()> &adjusted_time_callback, bool fCheckPOW, bool fCheckMerkleRoot)
Check a block is completely valid from start to finish (only works on top of our current best block) ...
WarningBitsConditionChecker(const ChainstateManager &chainman, int bit)
std::optional< uint256 > SnapshotBlockhash() const
const Consensus::Params & GetConsensus() const
int64_t ancestor_count
The maximum allowed number of transactions in a package including the entry and its ancestors...
bool FatalError(Notifications ¬ifications, BlockValidationState &state, const std::string &strMessage, const bilingual_str &userMessage)
int64_t m_total_coinsdb_cache
The total number of bytes available for us to use across all leveldb coins databases.
CCoinsView that adds a memory cache for transactions to another CCoinsView.
std::vector< PerBlockConnectTrace > & GetBlocksConnected()
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
std::string GetRejectReason() const
full block available in blk*.dat
#define LOG_TIME_MILLIS_WITH_CATEGORY(end_msg, log_category)
static bool exists(const path &p)
int GetWitnessCommitmentIndex(const CBlock &block)
Compute at which vout of the block's coinbase transaction the witness commitment occurs, or -1 if not found.
A hasher class for SHA-256.
int64_t GetTime()
DEPRECATED, see GetTime.
std::vector< CTxUndo > vtxundo
static SynchronizationState GetSynchronizationState(bool init)
static int64_t GetBlockWeight(const CBlock &block)
std::optional< AssumeutxoData > AssumeutxoForHeight(int height) const
static constexpr unsigned int STANDARD_LOCKTIME_VERIFY_FLAGS
Used as the flags parameter to sequence and nLocktime checks in non-consensus code.
static const int32_t VERSIONBITS_NUM_BITS
Total bits available for versionbits.
Path class wrapper to block calls to the fs::path(std::string) implicit constructor and the fs::path:...
bool CheckTransaction(const CTransaction &tx, TxValidationState &state)
If ASSUMED_VALID is set, it means that this block has not been validated and has validity status less...
bool PreciousBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(bool InvalidateBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(voi ResetBlockFailureFlags)(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as precious and reorganize.
ScriptError GetScriptError() const
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate...
const CBlockIndex * FindFork(const CBlockIndex *pindex) const
Find the last common block between this chain and a block index entry.
CCoinsView that brings transactions from a mempool into view.
Tx already in mempool or conflicts with a tx in the chain (if it conflicts with another tx in mempool...
void ReportHeadersPresync(const arith_uint256 &work, int64_t height, int64_t timestamp)
This is used by net_processing to report pre-synchronization progress of headers, as headers are not ...
static SteadyClock::duration time_total
void removeForBlock(const std::vector< CTransactionRef > &vtx, unsigned int nBlockHeight) EXCLUSIVE_LOCKS_REQUIRED(cs)
Called when a block is connected.
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void BlockChecked(const CBlock &, const BlockValidationState &)
CHash256 & Write(Span< const unsigned char > input)
otherwise didn't meet our local policy rules
int32_t nBlockReverseSequenceId
Decreasing counter (used by subsequent preciousblock calls).
#define LOG_TIME_MILLIS_WITH_CATEGORY_MSG_ONCE(end_msg, log_category)
unsigned int nTx
Number of transactions in this block.
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...
#define PACKAGE_BUGREPORT
static constexpr unsigned int EXTRA_DESCENDANT_TX_SIZE_LIMIT
An extra transaction can be added to a package, as long as it only has one ancestor and is no larger ...
void Add(std::vector< T > &&vChecks)
static GenTxid Txid(const uint256 &hash)
static MempoolAcceptResult MempoolTxDifferentWitness(const uint256 &other_wtxid)
CTxMemPool * GetMempool()
bool IsSnapshotValidated() const EXCLUSIVE_LOCKS_REQUIRED(
Is there a snapshot in use and has it been fully validated?
static constexpr std::chrono::hours DATABASE_FLUSH_INTERVAL
Time to wait between flushing chainstate to disk.
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it...
virtual InterruptResult blockTip(SynchronizationState state, CBlockIndex &index)
void UpdateCoins(const CTransaction &tx, CCoinsViewCache &inputs, CTxUndo &txundo, int nHeight)
#define Assert(val)
Identity function.
static int64_t num_blocks_total
std::shared_ptr< Chain::Notifications > m_notifications
bool m_checked_merkle_root
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...
static const unsigned int MAX_DISCONNECTED_TX_POOL_SIZE
Maximum kilobytes for transactions to store for processing during reorg.
GlobalMutex g_best_block_mutex
bool CheckSignetBlockSolution(const CBlock &block, const Consensus::Params &consensusParams)
Extract signature and check whether a block has a valid solution.
Used to track blocks whose transactions were applied to the UTXO state as a part of a single Activate...
bool ProcessNewBlockHeaders(const std::vector< CBlockHeader > &block, bool min_pow_checked, BlockValidationState &state, const CBlockIndex **ppindex=nullptr) LOCKS_EXCLUDED(cs_main)
Process incoming block headers.
bool HaveCoin(const COutPoint &outpoint) const override
Just check whether a given outpoint is unspent.
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.
static constexpr CAmount COIN
The amount of satoshis in one BTC.
const uint256 * phashBlock
pointer to the hash of the block, if any. Memory is owned by this CBlockIndex
PrecomputedTransactionData * txdata
Threshold condition checker that triggers when unknown versionbits are seen on the network...