96 "level 0 reads the blocks from disk",
97 "level 1 verifies block validity",
98 "level 2 verifies undo data",
99 "level 3 checks disconnection of tip blocks",
100 "level 4 tries to reconnect the blocks",
101 "each level includes the checks of the previous levels",
138 std::vector<CScriptCheck>* pvChecks =
nullptr)
151 const int nBlockHeight = active_chain_tip.nHeight + 1;
158 const int64_t nBlockTime{active_chain_tip.GetMedianTimePast()};
160 return IsFinalTx(tx, nBlockHeight, nBlockTime);
174 std::optional<std::vector<int>> CalculatePrevHeights(
179 std::vector<int> prev_heights;
180 prev_heights.resize(tx.
vin.size());
181 for (
size_t i = 0; i < tx.
vin.size(); ++i) {
190 prev_heights[i] = tip.
nHeight + 1;
192 prev_heights[i] = coin.
nHeight;
206 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
207 if (!prev_heights.has_value())
return std::nullopt;
210 next_tip.
pprev = tip;
228 int max_input_height{0};
229 for (
const int height : prev_heights.value()) {
231 if (height != next_tip.
nHeight) {
232 max_input_height = std::max(max_input_height, height);
270 int expired = pool.Expire(GetTime<std::chrono::seconds>() - pool.m_opts.expiry);
275 std::vector<COutPoint> vNoSpendsRemaining;
276 pool.TrimToSize(pool.m_opts.max_size_bytes, &vNoSpendsRemaining);
277 for (
const COutPoint& removed : vNoSpendsRemaining)
278 coins_cache.Uncache(removed);
284 if (active_chainstate.m_chainman.IsInitialBlockDownload()) {
289 if (active_chainstate.m_chain.Height() < active_chainstate.m_chainman.m_best_header->nHeight - 1) {
303 std::vector<uint256> vHashUpdate;
310 const auto queuedTx = disconnectpool.
take();
311 auto it = queuedTx.rbegin();
312 while (it != queuedTx.rend()) {
314 if (!fAddToMempool || (*it)->IsCoinBase() ||
322 vHashUpdate.push_back((*it)->GetHash());
363 it->UpdateLockPoints(*new_lock_points);
370 if (it->GetSpendsCoinbase()) {
412 if (coin.
IsSpent())
return false;
424 const Coin& coinFromUTXOSet = coins_tip.AccessCoin(txin.
prevout);
442 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
443 m_active_chainstate(active_chainstate)
451 const int64_t m_accept_time;
452 const bool m_bypass_limits;
460 std::vector<COutPoint>& m_coins_to_uncache;
462 const bool m_test_accept;
466 const bool m_allow_replacement;
468 const bool m_allow_sibling_eviction;
473 const bool m_package_submission;
477 const bool m_package_feerates;
482 const std::optional<CFeeRate> m_client_maxfeerate;
485 const bool m_allow_carveouts;
488 static ATMPArgs SingleAccept(
const CChainParams& chainparams, int64_t accept_time,
489 bool bypass_limits, std::vector<COutPoint>& coins_to_uncache,
491 return ATMPArgs{ chainparams,
506 static ATMPArgs PackageTestAccept(
const CChainParams& chainparams, int64_t accept_time,
507 std::vector<COutPoint>& coins_to_uncache) {
508 return ATMPArgs{ chainparams,
523 static ATMPArgs PackageChildWithParents(
const CChainParams& chainparams, int64_t accept_time,
524 std::vector<COutPoint>& coins_to_uncache,
const std::optional<CFeeRate>& client_maxfeerate) {
525 return ATMPArgs{ chainparams,
540 static ATMPArgs SingleInPackageAccept(
const ATMPArgs& package_args) {
541 return ATMPArgs{ package_args.m_chainparams,
542 package_args.m_accept_time,
544 package_args.m_coins_to_uncache,
545 package_args.m_test_accept,
550 package_args.m_client_maxfeerate,
561 std::vector<COutPoint>& coins_to_uncache,
563 bool allow_replacement,
564 bool allow_sibling_eviction,
565 bool package_submission,
566 bool package_feerates,
567 std::optional<CFeeRate> client_maxfeerate,
568 bool allow_carveouts)
569 : m_chainparams{chainparams},
570 m_accept_time{accept_time},
571 m_bypass_limits{bypass_limits},
572 m_coins_to_uncache{coins_to_uncache},
573 m_test_accept{test_accept},
574 m_allow_replacement{allow_replacement},
575 m_allow_sibling_eviction{allow_sibling_eviction},
576 m_package_submission{package_submission},
577 m_package_feerates{package_feerates},
578 m_client_maxfeerate{client_maxfeerate},
579 m_allow_carveouts{allow_carveouts}
583 if (m_package_feerates) {
584 Assume(m_package_submission);
585 Assume(!m_allow_carveouts);
586 Assume(!m_allow_sibling_eviction);
588 if (m_allow_sibling_eviction)
Assume(m_allow_replacement);
628 explicit Workspace(
const CTransactionRef& ptx) : m_ptx(ptx), m_hash(ptx->GetHash()) {}
631 std::set<Txid> m_conflicts;
639 std::unique_ptr<CTxMemPoolEntry> m_entry;
642 bool m_sibling_eviction{
false};
677 bool PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
678 std::vector<Workspace>& workspaces,
701 std::map<uint256, MempoolAcceptResult>& results)
709 CAmount mempoolRejectFee = m_pool.GetMinFee().GetFee(package_size);
710 if (mempoolRejectFee > 0 && package_fee < mempoolRejectFee) {
714 if (package_fee < m_pool.m_opts.min_relay_feerate.GetFee(package_size)) {
716 strprintf(
"%d < %d", package_fee, m_pool.m_opts.min_relay_feerate.GetFee(package_size)));
723 return m_active_chainstate.m_chainman.m_validation_cache;
736 struct SubPackageState {
738 CAmount m_total_modified_fees{0};
740 int64_t m_total_vsize{0};
749 std::list<CTransactionRef> m_replaced_transactions;
754 size_t m_conflicting_size{0};
757 struct SubPackageState m_subpackage;
762 m_subpackage = SubPackageState{};
765 CleanupTemporaryCoins();
769 bool MemPoolAccept::PreChecks(ATMPArgs&
args, Workspace& ws)
775 const Txid& hash = ws.m_hash;
778 const int64_t nAcceptTime =
args.m_accept_time;
779 const bool bypass_limits =
args.m_bypass_limits;
780 std::vector<COutPoint>& coins_to_uncache =
args.m_coins_to_uncache;
784 std::unique_ptr<CTxMemPoolEntry>& entry = ws.m_entry;
796 if (m_pool.m_opts.require_standard && !
IsStandardTx(tx, m_pool.m_opts.max_datacarrier_bytes, m_pool.m_opts.permit_bare_multisig, m_pool.m_opts.dust_relay_feerate, reason)) {
824 if (ptxConflicting) {
825 if (!
args.m_allow_replacement) {
829 if (!ws.m_conflicts.count(ptxConflicting->
GetHash()))
848 ws.m_conflicts.insert(ptxConflicting->
GetHash());
853 m_view.SetBackend(m_viewmempool);
859 coins_to_uncache.push_back(txin.
prevout);
865 if (!m_view.HaveCoin(txin.
prevout)) {
880 m_view.GetBestBlock();
885 m_view.SetBackend(m_dummy);
887 assert(m_active_chainstate.m_blockman.LookupBlockIndex(m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
894 const std::optional<LockPoints> lock_points{
CalculateLockPointsAtTip(m_active_chainstate.m_chain.Tip(), m_view, tx)};
916 ws.m_modified_fees = ws.m_base_fees;
917 m_pool.ApplyDelta(hash, ws.m_modified_fees);
921 bool fSpendsCoinbase =
false;
923 const Coin &coin = m_view.AccessCoin(txin.
prevout);
925 fSpendsCoinbase =
true;
932 const uint64_t entry_sequence = bypass_limits ? 0 : m_pool.GetSequence();
933 entry.reset(
new CTxMemPoolEntry(ptx, ws.m_base_fees, nAcceptTime, m_active_chainstate.m_chain.Height(), entry_sequence,
934 fSpendsCoinbase, nSigOpsCost, lock_points.value()));
935 ws.m_vsize = entry->GetTxSize();
947 if (!bypass_limits && ws.m_ptx->version !=
TRUC_VERSION && ws.m_modified_fees < m_pool.m_opts.min_relay_feerate.GetFee(ws.m_vsize)) {
951 strprintf(
"%d < %d", ws.m_modified_fees, m_pool.m_opts.min_relay_feerate.GetFee(ws.m_vsize)));
956 if (!bypass_limits && !
args.m_package_feerates && !
CheckFeeRate(ws.m_vsize, ws.m_modified_fees, state))
return false;
958 ws.m_iters_conflicting = m_pool.GetIterSet(ws.m_conflicts);
965 if (ws.m_conflicts.size() == 1 &&
args.m_allow_carveouts) {
993 assert(ws.m_iters_conflicting.size() == 1);
1000 if (
auto ancestors{m_pool.CalculateMemPoolAncestors(*entry, maybe_rbf_limits)}) {
1001 ws.m_ancestors = std::move(*ancestors);
1007 if (!
args.m_allow_carveouts) {
1032 if (
auto ancestors_retry{m_pool.CalculateMemPoolAncestors(*entry, cpfp_carve_out_limits)}) {
1033 ws.m_ancestors = std::move(*ancestors_retry);
1042 if (
const auto err{
SingleTRUCChecks(ws.m_ptx, ws.m_ancestors, ws.m_conflicts, ws.m_vsize)}) {
1044 if (
args.m_allow_sibling_eviction && err->second !=
nullptr) {
1050 ws.m_conflicts.insert(err->second->GetHash());
1054 ws.m_iters_conflicting.insert(m_pool.GetIter(err->second->GetHash()).value());
1055 ws.m_sibling_eviction =
true;
1076 m_subpackage.m_rbf |= !ws.m_conflicts.empty();
1080 bool MemPoolAccept::ReplacementChecks(Workspace& ws)
1086 const uint256& hash = ws.m_hash;
1089 CFeeRate newFeeRate(ws.m_modified_fees, ws.m_vsize);
1103 strprintf(
"insufficient fee%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1107 if (
const auto err_string{
GetEntriesForConflicts(tx, m_pool, ws.m_iters_conflicting, m_subpackage.m_all_conflicts)}) {
1109 strprintf(
"too many potential replacements%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1112 if (
const auto err_string{
HasNoNewUnconfirmed(tx, m_pool, m_subpackage.m_all_conflicts)}) {
1114 Assume(!ws.m_sibling_eviction);
1116 strprintf(
"replacement-adds-unconfirmed%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1122 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1123 m_subpackage.m_conflicting_size += it->GetTxSize();
1125 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees, ws.m_modified_fees, ws.m_vsize,
1126 m_pool.m_opts.incremental_relay_feerate, hash)}) {
1129 strprintf(
"insufficient fee%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1134 bool MemPoolAccept::PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
1135 std::vector<Workspace>& workspaces,
1136 const int64_t total_vsize,
1143 assert(std::all_of(txns.cbegin(), txns.cend(), [
this](
const auto& tx)
1146 assert(txns.size() == workspaces.size());
1148 auto result = m_pool.CheckPackageLimits(txns, total_vsize);
1155 if (!m_subpackage.m_rbf)
return true;
1166 for (
const auto& ws : workspaces) {
1167 if (!ws.m_ancestors.empty()) {
1174 for (Workspace& ws : workspaces) {
1176 direct_conflict_iters.merge(ws.m_iters_conflicting);
1179 const auto& parent_ws = workspaces[0];
1180 const auto& child_ws = workspaces[1];
1185 m_subpackage.m_all_conflicts)}) {
1187 "package RBF failed: too many potential replacements", *err_string);
1191 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1192 m_subpackage.m_conflicting_size += it->GetTxSize();
1196 const Txid& child_hash = child_ws.m_ptx->GetHash();
1197 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees,
1198 m_subpackage.m_total_modified_fees,
1199 m_subpackage.m_total_vsize,
1200 m_pool.m_opts.incremental_relay_feerate, child_hash)}) {
1202 "package RBF failed: insufficient anti-DoS fees", *err_string);
1207 const CFeeRate parent_feerate(parent_ws.m_modified_fees, parent_ws.m_vsize);
1208 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1209 if (package_feerate <= parent_feerate) {
1211 "package RBF failed: package feerate is less than or equal to parent feerate",
1212 strprintf(
"package feerate %s <= parent feerate is %s", package_feerate.ToString(), parent_feerate.ToString()));
1217 if (
const auto err_tup{
ImprovesFeerateDiagram(m_pool, direct_conflict_iters, m_subpackage.m_all_conflicts, m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize)}) {
1219 "package RBF failed: " + err_tup.value().second,
"");
1223 txns.front()->GetHash().ToString(), txns.front()->GetWitnessHash().ToString(),
1224 txns.back()->GetHash().ToString(), txns.back()->GetWitnessHash().ToString());
1230 bool MemPoolAccept::PolicyScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1241 if (!
CheckInputScripts(tx, state, m_view, scriptVerifyFlags,
true,
false, ws.m_precomputed_txdata, GetValidationCache())) {
1258 bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1263 const uint256& hash = ws.m_hash;
1281 unsigned int currentBlockScriptVerifyFlags{
GetBlockScriptFlags(*m_active_chainstate.m_chain.Tip(), m_active_chainstate.m_chainman)};
1283 ws.m_precomputed_txdata, m_active_chainstate.CoinsTip(), GetValidationCache())) {
1284 LogPrintf(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against latest-block but not STANDARD flags %s, %s\n", hash.
ToString(), state.
ToString());
1291 bool MemPoolAccept::Finalize(
const ATMPArgs&
args, Workspace& ws)
1296 const uint256& hash = ws.m_hash;
1298 const bool bypass_limits =
args.m_bypass_limits;
1299 std::unique_ptr<CTxMemPoolEntry>& entry = ws.m_entry;
1301 if (!m_subpackage.m_all_conflicts.empty())
Assume(
args.m_allow_replacement);
1305 LogPrint(
BCLog::MEMPOOL,
"replacing mempool tx %s (wtxid=%s, fees=%s, vsize=%s). New tx %s (wtxid=%s, fees=%s, vsize=%s)\n",
1306 it->GetTx().GetHash().ToString(),
1307 it->GetTx().GetWitnessHash().ToString(),
1313 entry->GetTxSize());
1314 TRACE7(mempool, replaced,
1315 it->GetTx().GetHash().data(),
1318 std::chrono::duration_cast<std::chrono::duration<std::uint64_t>>(it->GetTime()).
count(),
1323 m_subpackage.m_replaced_transactions.push_back(it->GetSharedTx());
1328 m_subpackage.m_all_conflicts.clear();
1330 m_pool.addUnchecked(*entry, ws.m_ancestors);
1336 if (!
args.m_package_submission && !bypass_limits) {
1345 bool MemPoolAccept::SubmitPackage(
const ATMPArgs&
args, std::vector<Workspace>& workspaces,
1347 std::map<uint256, MempoolAcceptResult>& results)
1353 assert(std::all_of(workspaces.cbegin(), workspaces.cend(), [
this](
const auto& ws){
1354 return !m_pool.exists(
GenTxid::Txid(ws.m_ptx->GetHash())); }));
1356 bool all_submitted =
true;
1361 for (Workspace& ws : workspaces) {
1362 if (!ConsensusScriptChecks(
args, ws)) {
1366 all_submitted =
false;
1368 strprintf(
"BUG! PolicyScriptChecks succeeded but ConsensusScriptChecks failed: %s",
1369 ws.m_ptx->GetHash().ToString()));
1375 auto ancestors{m_pool.CalculateMemPoolAncestors(*ws.m_entry, m_pool.m_opts.limits)};
1380 all_submitted =
false;
1382 strprintf(
"BUG! Mempool ancestors or descendants were underestimated: %s",
1383 ws.m_ptx->GetHash().ToString()));
1385 ws.m_ancestors = std::move(ancestors).value_or(ws.m_ancestors);
1392 if (!Finalize(
args, ws)) {
1396 all_submitted =
false;
1398 strprintf(
"BUG! Adding to mempool failed: %s", ws.m_ptx->GetHash().ToString()));
1402 std::vector<Wtxid> all_package_wtxids;
1403 all_package_wtxids.reserve(workspaces.size());
1404 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1405 [](
const auto& ws) {
return ws.m_ptx->GetWitnessHash(); });
1407 if (!m_subpackage.m_replaced_transactions.empty()) {
1408 LogPrint(
BCLog::MEMPOOL,
"replaced %u mempool transactions with %u new one(s) for %s additional fees, %d delta bytes\n",
1409 m_subpackage.m_replaced_transactions.size(), workspaces.size(),
1410 m_subpackage.m_total_modified_fees - m_subpackage.m_conflicting_fees,
1411 m_subpackage.m_total_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1415 for (Workspace& ws : workspaces) {
1416 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1417 CFeeRate{ws.m_modified_fees,
static_cast<uint32_t
>(ws.m_vsize)};
1418 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1419 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1420 results.emplace(ws.m_ptx->GetWitnessHash(),
1422 ws.m_base_fees, effective_feerate, effective_feerate_wtxids));
1423 if (!m_pool.m_opts.signals)
continue;
1426 ws.m_vsize, ws.m_entry->GetHeight(),
1427 args.m_bypass_limits,
args.m_package_submission,
1429 m_pool.HasNoInputsOf(tx));
1430 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1432 return all_submitted;
1441 const std::vector<Wtxid> single_wtxid{ws.m_ptx->GetWitnessHash()};
1443 if (!PreChecks(
args, ws)) {
1452 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1457 if (m_subpackage.m_rbf && !ReplacementChecks(ws)) {
1471 const CFeeRate effective_feerate{ws.m_modified_fees,
static_cast<uint32_t
>(ws.m_vsize)};
1473 if (
args.m_test_accept) {
1475 ws.m_base_fees, effective_feerate, single_wtxid);
1478 if (!Finalize(
args, ws)) {
1485 if (m_pool.m_opts.signals) {
1488 ws.m_vsize, ws.m_entry->GetHeight(),
1489 args.m_bypass_limits,
args.m_package_submission,
1491 m_pool.HasNoInputsOf(tx));
1492 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1495 if (!m_subpackage.m_replaced_transactions.empty()) {
1496 LogPrint(
BCLog::MEMPOOL,
"replaced %u mempool transactions with 1 new transaction for %s additional fees, %d delta bytes\n",
1497 m_subpackage.m_replaced_transactions.size(),
1498 ws.m_modified_fees - m_subpackage.m_conflicting_fees,
1499 ws.m_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1503 effective_feerate, single_wtxid);
1514 std::vector<Workspace> workspaces{};
1515 workspaces.reserve(txns.size());
1516 std::transform(txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1517 [](
const auto& tx) {
return Workspace(tx); });
1518 std::map<uint256, MempoolAcceptResult> results;
1523 for (Workspace& ws : workspaces) {
1524 if (!PreChecks(
args, ws)) {
1533 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1551 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1556 for (Workspace& ws : workspaces) {
1572 m_subpackage.m_total_vsize = std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1573 [](int64_t
sum,
auto& ws) {
return sum + ws.m_vsize; });
1574 m_subpackage.m_total_modified_fees = std::accumulate(workspaces.cbegin(), workspaces.cend(),
CAmount{0},
1575 [](
CAmount sum,
auto& ws) {
return sum + ws.m_modified_fees; });
1576 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1577 std::vector<Wtxid> all_package_wtxids;
1578 all_package_wtxids.reserve(workspaces.size());
1579 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1580 [](
const auto& ws) {
return ws.m_ptx->GetWitnessHash(); });
1582 if (
args.m_package_feerates &&
1583 !
CheckFeeRate(m_subpackage.m_total_vsize, m_subpackage.m_total_modified_fees, placeholder_state)) {
1591 if (txns.size() > 1 && !PackageMempoolChecks(txns, workspaces, m_subpackage.m_total_vsize, package_state)) {
1595 for (Workspace& ws : workspaces) {
1596 ws.m_package_feerate = package_feerate;
1597 if (!PolicyScriptChecks(
args, ws)) {
1603 if (
args.m_test_accept) {
1604 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1605 CFeeRate{ws.m_modified_fees,
static_cast<uint32_t
>(ws.m_vsize)};
1606 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1607 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1608 results.emplace(ws.m_ptx->GetWitnessHash(),
1610 ws.m_vsize, ws.m_base_fees, effective_feerate,
1611 effective_feerate_wtxids));
1617 if (!SubmitPackage(
args, workspaces, package_state, results)) {
1625 void MemPoolAccept::CleanupTemporaryCoins()
1645 for (
const auto& outpoint : m_viewmempool.GetNonBaseCoins()) {
1648 m_view.Uncache(outpoint);
1651 m_viewmempool.Reset();
1659 if (subpackage.size() > 1) {
1660 return AcceptMultipleTransactions(subpackage,
args);
1662 const auto& tx = subpackage.front();
1663 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(
args);
1664 const auto single_res = AcceptSingleTransaction(tx, single_args);
1675 ClearSubPackageState();
1702 assert(package.size() > 1);
1705 const auto& child = package.back();
1706 std::unordered_set<uint256, SaltedTxidHasher> unconfirmed_parent_txids;
1707 std::transform(package.cbegin(), package.cend() - 1,
1708 std::inserter(unconfirmed_parent_txids, unconfirmed_parent_txids.end()),
1709 [](
const auto& tx) {
return tx->
GetHash(); });
1716 const CCoinsViewCache& coins_tip_cache = m_active_chainstate.CoinsTip();
1717 for (
const auto& input : child->vin) {
1718 if (!coins_tip_cache.HaveCoinInCache(input.prevout)) {
1719 args.m_coins_to_uncache.push_back(input.prevout);
1725 m_view.SetBackend(m_active_chainstate.CoinsTip());
1726 const auto package_or_confirmed = [
this, &unconfirmed_parent_txids](
const auto& input) {
1727 return unconfirmed_parent_txids.count(input.prevout.hash) > 0 || m_view.HaveCoin(input.prevout);
1729 if (!std::all_of(child->vin.cbegin(), child->vin.cend(), package_or_confirmed)) {
1735 m_view.SetBackend(m_dummy);
1740 std::map<uint256, MempoolAcceptResult> results_final;
1744 std::map<uint256, MempoolAcceptResult> individual_results_nonfinal;
1745 bool quit_early{
false};
1746 std::vector<CTransactionRef> txns_package_eval;
1747 for (
const auto& tx : package) {
1749 const auto& txid = tx->
GetHash();
1763 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1773 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1779 const auto single_package_res = AcceptSubPackage({tx},
args);
1780 const auto& single_res = single_package_res.m_tx_results.at(wtxid);
1785 results_final.emplace(wtxid, single_res);
1799 individual_results_nonfinal.emplace(wtxid, single_res);
1801 individual_results_nonfinal.emplace(wtxid, single_res);
1802 txns_package_eval.push_back(tx);
1807 auto multi_submission_result = quit_early || txns_package_eval.empty() ?
PackageMempoolAcceptResult(package_state_quit_early, {}) :
1808 AcceptSubPackage(txns_package_eval,
args);
1815 for (
const auto& tx : package) {
1817 if (multi_submission_result.m_tx_results.count(wtxid) > 0) {
1819 Assume(results_final.count(wtxid) == 0);
1822 const auto& txresult = multi_submission_result.m_tx_results.at(wtxid);
1829 results_final.emplace(wtxid, txresult);
1831 }
else if (
const auto it{results_final.find(wtxid)}; it != results_final.end()) {
1835 Assume(individual_results_nonfinal.count(wtxid) == 0);
1842 results_final.erase(wtxid);
1845 }
else if (
const auto it{individual_results_nonfinal.find(wtxid)}; it != individual_results_nonfinal.end()) {
1848 results_final.emplace(wtxid, it->second);
1851 Assume(results_final.size() == package.size());
1858 int64_t accept_time,
bool bypass_limits,
bool test_accept)
1865 std::vector<COutPoint> coins_to_uncache;
1866 auto args = MemPoolAccept::ATMPArgs::SingleAccept(chainparams, accept_time, bypass_limits, coins_to_uncache, test_accept);
1874 for (
const COutPoint& hashTx : coins_to_uncache)
1876 TRACE2(mempool, rejected,
1877 tx->GetHash().data(),
1888 const Package& package,
bool test_accept,
const std::optional<CFeeRate>& client_maxfeerate)
1891 assert(!package.empty());
1892 assert(std::all_of(package.cbegin(), package.cend(), [](
const auto& tx){
return tx !=
nullptr;}));
1894 std::vector<COutPoint> coins_to_uncache;
1899 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(chainparams,
GetTime(), coins_to_uncache);
1900 return MemPoolAccept(pool, active_chainstate).AcceptMultipleTransactions(package,
args);
1902 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(chainparams,
GetTime(), coins_to_uncache, client_maxfeerate);
1903 return MemPoolAccept(pool, active_chainstate).AcceptPackage(package,
args);
1908 if (test_accept || result.m_state.IsInvalid()) {
1909 for (
const COutPoint& hashTx : coins_to_uncache) {
1928 nSubsidy >>= halvings;
1933 : m_dbview{std::move(db_params), std::move(options)},
1934 m_catcherview(&m_dbview) {}
1936 void CoinsViews::InitCache()
1939 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1946 std::optional<uint256> from_snapshot_blockhash)
1947 : m_mempool(mempool),
1948 m_blockman(blockman),
1949 m_chainman(chainman),
1950 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1956 return m_cached_snapshot_base;
1959 void Chainstate::InitCoinsDB(
1960 size_t cache_size_bytes,
1972 .cache_bytes = cache_size_bytes,
1973 .memory_only = in_memory,
1974 .wipe_data = should_wipe,
1980 void Chainstate::InitCoinsCache(
size_t cache_size_bytes)
2006 if (chain.Tip() ==
nullptr) {
2015 LogPrintf(
"Leaving InitialBlockDownload (latching to false)\n");
2032 LogPrintf(
"%s: Warning: Found invalid chain at least ~6 blocks longer than our best chain.\nChain state database corruption likely.\n", __func__);
2035 _(
"Warning: We do not appear to fully agree with our peers! You may need to upgrade, or other nodes may need to upgrade."));
2052 LogPrintf(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n", __func__,
2057 LogPrintf(
"%s: current best=%s height=%d log2_work=%f date=%s\n", __func__,
2095 return VerifyScript(scriptSig,
m_tx_out.
scriptPubKey, witness,
nFlags,
CachingTransactionSignatureChecker(
ptxTo,
nIn,
m_tx_out.
nValue,
cacheStore, *
m_signature_cache, *
txdata), &
error);
2099 : m_signature_cache{signature_cache_bytes}
2106 m_script_execution_cache_hasher.Write(
nonce.begin(), 32);
2107 m_script_execution_cache_hasher.Write(
nonce.begin(), 32);
2109 const auto [num_elems, approx_size_bytes] = m_script_execution_cache.setup_bytes(script_execution_cache_bytes);
2110 LogPrintf(
"Using %zu MiB out of %zu MiB requested for script execution cache, able to store %zu elements\n",
2111 approx_size_bytes >> 20, script_execution_cache_bytes >> 20, num_elems);
2137 std::vector<CScriptCheck>* pvChecks)
2142 pvChecks->reserve(tx.
vin.size());
2159 std::vector<CTxOut> spent_outputs;
2160 spent_outputs.reserve(tx.
vin.size());
2162 for (
const auto& txin : tx.
vin) {
2166 spent_outputs.emplace_back(coin.
out);
2168 txdata.
Init(tx, std::move(spent_outputs));
2172 for (
unsigned int i = 0; i < tx.
vin.size(); i++) {
2183 pvChecks->emplace_back(std::move(check));
2184 }
else if (!check()) {
2212 if (cacheFullScriptStore && !pvChecks) {
2240 if (undo.nHeight == 0) {
2246 undo.nHeight = alternate.
nHeight;
2271 LogError(
"DisconnectBlock(): failure reading undo data\n");
2275 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
2276 LogError(
"DisconnectBlock(): block and undo data inconsistent\n");
2286 bool fEnforceBIP30 = !((pindex->
nHeight==91722 && pindex->
GetBlockHash() ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
2287 (pindex->
nHeight==91812 && pindex->
GetBlockHash() ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"}));
2290 for (
int i = block.
vtx.size() - 1; i >= 0; i--) {
2294 bool is_bip30_exception = (is_coinbase && !fEnforceBIP30);
2298 for (
size_t o = 0; o < tx.
vout.size(); o++) {
2299 if (!tx.
vout[o].scriptPubKey.IsUnspendable()) {
2304 if (!is_bip30_exception) {
2315 LogError(
"DisconnectBlock(): transaction and undo data inconsistent\n");
2318 for (
unsigned int j = tx.
vin.size(); j > 0;) {
2416 const auto time_start{SteadyClock::now()};
2432 if (!
CheckBlock(block, state, params.GetConsensus(), !fJustCheck, !fJustCheck)) {
2444 uint256 hashPrevBlock = pindex->
pprev ==
nullptr ?
uint256() : pindex->pprev->GetBlockHash();
2451 if (block_hash == params.GetConsensus().hashGenesisBlock) {
2457 bool fScriptChecks =
true;
2466 if (it->second.GetAncestor(pindex->
nHeight) == pindex &&
2488 const auto time_1{SteadyClock::now()};
2489 m_chainman.time_check += time_1 - time_start;
2491 Ticks<MillisecondsDouble>(time_1 - time_start),
2533 static constexpr
int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2565 fEnforceBIP30 = fEnforceBIP30 && (!pindexBIP34height || !(pindexBIP34height->
GetBlockHash() == params.GetConsensus().BIP34Hash));
2570 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2571 for (
const auto& tx : block.
vtx) {
2572 for (
size_t o = 0; o < tx->
vout.size(); o++) {
2574 LogPrintf(
"ERROR: ConnectBlock(): tried to overwrite transaction\n");
2582 int nLockTimeFlags = 0;
2590 const auto time_2{SteadyClock::now()};
2593 Ticks<MillisecondsDouble>(time_2 - time_1),
2605 std::vector<PrecomputedTransactionData> txsdata(block.
vtx.size());
2607 std::vector<int> prevheights;
2610 int64_t nSigOpsCost = 0;
2611 blockundo.
vtxundo.reserve(block.
vtx.size() - 1);
2612 for (
unsigned int i = 0; i < block.
vtx.size(); i++)
2616 nInputs += tx.
vin.size();
2631 LogPrintf(
"ERROR: %s: accumulated fee in the block out of range.\n", __func__);
2638 prevheights.resize(tx.
vin.size());
2639 for (
size_t j = 0; j < tx.
vin.size(); j++) {
2643 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2644 LogPrintf(
"ERROR: %s: contains a non-BIP68-final transaction\n", __func__);
2655 LogPrintf(
"ERROR: ConnectBlock(): too many sigops\n");
2661 std::vector<CScriptCheck> vChecks;
2662 bool fCacheResults = fJustCheck;
2668 LogError(
"ConnectBlock(): CheckInputScripts on %s failed with %s\n",
2672 control.Add(std::move(vChecks));
2677 blockundo.
vtxundo.emplace_back();
2681 const auto time_3{SteadyClock::now()};
2683 LogPrint(
BCLog::BENCH,
" - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) [%.2fs (%.2fms/blk)]\n", (
unsigned)block.
vtx.size(),
2684 Ticks<MillisecondsDouble>(time_3 - time_2), Ticks<MillisecondsDouble>(time_3 - time_2) / block.
vtx.size(),
2685 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_3 - time_2) / (nInputs - 1),
2686 Ticks<SecondsDouble>(
m_chainman.time_connect),
2690 if (block.
vtx[0]->GetValueOut() > blockReward) {
2691 LogPrintf(
"ERROR: ConnectBlock(): coinbase pays too much (actual=%d vs limit=%d)\n", block.
vtx[0]->GetValueOut(), blockReward);
2695 if (!control.Wait()) {
2696 LogPrintf(
"ERROR: %s: CheckQueue failed\n", __func__);
2699 const auto time_4{SteadyClock::now()};
2701 LogPrint(
BCLog::BENCH,
" - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n", nInputs - 1,
2702 Ticks<MillisecondsDouble>(time_4 - time_2),
2703 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_4 - time_2) / (nInputs - 1),
2704 Ticks<SecondsDouble>(
m_chainman.time_verify),
2710 if (!
m_blockman.WriteUndoDataForBlock(blockundo, state, *pindex)) {
2714 const auto time_5{SteadyClock::now()};
2717 Ticks<MillisecondsDouble>(time_5 - time_4),
2729 const auto time_6{SteadyClock::now()};
2732 Ticks<MillisecondsDouble>(time_6 - time_5),
2736 TRACE6(validation, block_connected,
2751 return this->GetCoinsCacheSizeState(
2757 size_t max_coins_cache_size_bytes,
2758 size_t max_mempool_size_bytes)
2763 int64_t nTotalSpace =
2764 max_coins_cache_size_bytes + std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2767 static constexpr int64_t MAX_BLOCK_COINSDB_USAGE_BYTES = 10 * 1024 * 1024;
2768 int64_t large_threshold =
2769 std::max((9 * nTotalSpace) / 10, nTotalSpace - MAX_BLOCK_COINSDB_USAGE_BYTES);
2771 if (cacheSize > nTotalSpace) {
2772 LogPrintf(
"Cache size (%s) exceeds total space (%s)\n", cacheSize, nTotalSpace);
2774 }
else if (cacheSize > large_threshold) {
2780 bool Chainstate::FlushStateToDisk(
2783 int nManualPruneHeight)
2786 assert(this->CanFlushToDisk());
2787 std::set<int> setFilesToPrune;
2788 bool full_flush_completed =
false;
2795 bool fFlushForPrune =
false;
2796 bool fDoFullFlush =
false;
2804 std::optional<std::string> limiting_lock;
2806 for (
const auto& prune_lock :
m_blockman.m_prune_locks) {
2807 if (prune_lock.second.height_first == std::numeric_limits<int>::max())
continue;
2810 last_prune = std::max(1, std::min(last_prune, lock_height));
2811 if (last_prune == lock_height) {
2812 limiting_lock = prune_lock.first;
2816 if (limiting_lock) {
2817 LogPrint(
BCLog::PRUNE,
"%s limited pruning to height %d\n", limiting_lock.value(), last_prune);
2820 if (nManualPruneHeight > 0) {
2825 std::min(last_prune, nManualPruneHeight),
2833 if (!setFilesToPrune.empty()) {
2834 fFlushForPrune =
true;
2836 m_blockman.m_block_tree_db->WriteFlag(
"prunedblockfiles",
true);
2841 const auto nNow{SteadyClock::now()};
2858 fDoFullFlush = (mode ==
FlushStateMode::ALWAYS) || fCacheLarge || fCacheCritical || fPeriodicFlush || fFlushForPrune;
2860 if (fDoFullFlush || fPeriodicWrite) {
2885 if (fFlushForPrune) {
2893 if (fDoFullFlush && !
CoinsTip().GetBestBlock().IsNull()) {
2911 full_flush_completed =
true;
2913 int64_t{Ticks<std::chrono::microseconds>(SteadyClock::now() - nNow)},
2915 (uint64_t)coins_count,
2916 (uint64_t)coins_mem_usage,
2917 (
bool)fFlushForPrune);
2924 }
catch (
const std::runtime_error& e) {
2951 const std::string& func_name,
2952 const std::string&
prefix,
2957 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",
2963 coins_tip.DynamicMemoryUsage() * (1.0 / (1 << 20)),
2964 coins_tip.GetCacheSize(),
2965 !warning_messages.empty() ?
strprintf(
" warning='%s'", warning_messages) :
"");
2968 void Chainstate::UpdateTip(
const CBlockIndex* pindexNew)
2971 const auto& coins_tip = this->
CoinsTip();
2979 constexpr
int BACKGROUND_LOG_INTERVAL = 2000;
2980 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2981 UpdateTipLog(coins_tip, pindexNew, params, __func__,
"[background validation] ",
"");
2997 std::vector<bilingual_str> warning_messages;
3008 warning_messages.push_back(warning);
3013 UpdateTipLog(coins_tip, pindexNew, params, __func__,
"",
3036 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
3039 LogError(
"DisconnectTip(): Failed to read block\n");
3043 const auto time_start{SteadyClock::now()};
3047 if (DisconnectBlock(block, pindexDelete, view) !=
DISCONNECT_OK) {
3051 bool flushed = view.
Flush();
3055 Ticks<MillisecondsDouble>(SteadyClock::now() - time_start));
3059 const int max_height_first{pindexDelete->
nHeight - 1};
3060 for (
auto& prune_lock :
m_blockman.m_prune_locks) {
3061 if (prune_lock.second.height_first <= max_height_first)
continue;
3063 prune_lock.second.height_first = max_height_first;
3064 LogPrint(
BCLog::PRUNE,
"%s prune lock moved back to %d\n", prune_lock.first, max_height_first);
3083 UpdateTip(pindexDelete->
pprev);
3145 const auto time_1{SteadyClock::now()};
3146 std::shared_ptr<const CBlock> pthisBlock;
3148 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
3152 pthisBlock = pblockNew;
3155 pthisBlock = pblock;
3157 const CBlock& blockConnecting = *pthisBlock;
3159 const auto time_2{SteadyClock::now()};
3160 SteadyClock::time_point time_3;
3164 Ticks<MillisecondsDouble>(time_2 - time_1));
3167 bool rv =
ConnectBlock(blockConnecting, state, pindexNew, view);
3177 time_3 = SteadyClock::now();
3178 m_chainman.time_connect_total += time_3 - time_2;
3181 Ticks<MillisecondsDouble>(time_3 - time_2),
3182 Ticks<SecondsDouble>(
m_chainman.time_connect_total),
3184 bool flushed = view.
Flush();
3187 const auto time_4{SteadyClock::now()};
3190 Ticks<MillisecondsDouble>(time_4 - time_3),
3197 const auto time_5{SteadyClock::now()};
3198 m_chainman.time_chainstate += time_5 - time_4;
3200 Ticks<MillisecondsDouble>(time_5 - time_4),
3201 Ticks<SecondsDouble>(
m_chainman.time_chainstate),
3210 UpdateTip(pindexNew);
3212 const auto time_6{SteadyClock::now()};
3213 m_chainman.time_post_connect += time_6 - time_5;
3216 Ticks<MillisecondsDouble>(time_6 - time_5),
3217 Ticks<SecondsDouble>(
m_chainman.time_post_connect),
3220 Ticks<MillisecondsDouble>(time_6 - time_1),
3229 m_chainman.MaybeCompleteSnapshotValidation();
3257 bool fInvalidAncestor =
false;
3267 if (fFailedChain || fMissingData) {
3274 while (pindexTest != pindexFailed) {
3278 }
else if (fMissingData) {
3283 std::make_pair(pindexFailed->
pprev, pindexFailed));
3286 pindexFailed = pindexFailed->
pprev;
3289 fInvalidAncestor =
true;
3292 pindexTest = pindexTest->
pprev;
3294 if (!fInvalidAncestor)
3326 bool fBlocksDisconnected =
false;
3340 fBlocksDisconnected =
true;
3344 std::vector<CBlockIndex*> vpindexToConnect;
3345 bool fContinue =
true;
3350 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3351 vpindexToConnect.clear();
3352 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3355 vpindexToConnect.push_back(pindexIter);
3356 pindexIter = pindexIter->
pprev;
3361 for (
CBlockIndex* pindexConnect : vpindexToConnect | std::views::reverse) {
3362 if (!
ConnectTip(state, pindexConnect, pindexConnect == pindexMostWork ? pblock : std::shared_ptr<const CBlock>(), connectTrace, disconnectpool)) {
3369 fInvalidFound =
true;
3390 if (fBlocksDisconnected) {
3411 bool fNotify =
false;
3412 bool fInitialBlockDownload =
false;
3416 pindexHeader = m_best_header;
3418 if (pindexHeader != m_last_notified_header) {
3421 m_last_notified_header = pindexHeader;
3434 if (signals.CallbacksPending() > 10) {
3435 signals.SyncWithValidationInterfaceQueue();
3439 bool Chainstate::ActivateBestChain(
BlockValidationState& state, std::shared_ptr<const CBlock> pblock)
3458 LogPrintf(
"m_disabled is set - this chainstate should not be in operation. " 3465 bool exited_ibd{
false};
3479 LOCK(MempoolMutex());
3482 bool blocks_connected =
false;
3488 if (pindexMostWork ==
nullptr) {
3493 if (pindexMostWork ==
nullptr || pindexMostWork ==
m_chain.
Tip()) {
3497 bool fInvalidFound =
false;
3498 std::shared_ptr<const CBlock> nullBlockPtr;
3499 if (!
ActivateBestChainStep(state, pindexMostWork, pblock && pblock->GetHash() == pindexMostWork->
GetBlockHash() ? pblock : nullBlockPtr, fInvalidFound, connectTrace)) {
3503 blocks_connected =
true;
3505 if (fInvalidFound) {
3507 pindexMostWork =
nullptr;
3512 assert(trace.pblock && trace.pindex);
3527 if (!blocks_connected)
return true;
3532 if (was_in_ibd && !still_in_ibd) {
3589 }
while (pindexNewTip != pindexMostWork);
3629 return ActivateBestChain(state, std::shared_ptr<const CBlock>());
3639 if (pindex->
nHeight == 0)
return false;
3642 bool pindex_was_in_chain =
false;
3643 int disconnected = 0;
3657 std::multimap<const arith_uint256, CBlockIndex *> candidate_blocks_by_work;
3661 for (
auto& entry :
m_blockman.m_block_index) {
3672 candidate_blocks_by_work.insert(std::make_pair(candidate->
nChainWork, candidate));
3687 LOCK(MempoolMutex());
3689 pindex_was_in_chain =
true;
3702 if (!
ret)
return false;
3722 auto candidate_it = candidate_blocks_by_work.lower_bound(invalid_walk_tip->
pprev->
nChainWork);
3723 while (candidate_it != candidate_blocks_by_work.end()) {
3726 candidate_it = candidate_blocks_by_work.erase(candidate_it);
3734 to_mark_failed = invalid_walk_tip;
3759 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3769 if (pindex_was_in_chain) {
3794 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3795 if (!block_index.IsValid() && block_index.GetAncestor(
nHeight) == pindex) {
3801 if (&block_index ==
m_chainman.m_best_invalid) {
3810 while (pindex !=
nullptr) {
3816 pindex = pindex->
pprev;
3830 if (is_active_chainstate) {
3834 }
else if (!m_disabled) {
3839 if (snapshot_base->GetAncestor(pindex->
nHeight) == pindex) {
3849 pindexNew->
nTx = block.
vtx.size();
3855 auto prev_tx_sum = [](
CBlockIndex& block) {
return block.nTx + (block.pprev ? block.pprev->m_chain_tx_count : 0); };
3857 pindexNew == GetSnapshotBaseBlock())) {
3858 LogWarning(
"Internal bug detected: block %d has unexpected m_chain_tx_count %i that should be %i (%s %s). Please report this issue here: %s\n",
3862 pindexNew->nFile = pos.
nFile;
3863 pindexNew->nDataPos = pos.
nPos;
3864 pindexNew->nUndoPos = 0;
3874 std::deque<CBlockIndex*> queue;
3875 queue.push_back(pindexNew);
3878 while (!queue.empty()) {
3886 LogWarning(
"Internal bug detected: block %d has unexpected m_chain_tx_count %i that should be %i (%s %s). Please report this issue here: %s\n",
3892 c->TryAddBlockIndexCandidate(pindex);
3894 std::pair<std::multimap<CBlockIndex*, CBlockIndex*>::iterator, std::multimap<CBlockIndex*, CBlockIndex*>::iterator> range =
m_blockman.
m_blocks_unlinked.equal_range(pindex);
3895 while (range.first != range.second) {
3896 std::multimap<CBlockIndex*, CBlockIndex*>::iterator it = range.first;
3897 queue.push_back(it->second);
3928 "hashMerkleRoot mismatch");
3937 "bad-txns-duplicate",
3938 "duplicate transaction");
3953 if (expect_witness_commitment) {
3958 assert(!block.
vtx.empty() && !block.
vtx[0]->vin.empty());
3959 const auto& witness_stack{block.
vtx[0]->vin[0].scriptWitness.stack};
3961 if (witness_stack.size() != 1 || witness_stack[0].size() != 32) {
3964 "bad-witness-nonce-size",
3965 strprintf(
"%s : invalid witness reserved value size", __func__));
3974 if (memcmp(hash_witness.
begin(), &block.
vtx[0]->vout[commitpos].scriptPubKey[6], 32)) {
3977 "bad-witness-merkle-match",
3978 strprintf(
"%s : witness merkle commitment mismatch", __func__));
3987 for (
const auto& tx : block.
vtx) {
3991 "unexpected-witness",
3992 strprintf(
"%s : unexpected witness data found", __func__));
4032 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase())
4034 for (
unsigned int i = 1; i < block.
vtx.size(); i++)
4035 if (block.
vtx[i]->IsCoinBase())
4040 for (
const auto& tx : block.
vtx) {
4050 unsigned int nSigOps = 0;
4051 for (
const auto& tx : block.
vtx)
4058 if (fCheckPOW && fCheckMerkleRoot)
4064 void ChainstateManager::UpdateUncommittedBlockStructures(
CBlock& block,
const CBlockIndex* pindexPrev)
const 4067 static const std::vector<unsigned char>
nonce(32, 0x00);
4070 tx.
vin[0].scriptWitness.stack.resize(1);
4071 tx.
vin[0].scriptWitness.stack[0] =
nonce;
4078 std::vector<unsigned char> commitment;
4080 std::vector<unsigned char>
ret(32, 0x00);
4088 out.scriptPubKey[1] = 0x24;
4089 out.scriptPubKey[2] = 0xaa;
4090 out.scriptPubKey[3] = 0x21;
4091 out.scriptPubKey[4] = 0xa9;
4092 out.scriptPubKey[5] = 0xed;
4093 memcpy(&
out.scriptPubKey[6], witnessroot.
begin(), 32);
4094 commitment = std::vector<unsigned char>(
out.scriptPubKey.begin(),
out.scriptPubKey.end());
4096 tx.vout.push_back(
out);
4099 UpdateUncommittedBlockStructures(block, pindexPrev);
4105 return std::all_of(headers.cbegin(), headers.cend(),
4106 [&](
const auto& header) {
return CheckProofOfWork(header.GetHash(), header.nBits, consensusParams);});
4117 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase()) {
4124 return std::any_of(block.
vtx.begin(), block.
vtx.end(),
4162 assert(pindexPrev !=
nullptr);
4171 if (chainman.m_options.checkpoints_enabled) {
4175 const CBlockIndex* pcheckpoint = blockman.GetLastCheckpoint(chainman.GetParams().Checkpoints());
4176 if (pcheckpoint && nHeight < pcheckpoint->
nHeight) {
4177 LogPrintf(
"ERROR: %s: forked chain older than last checkpoint (height %d)\n", __func__,
nHeight);
4222 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4225 bool enforce_locktime_median_time_past{
false};
4227 assert(pindexPrev !=
nullptr);
4228 enforce_locktime_median_time_past =
true;
4231 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past ?
4236 for (
const auto& tx : block.
vtx) {
4246 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4247 !std::equal(
expect.begin(),
expect.end(), block.
vtx[0]->vin[0].scriptSig.begin())) {
4283 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4285 if (miSelf !=
m_blockman.m_block_index.end()) {
4309 pindexPrev = &((*mi).second);
4344 if (pindexPrev->
GetAncestor(failedit->nHeight) == failedit) {
4347 while (invalid_walk != failedit) {
4350 invalid_walk = invalid_walk->
pprev;
4358 if (!min_pow_checked) {
4376 "Saw new header hash=%s height=%d", hash.
ToString(), pindex->
nHeight);
4410 blocks_left = std::max<int64_t>(0, blocks_left);
4411 const double progress{100.0 * last_accepted.nHeight / (last_accepted.nHeight + blocks_left)};
4412 LogInfo(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n", last_accepted.nHeight, progress);
4429 auto now = std::chrono::steady_clock::now();
4430 if (now < m_last_presync_update + std::chrono::milliseconds{250})
return;
4431 m_last_presync_update = now;
4435 if (initial_download) {
4437 blocks_left = std::max<int64_t>(0, blocks_left);
4438 const double progress{100.0 * height / (height + blocks_left)};
4439 LogInfo(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n", height, progress);
4446 const CBlock& block = *pblock;
4448 if (fNewBlock) *fNewBlock =
false;
4452 CBlockIndex *&pindex = ppindex ? *ppindex : pindexDummy;
4454 bool accepted_header{
AcceptBlockHeader(block, state, &pindex, min_pow_checked)};
4457 if (!accepted_header)
4481 if (fAlreadyHave)
return true;
4483 if (pindex->
nTx != 0)
return true;
4484 if (!fHasMoreOrSameWork)
return true;
4485 if (fTooFarAhead)
return true;
4496 if (!
CheckBlock(block, state, params.GetConsensus()) ||
4513 if (fNewBlock) *fNewBlock =
true;
4521 if (blockPos.IsNull()) {
4522 state.
Error(
strprintf(
"%s: Failed to find position to write new block to disk", __func__));
4527 }
catch (
const std::runtime_error& e) {
4551 if (new_block) *new_block =
false;
4566 ret =
AcceptBlock(block, state, &pindex, force_processing,
nullptr, new_block, min_pow_checked);
4581 LogError(
"%s: ActivateBestChain failed (%s)\n", __func__, state.
ToString());
4587 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
4588 LogError(
"%s: [background] ActivateBestChain failed (%s)\n", __func__, bg_state.
ToString());
4615 bool fCheckMerkleRoot)
4622 indexDummy.pprev = pindexPrev;
4624 indexDummy.phashBlock = &block_hash;
4628 LogError(
"%s: Consensus::ContextualCheckBlockHeader: %s\n", __func__, state.
ToString());
4636 LogError(
"%s: Consensus::ContextualCheckBlock: %s\n", __func__, state.
ToString());
4639 if (!chainstate.
ConnectBlock(block, state, &indexDummy, viewNew,
true)) {
4651 if (!active_chainstate.FlushStateToDisk(
4664 if (tip && tip->GetBlockHash() == coins_cache.
GetBestBlock()) {
4677 LogPrintf(
"Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f\n",
4678 tip->GetBlockHash().ToString(),
4700 int nCheckLevel,
int nCheckDepth)
4709 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
4712 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
4713 LogPrintf(
"Verifying last %i blocks at level %i\n", nCheckDepth, nCheckLevel);
4717 int nGoodTransactions = 0;
4720 bool skipped_no_block_data{
false};
4721 bool skipped_l3_checks{
false};
4722 LogPrintf(
"Verification progress: 0%%\n");
4727 const int percentageDone = std::max(1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (double)nCheckDepth * (nCheckLevel >= 4 ? 50 : 100))));
4728 if (reportDone < percentageDone / 10) {
4730 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
4731 reportDone = percentageDone / 10;
4740 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);
4741 skipped_no_block_data =
true;
4751 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensus_params)) {
4752 LogPrintf(
"Verification error: found bad block at %d, hash=%s (%s)\n",
4757 if (nCheckLevel >= 2 && pindex) {
4769 if (nCheckLevel >= 3) {
4778 nGoodTransactions = 0;
4779 pindexFailure = pindex;
4781 nGoodTransactions += block.
vtx.size();
4784 skipped_l3_checks =
true;
4789 if (pindexFailure) {
4790 LogPrintf(
"Verification error: coin database inconsistencies found (last %i blocks, %i good transactions before that)\n", chainstate.
m_chain.
Height() - pindexFailure->
nHeight + 1, nGoodTransactions);
4793 if (skipped_l3_checks) {
4794 LogPrintf(
"Skipped verification of level >=3 (insufficient database cache size). Consider increasing -dbcache.\n");
4801 if (nCheckLevel >= 4 && !skipped_l3_checks) {
4803 const int percentageDone = std::max(1, std::min(99, 100 - (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (double)nCheckDepth * 50)));
4804 if (reportDone < percentageDone / 10) {
4806 LogPrintf(
"Verification progress: %d%%\n", percentageDone);
4807 reportDone = percentageDone / 10;
4816 if (!chainstate.
ConnectBlock(block, state, pindex, coins)) {
4824 LogPrintf(
"Verification: No coin database inconsistencies in last %i blocks (%i transactions)\n", block_count, nGoodTransactions);
4826 if (skipped_l3_checks) {
4829 if (skipped_no_block_data) {
4847 if (!tx->IsCoinBase()) {
4848 for (
const CTxIn &txin : tx->vin) {
4866 if (hashHeads.empty())
return true;
4867 if (hashHeads.size() != 2) {
4868 LogError(
"ReplayBlocks(): unknown inconsistent state\n");
4879 if (
m_blockman.m_block_index.count(hashHeads[0]) == 0) {
4880 LogError(
"ReplayBlocks(): reorganization to unknown block requested\n");
4883 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
4885 if (!hashHeads[1].IsNull()) {
4886 if (
m_blockman.m_block_index.count(hashHeads[1]) == 0) {
4887 LogError(
"ReplayBlocks(): reorganization from unknown block requested\n");
4890 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
4892 assert(pindexFork !=
nullptr);
4896 while (pindexOld != pindexFork) {
4914 pindexOld = pindexOld->
pprev;
4918 int nForkHeight = pindexFork ? pindexFork->
nHeight : 0;
4945 block = block->pprev;
4951 void Chainstate::ClearBlockIndexCandidates()
4963 if (!
ret)
return false;
4965 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
4967 std::vector<CBlockIndex*> vSortedByHeight{
m_blockman.GetAllBlockIndices()};
4968 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
4978 if (pindex == GetSnapshotBaseBlock() ||
4983 chainstate->TryAddBlockIndexCandidate(pindex);
4987 m_best_invalid = pindex;
4990 m_best_header = pindex;
5006 if (
m_blockman.m_block_index.count(params.GenesisBlock().GetHash()))
5010 const CBlock& block = params.GenesisBlock();
5012 if (blockPos.IsNull()) {
5013 LogError(
"%s: writing genesis block to disk failed\n", __func__);
5018 }
catch (
const std::runtime_error& e) {
5019 LogError(
"%s: failed to write genesis block: %s\n", __func__, e.what());
5029 std::multimap<uint256, FlatFilePos>* blocks_with_unknown_parent)
5032 assert(!dbp == !blocks_with_unknown_parent);
5034 const auto start{SteadyClock::now()};
5042 uint64_t nRewind = blkdat.GetPos();
5043 while (!blkdat.eof()) {
5046 blkdat.SetPos(nRewind);
5049 unsigned int nSize = 0;
5053 blkdat.FindByte(std::byte(params.MessageStart()[0]));
5054 nRewind = blkdat.GetPos() + 1;
5056 if (buf != params.MessageStart()) {
5063 }
catch (
const std::exception&) {
5070 const uint64_t nBlockPos{blkdat.GetPos()};
5072 dbp->
nPos = nBlockPos;
5073 blkdat.SetLimit(nBlockPos + nSize);
5076 const uint256 hash{header.GetHash()};
5079 nRewind = nBlockPos + nSize;
5080 blkdat.SkipTo(nRewind);
5082 std::shared_ptr<CBlock> pblock{};
5089 header.hashPrevBlock.ToString());
5090 if (dbp && blocks_with_unknown_parent) {
5091 blocks_with_unknown_parent->emplace(header.hashPrevBlock, *dbp);
5100 blkdat.SetPos(nBlockPos);
5101 pblock = std::make_shared<CBlock>();
5103 nRewind = blkdat.GetPos();
5106 if (
AcceptBlock(pblock, state,
nullptr,
true, dbp,
nullptr,
true)) {
5112 }
else if (hash != params.GetConsensus().hashGenesisBlock && pindex->
nHeight % 1000 == 0) {
5118 if (hash == params.GetConsensus().hashGenesisBlock) {
5119 bool genesis_activation_failure =
false;
5120 for (
auto c :
GetAll()) {
5122 if (!c->ActivateBestChain(state,
nullptr)) {
5123 genesis_activation_failure =
true;
5127 if (genesis_activation_failure) {
5140 bool activation_failure =
false;
5141 for (
auto c :
GetAll()) {
5143 if (!c->ActivateBestChain(state, pblock)) {
5145 activation_failure =
true;
5149 if (activation_failure) {
5156 if (!blocks_with_unknown_parent)
continue;
5159 std::deque<uint256> queue;
5160 queue.push_back(hash);
5161 while (!queue.empty()) {
5164 auto range = blocks_with_unknown_parent->equal_range(head);
5165 while (range.first != range.second) {
5166 std::multimap<uint256, FlatFilePos>::iterator it = range.first;
5167 std::shared_ptr<CBlock> pblockrecursive = std::make_shared<CBlock>();
5169 LogPrint(
BCLog::REINDEX,
"%s: Processing out of order child %s of %s\n", __func__, pblockrecursive->GetHash().ToString(),
5173 if (
AcceptBlock(pblockrecursive, dummy,
nullptr,
true, &it->second,
nullptr,
true)) {
5175 queue.push_back(pblockrecursive->GetHash());
5179 blocks_with_unknown_parent->erase(it);
5183 }
catch (
const std::exception& e) {
5195 LogPrint(
BCLog::REINDEX,
"%s: unexpected data at file offset 0x%x - %s. continuing\n", __func__, (nRewind - 1), e.what());
5198 }
catch (
const std::runtime_error& e) {
5201 LogPrintf(
"Loaded %i blocks from external file in %dms\n", nLoaded, Ticks<std::chrono::milliseconds>(SteadyClock::now() - start));
5235 best_hdr_chain.
SetTip(*m_best_header);
5237 std::multimap<CBlockIndex*,CBlockIndex*> forward;
5238 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
5240 if (!best_hdr_chain.
Contains(&block_index)) {
5242 assert(block_index.pprev);
5243 forward.emplace(block_index.pprev, &block_index);
5259 CBlockIndex* pindexFirstNotTransactionsValid =
nullptr;
5261 CBlockIndex* pindexFirstNotScriptsValid =
nullptr;
5267 const CBlockIndex* snap_base{GetSnapshotBaseBlock()};
5268 CBlockIndex *snap_first_missing{}, *snap_first_notx{}, *snap_first_notv{}, *snap_first_nocv{}, *snap_first_nosv{};
5269 auto snap_update_firsts = [&] {
5270 if (pindex == snap_base) {
5271 std::swap(snap_first_missing, pindexFirstMissing);
5272 std::swap(snap_first_notx, pindexFirstNeverProcessed);
5273 std::swap(snap_first_notv, pindexFirstNotTransactionsValid);
5274 std::swap(snap_first_nocv, pindexFirstNotChainValid);
5275 std::swap(snap_first_nosv, pindexFirstNotScriptsValid);
5279 while (pindex !=
nullptr) {
5281 if (pindexFirstInvalid ==
nullptr && pindex->nStatus &
BLOCK_FAILED_VALID) pindexFirstInvalid = pindex;
5282 if (pindexFirstMissing ==
nullptr && !(pindex->nStatus &
BLOCK_HAVE_DATA)) {
5283 pindexFirstMissing = pindex;
5285 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) pindexFirstNeverProcessed = pindex;
5286 if (pindex->pprev !=
nullptr && pindexFirstNotTreeValid ==
nullptr && (pindex->nStatus &
BLOCK_VALID_MASK) <
BLOCK_VALID_TREE) pindexFirstNotTreeValid = pindex;
5288 if (pindex->pprev !=
nullptr) {
5289 if (pindexFirstNotTransactionsValid ==
nullptr &&
5291 pindexFirstNotTransactionsValid = pindex;
5294 if (pindexFirstNotChainValid ==
nullptr &&
5296 pindexFirstNotChainValid = pindex;
5299 if (pindexFirstNotScriptsValid ==
nullptr &&
5301 pindexFirstNotScriptsValid = pindex;
5306 if (pindex->pprev ==
nullptr) {
5309 for (
auto c :
GetAll()) {
5310 if (c->m_chain.Genesis() !=
nullptr) {
5311 assert(pindex == c->m_chain.Genesis());
5315 if (!pindex->HaveNumChainTxs())
assert(pindex->nSequenceId <= 0);
5321 assert(pindexFirstMissing == pindexFirstNeverProcessed);
5327 if (snap_base && snap_base->GetAncestor(pindex->nHeight) == pindex) {
5336 assert((pindexFirstNeverProcessed ==
nullptr || pindex == snap_base) == pindex->HaveNumChainTxs());
5337 assert((pindexFirstNotTransactionsValid ==
nullptr || pindex == snap_base) == pindex->HaveNumChainTxs());
5339 assert(pindex->pprev ==
nullptr || pindex->nChainWork >= pindex->pprev->nChainWork);
5341 assert(pindexFirstNotTreeValid ==
nullptr);
5345 if (pindexFirstInvalid ==
nullptr) {
5350 if (!pindex->pprev) {
5352 assert(pindex->m_chain_tx_count == pindex->nTx);
5353 }
else if (pindex->pprev->m_chain_tx_count > 0 && pindex->nTx > 0) {
5355 assert(pindex->m_chain_tx_count == pindex->nTx + pindex->pprev->m_chain_tx_count);
5359 assert((pindex->m_chain_tx_count != 0) == (pindex == snap_base));
5363 for (
auto c :
GetAll()) {
5364 if (c->m_chain.Tip() ==
nullptr)
continue;
5378 if (!
CBlockIndexWorkComparator()(pindex, c->m_chain.Tip()) && (pindexFirstNeverProcessed ==
nullptr || pindex == snap_base)) {
5382 if (pindexFirstInvalid ==
nullptr) {
5401 if (pindexFirstMissing ==
nullptr || pindex == c->m_chain.Tip() || pindex == c->SnapshotBase()) {
5408 assert(c->setBlockIndexCandidates.count(pindex));
5416 assert(c->setBlockIndexCandidates.count(pindex) == 0);
5420 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> rangeUnlinked =
m_blockman.
m_blocks_unlinked.equal_range(pindex->pprev);
5421 bool foundInUnlinked =
false;
5422 while (rangeUnlinked.first != rangeUnlinked.second) {
5423 assert(rangeUnlinked.first->first == pindex->pprev);
5424 if (rangeUnlinked.first->second == pindex) {
5425 foundInUnlinked =
true;
5428 rangeUnlinked.first++;
5430 if (pindex->pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed !=
nullptr && pindexFirstInvalid ==
nullptr) {
5435 if (pindexFirstMissing ==
nullptr)
assert(!foundInUnlinked);
5436 if (pindex->pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed ==
nullptr && pindexFirstMissing !=
nullptr) {
5447 for (
auto c :
GetAll()) {
5450 if (pindexFirstInvalid ==
nullptr) {
5451 if (is_active || snap_base->GetAncestor(pindex->nHeight) == pindex) {
5463 snap_update_firsts();
5464 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> range = forward.equal_range(pindex);
5465 if (range.first != range.second) {
5467 pindex = range.first->second;
5470 }
else if (best_hdr_chain.
Contains(pindex)) {
5473 pindex = best_hdr_chain[
nHeight];
5481 snap_update_firsts();
5483 if (pindex == pindexFirstInvalid) pindexFirstInvalid =
nullptr;
5484 if (pindex == pindexFirstMissing) pindexFirstMissing =
nullptr;
5485 if (pindex == pindexFirstNeverProcessed) pindexFirstNeverProcessed =
nullptr;
5486 if (pindex == pindexFirstNotTreeValid) pindexFirstNotTreeValid =
nullptr;
5487 if (pindex == pindexFirstNotTransactionsValid) pindexFirstNotTransactionsValid =
nullptr;
5488 if (pindex == pindexFirstNotChainValid) pindexFirstNotChainValid =
nullptr;
5489 if (pindex == pindexFirstNotScriptsValid) pindexFirstNotScriptsValid =
nullptr;
5493 std::pair<std::multimap<CBlockIndex*,CBlockIndex*>::iterator,std::multimap<CBlockIndex*,CBlockIndex*>::iterator> rangePar = forward.equal_range(pindexPar);
5494 while (rangePar.first->second != pindex) {
5495 assert(rangePar.first != rangePar.second);
5500 if (rangePar.first != rangePar.second) {
5502 pindex = rangePar.first->second;
5504 }
else if (pindexPar == best_hdr_chain[
nHeight - 1]) {
5506 pindex = best_hdr_chain[
nHeight];
5508 assert((pindex ==
nullptr) == (pindexPar == best_hdr_chain.
Tip()));
5520 assert(nNodes == forward.size() + best_hdr_chain.
Height() + 1);
5527 return strprintf(
"Chainstate [%s] @ height %d (%s)",
5532 bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size)
5545 LogPrintf(
"[%s] resized coinsdb cache to %.1f MiB\n",
5546 this->
ToString(), coinsdb_size * (1.0 / 1024 / 1024));
5547 LogPrintf(
"[%s] resized coinstip cache to %.1f MiB\n",
5548 this->
ToString(), coinstip_size * (1.0 / 1024 / 1024));
5553 if (coinstip_size > old_coinstip_size) {
5566 if (pindex ==
nullptr)
5570 LogWarning(
"Internal bug detected: block %d has unset m_chain_tx_count (%s %s). Please report this issue here: %s\n",
5575 int64_t nNow = time(
nullptr);
5591 if (m_active_chainstate && m_active_chainstate->m_from_snapshot_blockhash) {
5593 return m_active_chainstate->m_from_snapshot_blockhash;
5595 return std::nullopt;
5601 std::vector<Chainstate*>
out;
5603 for (
Chainstate*
cs : {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
5613 assert(!m_ibd_chainstate);
5614 assert(!m_active_chainstate);
5616 m_ibd_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman, *
this);
5617 m_active_chainstate = m_ibd_chainstate.get();
5618 return *m_active_chainstate;
5630 bool existed = fs::remove(base_blockhash_path);
5632 LogPrintf(
"[snapshot] snapshot chainstate dir being removed lacks %s file\n",
5635 }
catch (
const fs::filesystem_error& e) {
5636 LogPrintf(
"[snapshot] failed to remove file %s: %s\n",
5642 LogPrintf(
"Removing leveldb dir at %s\n", path_str);
5646 const bool destroyed =
DestroyDB(path_str);
5649 LogPrintf(
"error: leveldb DestroyDB call failed on %s\n", path_str);
5677 if (!
GetParams().AssumeutxoForBlockhash(base_blockhash).has_value()) {
5679 std::string heights_formatted =
util::Join(available_heights,
", ", [&](
const auto& i) {
return util::ToString(i); });
5680 return util::Error{
strprintf(
Untranslated(
"assumeutxo block hash in snapshot metadata not recognized (hash: %s). The following snapshot heights are available: %s"),
5682 heights_formatted)};
5686 if (!snapshot_start_block) {
5687 return util::Error{
strprintf(
Untranslated(
"The base block header (%s) must appear in the headers chain. Make sure all headers are syncing, and call loadtxoutset again"),
5692 if (start_block_invalid) {
5696 if (!m_best_header || m_best_header->GetAncestor(snapshot_start_block->nHeight) != snapshot_start_block) {
5697 return util::Error{
Untranslated(
"A forked headers-chain with more work than the chain with the snapshot base block header exists. Please proceed to sync without AssumeUtxo.")};
5700 auto mempool{m_active_chainstate->GetMempool()};
5701 if (mempool && mempool->size() > 0) {
5706 int64_t current_coinsdb_cache_size{0};
5707 int64_t current_coinstip_cache_size{0};
5715 static constexpr
double IBD_CACHE_PERC = 0.01;
5716 static constexpr
double SNAPSHOT_CACHE_PERC = 0.99;
5728 current_coinsdb_cache_size = this->
ActiveChainstate().m_coinsdb_cache_size_bytes;
5729 current_coinstip_cache_size = this->
ActiveChainstate().m_coinstip_cache_size_bytes;
5734 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
5735 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
5739 return std::make_unique<Chainstate>(
5740 nullptr,
m_blockman, *
this, base_blockhash));
5744 snapshot_chainstate->InitCoinsDB(
5745 static_cast<size_t>(current_coinsdb_cache_size * SNAPSHOT_CACHE_PERC),
5746 in_memory,
false,
"chainstate");
5747 snapshot_chainstate->InitCoinsCache(
5748 static_cast<size_t>(current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
5752 this->MaybeRebalanceCaches();
5760 snapshot_chainstate.reset();
5764 "Manually remove it before restarting.\n"),
fs::PathToString(*snapshot_datadir)));
5781 return cleanup_bad_snapshot(
Untranslated(
"work does not exceed active chainstate"));
5787 return cleanup_bad_snapshot(
Untranslated(
"could not write base blockhash"));
5791 assert(!m_snapshot_chainstate);
5792 m_snapshot_chainstate.swap(snapshot_chainstate);
5793 const bool chaintip_loaded = m_snapshot_chainstate->LoadChainTip();
5798 Assert(m_active_chainstate->m_mempool->size() == 0);
5799 Assert(!m_snapshot_chainstate->m_mempool);
5800 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
5801 m_active_chainstate->m_mempool =
nullptr;
5802 m_active_chainstate = m_snapshot_chainstate.get();
5805 LogPrintf(
"[snapshot] successfully activated snapshot %s\n", base_blockhash.
ToString());
5807 m_snapshot_chainstate->CoinsTip().DynamicMemoryUsage() / (1000 * 1000));
5809 this->MaybeRebalanceCaches();
5810 return snapshot_start_block;
5817 snapshot_loaded ?
"saving snapshot chainstate" :
"flushing coins cache",
5821 coins_cache.
Flush();
5826 const char*
what() const noexcept
override 5828 return "ComputeUTXOStats interrupted.";
5850 if (!snapshot_start_block) {
5854 base_blockhash.ToString())};
5857 int base_height = snapshot_start_block->
nHeight;
5860 if (!maybe_au_data) {
5862 "(%d) - refusing to load snapshot"), base_height)};
5877 LogPrintf(
"[snapshot] loading %d coins from snapshot %s\n", coins_left, base_blockhash.ToString());
5878 int64_t coins_processed{0};
5880 while (coins_left > 0) {
5884 size_t coins_per_txid{0};
5887 if (coins_per_txid > coins_left) {
5891 for (
size_t i = 0; i < coins_per_txid; i++) {
5895 outpoint.
hash = txid;
5897 if (coin.nHeight > base_height ||
5898 outpoint.
n >= std::numeric_limits<decltype(outpoint.
n)>::max()
5901 coins_count - coins_left)};
5905 coins_count - coins_left)};
5912 if (coins_processed % 1000000 == 0) {
5913 LogPrintf(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)\n",
5915 static_cast<float>(coins_processed) * 100 / static_cast<float>(coins_count),
5923 if (coins_processed % 120000 == 0) {
5929 return snapshot_chainstate.GetCoinsCacheSizeState());
5942 }
catch (
const std::ios_base::failure&) {
5955 bool out_of_coins{
false};
5957 std::byte left_over_byte;
5958 coins_file >> left_over_byte;
5959 }
catch (
const std::ios_base::failure&) {
5961 out_of_coins =
true;
5963 if (!out_of_coins) {
5968 LogPrintf(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s\n",
5971 base_blockhash.ToString());
5982 std::optional<CCoinsStats> maybe_stats;
5990 if (!maybe_stats.has_value()) {
6011 constexpr
int AFTER_GENESIS_START{1};
6013 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height(); ++i) {
6014 index = snapshot_chainstate.
m_chain[i];
6031 assert(index == snapshot_start_block);
6035 LogPrintf(
"[snapshot] validated snapshot (%.2f MB)\n",
6058 !this->
IsUsable(m_snapshot_chainstate.get()) ||
6059 !this->
IsUsable(m_ibd_chainstate.get()) ||
6060 !m_ibd_chainstate->m_chain.Tip()) {
6066 const int snapshot_base_height = *
Assert(this->GetSnapshotBaseHeight());
6069 if (index_new.
nHeight < snapshot_base_height) {
6079 "%s failed to validate the -assumeutxo snapshot state. " 6080 "This indicates a hardware problem, or a bug in the software, or a " 6081 "bad software modification that allowed an invalid snapshot to be " 6082 "loaded. As a result of this, the node will shut down and stop using any " 6083 "state that was built on the snapshot, resetting the chain height " 6084 "from %d to %d. On the next " 6085 "restart, the node will resume syncing from %d " 6086 "without using any snapshot data. " 6087 "Please report this incident to %s, including how you obtained the snapshot. " 6088 "The invalid snapshot chainstate will be left on disk in case it is " 6089 "helpful in diagnosing the issue that caused this error."),
6094 LogError(
"[snapshot] deleting snapshot, reverting to validated chain, and stopping node\n");
6096 m_active_chainstate = m_ibd_chainstate.get();
6097 m_snapshot_chainstate->m_disabled =
true;
6101 auto rename_result = m_snapshot_chainstate->InvalidateCoinsDBOnDisk();
6102 if (!rename_result) {
6110 LogPrintf(
"[snapshot] supposed base block %s does not match the " 6111 "snapshot base block %s (height %d). Snapshot is not valid.\n",
6112 index_new.
ToString(), snapshot_blockhash.
ToString(), snapshot_base_height);
6113 handle_invalid_snapshot();
6119 int curr_height = m_ibd_chainstate->m_chain.Height();
6121 assert(snapshot_base_height == curr_height);
6126 CCoinsViewDB& ibd_coins_db = m_ibd_chainstate->CoinsDB();
6127 m_ibd_chainstate->ForceFlushStateToDisk();
6130 if (!maybe_au_data) {
6131 LogPrintf(
"[snapshot] assumeutxo data not found for height " 6132 "(%d) - refusing to validate snapshot\n", curr_height);
6133 handle_invalid_snapshot();
6138 std::optional<CCoinsStats> maybe_ibd_stats;
6139 LogPrintf(
"[snapshot] computing UTXO stats for background chainstate to validate " 6140 "snapshot - this could take a few minutes\n");
6143 CoinStatsHashType::HASH_SERIALIZED,
6152 if (!maybe_ibd_stats) {
6153 LogPrintf(
"[snapshot] failed to generate stats for validation coins db\n");
6157 handle_invalid_snapshot();
6160 const auto& ibd_stats = *maybe_ibd_stats;
6169 LogPrintf(
"[snapshot] hash mismatch: actual=%s, expected=%s\n",
6170 ibd_stats.hashSerialized.ToString(),
6172 handle_invalid_snapshot();
6176 LogPrintf(
"[snapshot] snapshot beginning at %s has been fully validated\n",
6179 m_ibd_chainstate->m_disabled =
true;
6180 this->MaybeRebalanceCaches();
6188 assert(m_active_chainstate);
6189 return *m_active_chainstate;
6195 return m_snapshot_chainstate && m_active_chainstate == m_snapshot_chainstate.get();
6198 void ChainstateManager::MaybeRebalanceCaches()
6201 bool ibd_usable = this->
IsUsable(m_ibd_chainstate.get());
6202 bool snapshot_usable = this->
IsUsable(m_snapshot_chainstate.get());
6203 assert(ibd_usable || snapshot_usable);
6205 if (ibd_usable && !snapshot_usable) {
6210 else if (snapshot_usable && !ibd_usable) {
6212 LogPrintf(
"[snapshot] allocating all cache to the snapshot chainstate\n");
6216 else if (ibd_usable && snapshot_usable) {
6221 m_ibd_chainstate->ResizeCoinsCaches(
6223 m_snapshot_chainstate->ResizeCoinsCaches(
6226 m_snapshot_chainstate->ResizeCoinsCaches(
6228 m_ibd_chainstate->ResizeCoinsCaches(
6234 void ChainstateManager::ResetChainstates()
6236 m_ibd_chainstate.reset();
6237 m_snapshot_chainstate.reset();
6238 m_active_chainstate =
nullptr;
6248 if (!opts.check_block_index.has_value()) opts.
check_block_index = opts.chainparams.DefaultConsistencyChecks();
6249 if (!opts.minimum_chain_work.has_value()) opts.minimum_chain_work =
UintToArith256(opts.chainparams.GetConsensus().nMinimumChainWork);
6250 if (!opts.assumed_valid_block.has_value()) opts.assumed_valid_block = opts.chainparams.GetConsensus().defaultAssumeValid;
6251 return std::move(opts);
6256 m_interrupt{interrupt},
6257 m_options{
Flatten(std::move(options))},
6258 m_blockman{interrupt, std::move(blockman_options)},
6270 bool ChainstateManager::DetectSnapshotChainstate()
6272 assert(!m_snapshot_chainstate);
6278 if (!base_blockhash) {
6281 LogPrintf(
"[snapshot] detected active snapshot chainstate (%s) - loading\n",
6284 this->ActivateExistingSnapshot(*base_blockhash);
6290 assert(!m_snapshot_chainstate);
6291 m_snapshot_chainstate =
6292 std::make_unique<Chainstate>(
nullptr,
m_blockman, *
this, base_blockhash);
6293 LogPrintf(
"[snapshot] switching active chainstate to %s\n", m_snapshot_chainstate->ToString());
6296 Assert(m_active_chainstate->m_mempool->size() == 0);
6297 Assert(!m_snapshot_chainstate->m_mempool);
6298 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
6299 m_active_chainstate->m_mempool =
nullptr;
6300 m_active_chainstate = m_snapshot_chainstate.get();
6301 return *m_snapshot_chainstate;
6306 return (block_index.
nHeight==91842 && block_index.
GetBlockHash() ==
uint256{
"00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec"}) ||
6307 (block_index.
nHeight==91880 && block_index.
GetBlockHash() ==
uint256{
"00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721"});
6312 return (block_index.
nHeight==91722 && block_index.
GetBlockHash() ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
6313 (block_index.
nHeight==91812 && block_index.
GetBlockHash() ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"});
6320 assert(
cs.m_from_snapshot_blockhash);
6321 auto storage_path_maybe =
cs.CoinsDB().StoragePath();
6323 assert(storage_path_maybe);
6324 return *storage_path_maybe;
6334 auto invalid_path = snapshot_datadir +
"_INVALID";
6337 LogPrintf(
"[snapshot] renaming snapshot datadir %s to %s\n", dbpath, target);
6343 fs::rename(snapshot_datadir, invalid_path);
6344 }
catch (
const fs::filesystem_error& e) {
6348 LogPrintf(
"%s: error renaming file '%s' -> '%s': %s\n",
6349 __func__, src_str, dest_str, e.what());
6351 "Rename of '%s' -> '%s' failed. " 6352 "You should resolve this by manually moving or deleting the invalid " 6353 "snapshot directory %s, otherwise you will encounter the same error again " 6354 "on the next startup."),
6355 src_str, dest_str, src_str)};
6360 bool ChainstateManager::DeleteSnapshotChainstate()
6363 Assert(m_snapshot_chainstate);
6364 Assert(m_ibd_chainstate);
6368 LogPrintf(
"Deletion of %s failed. Please remove it manually to continue reindexing.\n",
6372 m_active_chainstate = m_ibd_chainstate.get();
6373 m_active_chainstate->m_mempool = m_snapshot_chainstate->m_mempool;
6374 m_snapshot_chainstate.reset();
6388 const CBlockIndex* ChainstateManager::GetSnapshotBaseBlock()
const 6390 return m_active_chainstate ? m_active_chainstate->SnapshotBase() :
nullptr;
6393 std::optional<int> ChainstateManager::GetSnapshotBaseHeight()
const 6395 const CBlockIndex* base = this->GetSnapshotBaseBlock();
6396 return base ? std::make_optional(base->
nHeight) :
std::nullopt;
6399 bool ChainstateManager::ValidatedSnapshotCleanup()
6403 if (!(chainstate && chainstate->HasCoinsViews())) {
6406 return chainstate->CoinsDB().StoragePath();
6408 std::optional<fs::path> ibd_chainstate_path_maybe = get_storage_path(m_ibd_chainstate);
6409 std::optional<fs::path> snapshot_chainstate_path_maybe = get_storage_path(m_snapshot_chainstate);
6418 if (!ibd_chainstate_path_maybe || !snapshot_chainstate_path_maybe) {
6419 LogPrintf(
"[snapshot] snapshot chainstate cleanup cannot happen with " 6420 "in-memory chainstates. You are testing, right?\n");
6424 const auto& snapshot_chainstate_path = *snapshot_chainstate_path_maybe;
6425 const auto& ibd_chainstate_path = *ibd_chainstate_path_maybe;
6433 this->ResetChainstates();
6438 LogPrintf(
"[snapshot] deleting background chainstate directory (now unnecessary) (%s)\n",
6441 fs::path tmp_old{ibd_chainstate_path +
"_todelete"};
6443 auto rename_failed_abort = [
this](
6446 const fs::filesystem_error& err) {
6447 LogError(
"[snapshot] Error renaming path (%s) -> (%s): %s\n",
6450 "Rename of '%s' -> '%s' failed. " 6451 "Cannot clean up the background chainstate leveldb directory."),
6456 fs::rename(ibd_chainstate_path, tmp_old);
6457 }
catch (
const fs::filesystem_error& e) {
6458 rename_failed_abort(ibd_chainstate_path, tmp_old, e);
6462 LogPrintf(
"[snapshot] moving snapshot chainstate (%s) to " 6463 "default chainstate directory (%s)\n",
6467 fs::rename(snapshot_chainstate_path, ibd_chainstate_path);
6468 }
catch (
const fs::filesystem_error& e) {
6469 rename_failed_abort(snapshot_chainstate_path, ibd_chainstate_path, e);
6476 LogPrintf(
"Deletion of %s failed. Please remove it manually, as the " 6477 "directory is now unnecessary.\n",
6480 LogPrintf(
"[snapshot] deleted background chainstate directory (%s)\n",
6486 Chainstate& ChainstateManager::GetChainstateForIndexing()
6491 return (this->
GetAll().size() > 1) ? *m_ibd_chainstate : *m_active_chainstate;
6494 std::pair<int, int> ChainstateManager::GetPruneRange(
const Chainstate& chainstate,
int last_height_can_prune)
6501 if (this->
GetAll().size() > 1 && m_snapshot_chainstate.get() == &chainstate) {
6504 prune_start = *
Assert(GetSnapshotBaseHeight()) + 1;
6507 int max_prune = std::max<int>(
6516 int prune_end = std::min(last_height_can_prune, max_prune);
6518 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
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
CoinsViewOptions coins_view
static unsigned int GetBlockScriptFlags(const CBlockIndex &block_index, const ChainstateManager &chainman)
std::string GetHex() const
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. ...
std::optional< std::string > EntriesAndTxidsDisjoint(const CTxMemPool::setEntries &ancestors, const std::set< Txid > &direct_conflicts, const uint256 &txid)
Check the intersection between two sets of transactions (a set of mempool entries and a set of txids)...
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.
virtual void warningUnset(Warning id)
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 ReplayBlocks()
Replay blocks that aren't fully applied to the database.
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.
bool HaveNumChainTxs() const
Check whether this block and all previous blocks back to the genesis block or an assumeutxo snapshot ...
SignatureCache m_signature_cache
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)
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 MempoolAcceptResult FeeFailure(TxValidationState state, CFeeRate effective_feerate, const std::vector< Wtxid > &wtxids_fee_calculations)
static void FlushSnapshotToDisk(CCoinsViewCache &coins_cache, bool snapshot_loaded)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void UpdateBlockInfo(const CBlock &block, unsigned int nHeight, const FlatFilePos &pos)
Update blockfile info while processing a block during reindex.
bool Error(const std::string &reject_reason)
SynchronizationState
Current sync state passed to tip changed callbacks.
void BlockConnected(ChainstateRole, const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
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
Convenience class for initializing and passing the script execution cache and signature cache...
#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.
ValidationSignals * signals
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::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...
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances...
bool exists(const GenTxid >xid) const
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.
uint64_t ReadCompactSize(Stream &is, bool range_check=true)
Decode a CompactSize-encoded variable-length integer.
void BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
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.
bool CheckBlockDataAvailability(const CBlockIndex &upper_block LIFETIMEBOUND, const CBlockIndex &lower_block LIFETIMEBOUND) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetFirstBlock(const CBlockIndex &upper_block LIFETIMEBOUND, uint32_t status_mask, const CBlockIndex *lower_block=nullptr) const EXCLUSIVE_LOCKS_REQUIRED(boo m_have_pruned)
Check if all blocks in the [upper_block, lower_block] range have data available.
bool 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.
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...
size_t GetSerializeSize(const T &t)
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
std::unique_ptr< CoinsViews > m_coins_views
Manages the UTXO set, which is a reflection of the contents of m_chain.
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.
std::optional< std::pair< DiagramCheckError, std::string > > ImprovesFeerateDiagram(CTxMemPool &pool, const CTxMemPool::setEntries &direct_conflicts, const CTxMemPool::setEntries &all_conflicts, CAmount replacement_fees, int64_t replacement_vsize)
The replacement transaction must improve the feerate diagram of the mempool.
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...
stage after last reached validness failed
constexpr const std::byte * begin() const
bool CheckFinalTxAtTip(const CBlockIndex &active_chain_tip, const CTransaction &tx)
static constexpr size_t MINIMUM_WITNESS_COMMITMENT
Minimum size of a witness commitment structure.
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...
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...
bool WriteUndoDataForBlock(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos SaveBlockToDisk(const CBlock &block, int nHeight)
Store block on disk and update block file statistics.
Wrapper around an AutoFile& 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
MempoolAcceptResult AcceptToMemoryPool(Chainstate &active_chainstate, const CTransactionRef &tx, int64_t accept_time, bool bypass_limits, bool test_accept)
Try to add a transaction to the mempool.
void Clear() EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
bool contains(const Element &e, const bool erase) const
contains iterates through the hash locations for a given element and checks to see if it is present...
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
void LoadExternalBlockFile(AutoFile &file_in, FlatFilePos *dbp=nullptr, std::multimap< uint256, FlatFilePos > *blocks_with_unknown_parent=nullptr)
Import blocks from an external file.
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 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)
static const unsigned int MAX_DISCONNECTED_TX_POOL_BYTES
Maximum bytes for transactions to store for processing during reorg.
util::Result< void > PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
int64_t nTime
UNIX timestamp of last known number of transactions.
void ChainStateFlushed(ChainstateRole, const CBlockLocator &)
static const int64_t MAX_BLOCK_SIGOPS_COST
The maximum allowed number of signature check operations in a block (network rule) ...
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
util::Result< CBlockIndex * > ActivateSnapshot(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Construct and activate a Chainstate on the basis of UTXO snapshot data.
bool TestBlockValidity(BlockValidationState &state, const CChainParams &chainparams, Chainstate &chainstate, const CBlock &block, CBlockIndex *pindexPrev, bool fCheckPOW, bool fCheckMerkleRoot)
Check a block is completely valid from start to finish (only works on top of our current best block) ...
Non-refcounted RAII wrapper for FILE*.
static constexpr unsigned int STANDARD_SCRIPT_VERIFY_FLAGS
Standard script verification flags that standard transactions will comply with.
void SetTip(CBlockIndex &block)
Set/initialize a chain with a given tip.
std::optional< std::pair< std::string, CTransactionRef > > SingleTRUCChecks(const CTransactionRef &ptx, const CTxMemPool::setEntries &mempool_ancestors, const std::set< Txid > &direct_conflicts, int64_t vsize)
Must be called for every transaction, even if not TRUC.
static void LimitMempoolSize(CTxMemPool &pool, CCoinsViewCache &coins_cache) EXCLUSIVE_LOCKS_REQUIRED(
virtual void fatalError(const bilingual_str &message)
The fatal error notification is sent to notify the user when an error occurs in kernel code that can'...
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 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)
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.
uint256 g_best_block
Used to notify getblocktemplate RPC of new tips.
arith_uint256 UintToArith256(const uint256 &a)
VersionBitsCache m_versionbitscache
Track versionbit status.
const Coin & AccessByTxid(const CCoinsViewCache &view, const Txid &txid)
Utility function to find any unspent output with a given txid.
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.
std::function< void()> snapshot_download_completed
Function to restart active indexes; set dynamically to avoid a circular dependency on base/index...
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.
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)
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds
void PruneAndFlush()
Prune blockfiles from the disk if necessary and then flush chainstate changes if we pruned...
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:
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.
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
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).
std::vector< int > GetAvailableSnapshotHeights() const
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) ...
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.
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)
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.
static decltype(CTransaction::version) constexpr TRUC_VERSION
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="")
static void LimitValidationInterfaceQueue(ValidationSignals &signals) LOCKS_EXCLUDED(cs_main)
std::optional< LockPoints > CalculateLockPointsAtTip(CBlockIndex *tip, const CCoinsView &coins_view, const CTransaction &tx)
arith_uint256 CalculateClaimedHeadersWork(const std::vector< CBlockHeader > &headers)
Return the sum of the claimed work on a given set of headers.
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.
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...
uint64_t m_chain_tx_count
(memory only) Number of transactions in the chain up to and including this block. ...
static constexpr int PRUNE_LOCK_BUFFER
The number of blocks to keep below the deepest prune lock.
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)
std::string ToString() const
ValidationCache(size_t script_execution_cache_bytes, size_t signature_cache_bytes)
the block failed to meet one of our checkpoints
static bool DeleteCoinsDBFromDisk(const fs::path db_path, bool is_snapshot) EXCLUSIVE_LOCKS_REQUIRED(
void ActiveTipChange(const CBlockIndex &, bool)
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
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)
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(CCheckQueue< CScriptCheck > & GetCheckQueue()
When starting up, search the datadir for a chainstate based on a UTXO snapshot that is in the process...
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...
std::string ToString() const
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
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)
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,...)
virtual std::vector< uint256 > GetHeadBlocks() const
Retrieve the range of blocks that may have been only partially written.
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.
std::atomic_bool m_blockfiles_indexed
Whether all blockfiles have been added to the block tree database.
An output of a transaction.
virtual void warningSet(Warning id, const bilingual_str &message)
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.
RecursiveMutex & GetMutex() const LOCK_RETURNED(
Alias for cs_main.
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...
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...
fails some policy, but might be acceptable if submitted in a (different) package
DisconnectedBlockTransactions.
Chainstate &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(std::vector< Chainstate * GetAll)()
Instantiate a new chainstate.
Validation result for a transaction evaluated by MemPoolAccept (single or package).
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)
std::string FormatFullVersion()
void BlockConnected(CBlockIndex *pindex, std::shared_ptr< const CBlock > pblock)
int32_t ComputeBlockVersion(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Determine what nVersion a new block should use.
std::chrono::seconds PowTargetSpacing() const
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...
ValidationCache m_validation_cache
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)
void insert(Element e)
insert loops at most depth_limit times trying to insert a hash at various locations in the table via ...
static bool CheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW=true)
int64_t DifficultyAdjustmentInterval() const
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
std::string get_filesystem_error_message(const fs::filesystem_error &e)
invalid by consensus rules (excluding any below reasons)
CSHA256 ScriptExecutionCacheHasher() const
Return a copy of the pre-initialized hasher.
static time_point now() noexcept
Return current system time or mocked time, if set.
const CTransaction * ptxTo
bool IsWellFormedPackage(const Package &txns, PackageValidationState &state, bool require_sorted)
Context-free package policy checks:
const arith_uint256 & MinimumChainWork() const
bool enforce_BIP94
Enfore BIP94 timewarp attack mitigation.
#define EXCLUSIVE_LOCKS_REQUIRED(...)
std::vector< CTransactionRef > vtx
const ChainTxData & TxData() const
kernel::Notifications & m_notifications
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 ...
#define LogDebug(category,...)
static bool ContextualCheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, BlockManager &blockman, const ChainstateManager &chainman, const CBlockIndex *pindexPrev) EXCLUSIVE_LOCKS_REQUIRED(
Context-dependent validity checks.
bool CheckInputScripts(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, unsigned int flags, bool cacheSigStore, bool cacheFullScriptStore, PrecomputedTransactionData &txdata, ValidationCache &validation_cache, std::vector< CScriptCheck > *pvChecks=nullptr) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Check whether all of this transaction's input scripts succeed.
constexpr const unsigned char * data() const
#define LOCKS_EXCLUDED(...)
static SynchronizationState GetSynchronizationState(bool init, bool blockfiles_indexed)
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
unsigned int GetLegacySigOpCount(const CTransaction &tx)
Auxiliary functions for transaction validation (ideally should not be exposed)
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.
const fs::path blocks_dir
uint64_t m_chain_tx_count
Used to populate the m_chain_tx_count value, which is used during BlockManager::LoadBlockIndex().
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.
int worker_threads_num
Number of script check worker threads. Zero means no parallel verification.
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) ...
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept, const std::optional< CFeeRate > &client_maxfeerate)
Validate (and maybe submit) a package to the mempool.
uint256 GetRandHash() noexcept
Generate a random uint256.
static bool CheckInputsFromMempoolAndCache(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &view, const CTxMemPool &pool, unsigned int flags, PrecomputedTransactionData &txdata, CCoinsViewCache &coins_tip, ValidationCache &validation_cache) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Checks to avoid mempool polluting consensus critical paths since cached signature and script validity...
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())
bool RaiseValidity(enum BlockStatus nUpTo) EXCLUSIVE_LOCKS_REQUIRED(
Raise the validity level of this block index entry.
std::optional< std::string > PackageTRUCChecks(const CTransactionRef &ptx, int64_t vsize, const Package &package, const CTxMemPool::setEntries &mempool_ancestors)
Must be called for every transaction that is submitted within a package, even if not TRUC...
Application-specific storage settings.
Fee rate in satoshis per kilovirtualbyte: CAmount / kvB.
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).
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
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.
unsigned char * UCharCast(char *c)
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) ...
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)
size_t script_execution_cache_bytes
bool FatalError(Notifications ¬ifications, BlockValidationState &state, const bilingual_str &message)
static constexpr int64_t MAX_FUTURE_BLOCK_TIME
Maximum amount of time that a block timestamp is allowed to exceed the current time before the block ...
bool NotifyHeaderTip() LOCKS_EXCLUDED(GetMutex())
std::list< CTransactionRef > take()
Clear all data structures and return the list of transactions.
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
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...
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)
CuckooCache::cache< uint256, SignatureCacheHasher > m_script_execution_cache
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.
uint64_t tx_count
total number of transactions between genesis and that timestamp
int64_t GetTime()
DEPRECATED, see GetTime.
std::vector< CTxUndo > vtxundo
static int64_t GetBlockWeight(const CBlock &block)
std::optional< int32_t > check_block_index
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)
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
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.
std::vector< CTransactionRef > AddTransactionsFromBlock(const std::vector< CTransactionRef > &vtx)
Add transactions from the block, iterating through vtx in reverse order.
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 ...
bilingual_str _(ConstevalStringLiteral str)
Translation function.
void removeForBlock(const std::vector< CTransactionRef > &vtx, unsigned int nBlockHeight) EXCLUSIVE_LOCKS_REQUIRED(cs)
Called when a block is connected.
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)
SignatureCache * m_signature_cache
bool Sync()
Push the modifications applied to this cache to its base while retaining the contents of this cache (...
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 ...
static GenTxid Txid(const uint256 &hash)
static MempoolAcceptResult MempoolTxDifferentWitness(const uint256 &other_wtxid)
CTxMemPool * GetMempool()
const Wtxid & GetWitnessHash() const LIFETIMEBOUND
std::string ToString(const T &t)
Locale-independent version of std::to_string.
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.
std::shared_ptr< Chain::Notifications > m_notifications
const Txid & GetHash() const LIFETIMEBOUND
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 MempoolAcceptResult Success(std::list< CTransactionRef > &&replaced_txns, int64_t vsize, CAmount fees, CFeeRate effective_feerate, const std::vector< Wtxid > &wtxids_fee_calculations)
void BlockChecked(const CBlock &, const BlockValidationState &)
static constexpr TransactionSerParams TX_WITH_WITNESS
static constexpr TransactionSerParams TX_NO_WITNESS
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...
static constexpr int64_t MAX_TIMEWARP
Maximum number of seconds that the timestamp of the first block of a difficulty adjustment period is ...
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...