6#include <bitcoin-build-config.h>
103 "level 0 reads the blocks from disk",
104 "level 1 verifies block validity",
105 "level 2 verifies undo data",
106 "level 3 checks disconnection of tip blocks",
107 "level 4 tries to reconnect the blocks",
108 "each level includes the checks of the previous levels",
146 std::vector<CScriptCheck>* pvChecks =
nullptr)
159 const int nBlockHeight = active_chain_tip.nHeight + 1;
166 const int64_t nBlockTime{active_chain_tip.GetMedianTimePast()};
168 return IsFinalTx(tx, nBlockHeight, nBlockTime);
182std::optional<std::vector<int>> CalculatePrevHeights(
187 std::vector<int> prev_heights;
188 prev_heights.resize(tx.
vin.size());
189 for (
size_t i = 0; i < tx.
vin.size(); ++i) {
190 if (
auto coin{coins.
GetCoin(tx.
vin[i].prevout)}) {
195 LogInfo(
"ERROR: %s: Missing input %d in transaction \'%s\'\n", __func__, i, tx.
GetHash().
GetHex());
210 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
211 if (!prev_heights.has_value())
return std::nullopt;
214 next_tip.
pprev = tip;
232 int max_input_height{0};
233 for (
const int height : prev_heights.value()) {
235 if (height != next_tip.
nHeight) {
236 max_input_height = std::max(max_input_height, height);
271 int expired = pool.Expire(GetTime<std::chrono::seconds>() - pool.m_opts.expiry);
276 std::vector<COutPoint> vNoSpendsRemaining;
277 pool.TrimToSize(pool.m_opts.max_size_bytes, &vNoSpendsRemaining);
278 for (
const COutPoint& removed : vNoSpendsRemaining)
279 coins_cache.Uncache(removed);
285 if (active_chainstate.m_chainman.IsInitialBlockDownload()) {
290 if (active_chainstate.m_chain.Height() < active_chainstate.m_chainman.m_best_header->nHeight - 1) {
304 std::vector<Txid> vHashUpdate;
311 const auto queuedTx = disconnectpool.
take();
312 auto it = queuedTx.rbegin();
313 while (it != queuedTx.rend()) {
315 if (!fAddToMempool || (*it)->IsCoinBase() ||
317 true,
false).m_result_type !=
323 vHashUpdate.push_back((*it)->GetHash());
364 it->UpdateLockPoints(*new_lock_points);
371 if (it->GetSpendsCoinbase()) {
377 if (coin.IsCoinBase() && mempool_spend_height - coin.nHeight <
COINBASE_MATURITY) {
413 if (coin.
IsSpent())
return false;
443 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
444 m_active_chainstate(active_chainstate)
452 const int64_t m_accept_time;
453 const bool m_bypass_limits;
461 std::vector<COutPoint>& m_coins_to_uncache;
463 const bool m_test_accept;
467 const bool m_allow_replacement;
469 const bool m_allow_sibling_eviction;
472 const bool m_package_submission;
476 const bool m_package_feerates;
481 const std::optional<CFeeRate> m_client_maxfeerate;
484 static ATMPArgs SingleAccept(
const CChainParams& chainparams, int64_t accept_time,
485 bool bypass_limits, std::vector<COutPoint>& coins_to_uncache,
487 return ATMPArgs{ chainparams,
501 static ATMPArgs PackageTestAccept(
const CChainParams& chainparams, int64_t accept_time,
502 std::vector<COutPoint>& coins_to_uncache) {
503 return ATMPArgs{ chainparams,
517 static ATMPArgs PackageChildWithParents(
const CChainParams& chainparams, int64_t accept_time,
518 std::vector<COutPoint>& coins_to_uncache,
const std::optional<CFeeRate>& client_maxfeerate) {
519 return ATMPArgs{ chainparams,
533 static ATMPArgs SingleInPackageAccept(
const ATMPArgs& package_args) {
534 return ATMPArgs{ package_args.m_chainparams,
535 package_args.m_accept_time,
537 package_args.m_coins_to_uncache,
538 package_args.m_test_accept,
543 package_args.m_client_maxfeerate,
553 std::vector<COutPoint>& coins_to_uncache,
555 bool allow_replacement,
556 bool allow_sibling_eviction,
557 bool package_submission,
558 bool package_feerates,
559 std::optional<CFeeRate> client_maxfeerate)
560 : m_chainparams{chainparams},
561 m_accept_time{accept_time},
562 m_bypass_limits{bypass_limits},
563 m_coins_to_uncache{coins_to_uncache},
564 m_test_accept{test_accept},
565 m_allow_replacement{allow_replacement},
566 m_allow_sibling_eviction{allow_sibling_eviction},
567 m_package_submission{package_submission},
568 m_package_feerates{package_feerates},
569 m_client_maxfeerate{client_maxfeerate}
573 if (m_package_feerates) {
574 Assume(m_package_submission);
575 Assume(!m_allow_sibling_eviction);
577 if (m_allow_sibling_eviction)
Assume(m_allow_replacement);
588 ClearSubPackageState();
601 ClearSubPackageState();
629 explicit Workspace(
const CTransactionRef& ptx) : m_ptx(ptx), m_hash(ptx->GetHash()) {}
632 std::set<Txid> m_conflicts;
637 std::vector<CTxMemPoolEntry::CTxMemPoolEntryRef> m_parents;
642 bool m_sibling_eviction{
false};
677 bool PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
678 std::vector<Workspace>& workspaces,
699 std::map<Wtxid, MempoolAcceptResult>& results)
707 CAmount mempoolRejectFee = m_pool.GetMinFee().GetFee(package_size);
708 if (mempoolRejectFee > 0 && package_fee < mempoolRejectFee) {
712 if (package_fee < m_pool.m_opts.min_relay_feerate.GetFee(package_size)) {
714 strprintf(
"%d < %d", package_fee, m_pool.m_opts.min_relay_feerate.GetFee(package_size)));
721 return m_active_chainstate.m_chainman.m_validation_cache;
753 struct SubPackageState {
755 CAmount m_total_modified_fees{0};
757 int64_t m_total_vsize{0};
764 std::list<CTransactionRef> m_replaced_transactions;
766 std::unique_ptr<CTxMemPool::ChangeSet> m_changeset;
771 size_t m_conflicting_size{0};
774 struct SubPackageState m_subpackage;
779 m_subpackage = SubPackageState{};
782 CleanupTemporaryCoins();
786bool MemPoolAccept::PreChecks(ATMPArgs&
args, Workspace& ws)
792 const Txid& hash = ws.m_hash;
795 const int64_t nAcceptTime =
args.m_accept_time;
796 const bool bypass_limits =
args.m_bypass_limits;
797 std::vector<COutPoint>& coins_to_uncache =
args.m_coins_to_uncache;
812 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)) {
830 }
else if (m_pool.exists(tx.
GetHash())) {
840 if (ptxConflicting) {
841 if (!
args.m_allow_replacement) {
845 ws.m_conflicts.insert(ptxConflicting->
GetHash());
849 m_view.SetBackend(m_viewmempool);
855 coins_to_uncache.push_back(txin.
prevout);
861 if (!m_view.HaveCoin(txin.
prevout)) {
876 m_view.GetBestBlock();
881 m_view.SetBackend(m_dummy);
883 assert(m_active_chainstate.m_blockman.LookupBlockIndex(m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
890 const std::optional<LockPoints> lock_points{
CalculateLockPointsAtTip(m_active_chainstate.m_chain.Tip(), m_view, tx)};
913 bool fSpendsCoinbase =
false;
915 const Coin &coin = m_view.AccessCoin(txin.
prevout);
917 fSpendsCoinbase =
true;
924 const uint64_t entry_sequence = bypass_limits ? 0 : m_pool.GetSequence();
925 if (!m_subpackage.m_changeset) {
926 m_subpackage.m_changeset = m_pool.GetChangeSet();
928 ws.m_tx_handle = m_subpackage.m_changeset->StageAddition(ptx, ws.m_base_fees, nAcceptTime, m_active_chainstate.m_chain.Height(), entry_sequence, fSpendsCoinbase, nSigOpsCost, lock_points.value());
931 ws.m_modified_fees = ws.m_tx_handle->GetModifiedFee();
933 ws.m_vsize = ws.m_tx_handle->GetTxSize();
936 if (m_pool.m_opts.require_standard) {
937 if (!
PreCheckEphemeralTx(*ptx, m_pool.m_opts.dust_relay_feerate, ws.m_base_fees, ws.m_modified_fees, state)) {
952 if (!bypass_limits && ws.m_ptx->version !=
TRUC_VERSION && ws.m_modified_fees < m_pool.m_opts.min_relay_feerate.GetFee(ws.m_vsize)) {
956 strprintf(
"%d < %d", ws.m_modified_fees, m_pool.m_opts.min_relay_feerate.GetFee(ws.m_vsize)));
961 if (!bypass_limits && !
args.m_package_feerates && !
CheckFeeRate(ws.m_vsize, ws.m_modified_fees, state))
return false;
963 ws.m_iters_conflicting = m_pool.GetIterSet(ws.m_conflicts);
965 ws.m_parents = m_pool.GetParents(*ws.m_tx_handle);
967 if (!
args.m_bypass_limits) {
969 if (
const auto err{
SingleTRUCChecks(m_pool, ws.m_ptx, ws.m_parents, ws.m_conflicts, ws.m_vsize)}) {
971 if (
args.m_allow_sibling_eviction && err->second !=
nullptr) {
976 ws.m_conflicts.insert(err->second->GetHash());
980 ws.m_iters_conflicting.insert(m_pool.GetIter(err->second->GetHash()).value());
981 ws.m_sibling_eviction =
true;
993 m_subpackage.m_rbf |= !ws.m_conflicts.empty();
997bool MemPoolAccept::ReplacementChecks(Workspace& ws)
1003 const Txid& hash = ws.m_hash;
1006 CFeeRate newFeeRate(ws.m_modified_fees, ws.m_vsize);
1013 strprintf(
"too many potential replacements%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1019 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1020 m_subpackage.m_conflicting_size += it->GetTxSize();
1023 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees, ws.m_modified_fees, ws.m_vsize,
1024 m_pool.m_opts.incremental_relay_feerate, hash)}) {
1027 strprintf(
"insufficient fee%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1031 for (
auto it : all_conflicts) {
1032 m_subpackage.m_changeset->StageRemoval(it);
1036 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1050bool MemPoolAccept::PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
1051 std::vector<Workspace>& workspaces,
1052 const int64_t total_vsize,
1058 assert(std::all_of(txns.cbegin(), txns.cend(), [
this](
const auto& tx)
1059 { return !m_pool.exists(tx->GetHash());}));
1061 assert(txns.size() == workspaces.size());
1064 if (!m_subpackage.m_rbf)
return true;
1079 for (
const auto& ws : workspaces) {
1080 if (!ws.m_parents.empty()) {
1087 for (Workspace& ws : workspaces) {
1089 direct_conflict_iters.merge(ws.m_iters_conflicting);
1092 const auto& parent_ws = workspaces[0];
1093 const auto& child_ws = workspaces[1];
1101 "package RBF failed: too many potential replacements", *err_string);
1105 m_subpackage.m_changeset->StageRemoval(it);
1106 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1107 m_subpackage.m_conflicting_size += it->GetTxSize();
1111 const Txid& child_hash = child_ws.m_ptx->GetHash();
1112 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees,
1113 m_subpackage.m_total_modified_fees,
1114 m_subpackage.m_total_vsize,
1115 m_pool.m_opts.incremental_relay_feerate, child_hash)}) {
1117 "package RBF failed: insufficient anti-DoS fees", *err_string);
1122 const CFeeRate parent_feerate(parent_ws.m_modified_fees, parent_ws.m_vsize);
1123 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1124 if (package_feerate <= parent_feerate) {
1126 "package RBF failed: package feerate is less than or equal to parent feerate",
1127 strprintf(
"package feerate %s <= parent feerate is %s", package_feerate.ToString(), parent_feerate.ToString()));
1131 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1139 "package RBF failed: " + err_tup.value().second,
"");
1142 LogDebug(
BCLog::TXPACKAGES,
"package RBF checks passed: parent %s (wtxid=%s), child %s (wtxid=%s), package hash (%s)\n",
1143 txns.front()->GetHash().ToString(), txns.front()->GetWitnessHash().ToString(),
1144 txns.back()->GetHash().ToString(), txns.back()->GetWitnessHash().ToString(),
1151bool MemPoolAccept::PolicyScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1162 if (!
CheckInputScripts(tx, state, m_view, scriptVerifyFlags,
true,
false, ws.m_precomputed_txdata, GetValidationCache())) {
1174bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1179 const Txid& hash = ws.m_hash;
1199 ws.m_precomputed_txdata, m_active_chainstate.CoinsTip(), GetValidationCache())) {
1200 LogError(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against latest-block but not STANDARD flags %s, %s", hash.
ToString(), state.
ToString());
1207void MemPoolAccept::FinalizeSubpackage(
const ATMPArgs&
args)
1212 if (!m_subpackage.m_changeset->GetRemovals().empty())
Assume(
args.m_allow_replacement);
1216 std::string log_string =
strprintf(
"replacing mempool tx %s (wtxid=%s, fees=%s, vsize=%s). ",
1217 it->GetTx().GetHash().ToString(),
1218 it->GetTx().GetWitnessHash().ToString(),
1221 FeeFrac feerate{m_subpackage.m_total_modified_fees, int32_t(m_subpackage.m_total_vsize)};
1223 const bool replaced_with_tx{m_subpackage.m_changeset->GetTxCount() == 1};
1224 if (replaced_with_tx) {
1225 const CTransaction& tx = m_subpackage.m_changeset->GetAddedTxn(0);
1227 log_string +=
strprintf(
"New tx %s (wtxid=%s, fees=%s, vsize=%s)",
1233 tx_or_package_hash =
GetPackageHash(m_subpackage.m_changeset->GetAddedTxns());
1234 log_string +=
strprintf(
"New package %s with %lu txs, fees=%s, vsize=%s",
1235 tx_or_package_hash.ToString(),
1236 m_subpackage.m_changeset->GetTxCount(),
1243 it->GetTx().GetHash().data(),
1246 std::chrono::duration_cast<std::chrono::duration<std::uint64_t>>(it->GetTime()).count(),
1247 tx_or_package_hash.data(),
1252 m_subpackage.m_replaced_transactions.push_back(it->GetSharedTx());
1254 m_subpackage.m_changeset->Apply();
1255 m_subpackage.m_changeset.reset();
1258bool MemPoolAccept::SubmitPackage(
const ATMPArgs&
args, std::vector<Workspace>& workspaces,
1260 std::map<Wtxid, MempoolAcceptResult>& results)
1266 assert(std::all_of(workspaces.cbegin(), workspaces.cend(), [
this](
const auto& ws) { return !m_pool.exists(ws.m_ptx->GetHash()); }));
1268 bool all_submitted =
true;
1269 FinalizeSubpackage(
args);
1274 for (Workspace& ws : workspaces) {
1275 if (!ConsensusScriptChecks(
args, ws)) {
1279 all_submitted =
false;
1281 strprintf(
"BUG! PolicyScriptChecks succeeded but ConsensusScriptChecks failed: %s",
1282 ws.m_ptx->GetHash().ToString()));
1284 if (!m_subpackage.m_changeset) m_subpackage.m_changeset = m_pool.GetChangeSet();
1285 m_subpackage.m_changeset->StageRemoval(m_pool.GetIter(ws.m_ptx->GetHash()).value());
1288 if (!all_submitted) {
1289 Assume(m_subpackage.m_changeset);
1293 m_subpackage.m_changeset->Apply();
1294 m_subpackage.m_changeset.reset();
1298 std::vector<Wtxid> all_package_wtxids;
1299 all_package_wtxids.reserve(workspaces.size());
1300 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1301 [](
const auto& ws) { return ws.m_ptx->GetWitnessHash(); });
1303 if (!m_subpackage.m_replaced_transactions.empty()) {
1304 LogDebug(
BCLog::MEMPOOL,
"replaced %u mempool transactions with %u new one(s) for %s additional fees, %d delta bytes\n",
1305 m_subpackage.m_replaced_transactions.size(), workspaces.size(),
1306 m_subpackage.m_total_modified_fees - m_subpackage.m_conflicting_fees,
1307 m_subpackage.m_total_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1311 for (Workspace& ws : workspaces) {
1312 auto iter = m_pool.GetIter(ws.m_ptx->GetHash());
1313 Assume(iter.has_value());
1314 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1315 CFeeRate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1316 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1317 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1318 results.emplace(ws.m_ptx->GetWitnessHash(),
1320 ws.m_base_fees, effective_feerate, effective_feerate_wtxids));
1321 if (!m_pool.m_opts.signals)
continue;
1324 ws.m_vsize, (*iter)->GetHeight(),
1325 args.m_bypass_limits,
args.m_package_submission,
1327 m_pool.HasNoInputsOf(tx));
1328 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1330 return all_submitted;
1339 const std::vector<Wtxid> single_wtxid{ws.m_ptx->GetWitnessHash()};
1341 if (!PreChecks(
args, ws)) {
1349 if (m_subpackage.m_rbf && !ReplacementChecks(ws)) {
1358 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1365 if (ws.m_conflicts.size()) {
1366 auto ancestors = m_subpackage.m_changeset->CalculateMemPoolAncestors(ws.m_tx_handle);
1380 m_subpackage.m_total_vsize = ws.m_vsize;
1381 m_subpackage.m_total_modified_fees = ws.m_modified_fees;
1384 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1389 if (m_pool.m_opts.require_standard) {
1391 if (!
CheckEphemeralSpends({ptx}, m_pool.m_opts.dust_relay_feerate, m_pool, ws.m_state, dummy_wtxid)) {
1402 const CFeeRate effective_feerate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1404 if (
args.m_test_accept) {
1406 ws.m_base_fees, effective_feerate, single_wtxid);
1409 FinalizeSubpackage(
args);
1412 if (!
args.m_package_submission && !
args.m_bypass_limits) {
1416 CleanupTemporaryCoins();
1418 if (!m_pool.exists(ws.m_hash)) {
1425 if (m_pool.m_opts.signals) {
1427 auto iter = m_pool.GetIter(tx.
GetHash());
1428 Assume(iter.has_value());
1430 ws.m_vsize, (*iter)->GetHeight(),
1431 args.m_bypass_limits,
args.m_package_submission,
1433 m_pool.HasNoInputsOf(tx));
1434 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1437 if (!m_subpackage.m_replaced_transactions.empty()) {
1438 LogDebug(
BCLog::MEMPOOL,
"replaced %u mempool transactions with 1 new transaction for %s additional fees, %d delta bytes\n",
1439 m_subpackage.m_replaced_transactions.size(),
1440 ws.m_modified_fees - m_subpackage.m_conflicting_fees,
1441 ws.m_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1445 effective_feerate, single_wtxid);
1457 std::vector<Workspace> workspaces{};
1458 workspaces.reserve(txns.size());
1459 std::transform(txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1460 [](
const auto& tx) { return Workspace(tx); });
1461 std::map<Wtxid, MempoolAcceptResult> results;
1464 for (Workspace& ws : workspaces) {
1465 if (!PreChecks(
args, ws)) {
1474 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1492 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1497 for (Workspace& ws : workspaces) {
1498 if (
auto err{
PackageTRUCChecks(m_pool, ws.m_ptx, ws.m_vsize, txns, ws.m_parents)}) {
1513 m_subpackage.m_total_vsize = std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1514 [](int64_t
sum,
auto& ws) { return sum + ws.m_vsize; });
1515 m_subpackage.m_total_modified_fees = std::accumulate(workspaces.cbegin(), workspaces.cend(),
CAmount{0},
1516 [](
CAmount sum,
auto& ws) { return sum + ws.m_modified_fees; });
1517 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1518 std::vector<Wtxid> all_package_wtxids;
1519 all_package_wtxids.reserve(workspaces.size());
1520 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1521 [](
const auto& ws) { return ws.m_ptx->GetWitnessHash(); });
1523 if (
args.m_package_feerates &&
1524 !
CheckFeeRate(m_subpackage.m_total_vsize, m_subpackage.m_total_modified_fees, placeholder_state)) {
1531 if (!PackageMempoolChecks(txns, workspaces, m_subpackage.m_total_vsize, package_state)) {
1536 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1542 if (m_pool.m_opts.require_standard) {
1545 if (!
CheckEphemeralSpends(txns, m_pool.m_opts.dust_relay_feerate, m_pool, child_state, child_wtxid)) {
1552 for (Workspace& ws : workspaces) {
1553 ws.m_package_feerate = package_feerate;
1554 if (!PolicyScriptChecks(
args, ws)) {
1560 if (
args.m_test_accept) {
1561 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1562 CFeeRate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1563 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1564 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1565 results.emplace(ws.m_ptx->GetWitnessHash(),
1567 ws.m_vsize, ws.m_base_fees, effective_feerate,
1568 effective_feerate_wtxids));
1574 if (!SubmitPackage(
args, workspaces, package_state, results)) {
1582void MemPoolAccept::CleanupTemporaryCoins()
1603 for (
const auto& outpoint : m_viewmempool.GetNonBaseCoins()) {
1606 m_view.Uncache(outpoint);
1609 m_viewmempool.Reset();
1617 if (subpackage.size() > 1) {
1618 return AcceptMultipleTransactionsInternal(subpackage,
args);
1620 const auto& tx = subpackage.front();
1621 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(
args);
1622 const auto single_res = AcceptSingleTransactionInternal(tx, single_args);
1633 ClearSubPackageState();
1640 Assert(!package.empty());
1666 std::map<Wtxid, MempoolAcceptResult> results_final;
1670 std::map<Wtxid, MempoolAcceptResult> individual_results_nonfinal;
1672 bool quit_early{
false};
1673 std::vector<CTransactionRef> txns_package_eval;
1674 for (
const auto& tx : package) {
1676 const auto& txid = tx->
GetHash();
1680 if (m_pool.exists(wtxid)) {
1690 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1692 }
else if (m_pool.exists(txid)) {
1700 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1706 const auto single_package_res = AcceptSubPackage({tx},
args);
1707 const auto& single_res = single_package_res.m_tx_results.at(wtxid);
1711 assert(m_pool.exists(wtxid));
1712 results_final.emplace(wtxid, single_res);
1713 }
else if (package.size() == 1 ||
1727 individual_results_nonfinal.emplace(wtxid, single_res);
1729 individual_results_nonfinal.emplace(wtxid, single_res);
1730 txns_package_eval.push_back(tx);
1735 auto multi_submission_result = quit_early || txns_package_eval.empty() ?
PackageMempoolAcceptResult(package_state_quit_early, {}) :
1736 AcceptSubPackage(txns_package_eval,
args);
1742 ClearSubPackageState();
1749 for (
const auto& tx : package) {
1751 if (multi_submission_result.m_tx_results.contains(wtxid)) {
1753 Assume(!results_final.contains(wtxid));
1756 const auto& txresult = multi_submission_result.m_tx_results.at(wtxid);
1763 results_final.emplace(wtxid, txresult);
1765 }
else if (
const auto it{results_final.find(wtxid)}; it != results_final.end()) {
1769 Assume(!individual_results_nonfinal.contains(wtxid));
1771 if (!m_pool.exists(tx->
GetHash())) {
1776 results_final.erase(wtxid);
1779 }
else if (
const auto it{individual_results_nonfinal.find(wtxid)}; it != individual_results_nonfinal.end()) {
1782 results_final.emplace(wtxid, it->second);
1785 Assume(results_final.size() == package.size());
1792 int64_t accept_time,
bool bypass_limits,
bool test_accept)
1799 std::vector<COutPoint> coins_to_uncache;
1801 auto args = MemPoolAccept::ATMPArgs::SingleAccept(chainparams, accept_time, bypass_limits, coins_to_uncache, test_accept);
1802 MempoolAcceptResult result = MemPoolAccept(pool, active_chainstate).AcceptSingleTransactionAndCleanup(tx,
args);
1810 for (
const COutPoint& hashTx : coins_to_uncache)
1813 tx->GetHash().data(),
1824 const Package& package,
bool test_accept,
const std::optional<CFeeRate>& client_maxfeerate)
1827 assert(!package.empty());
1828 assert(std::all_of(package.cbegin(), package.cend(), [](
const auto& tx){return tx != nullptr;}));
1830 std::vector<COutPoint> coins_to_uncache;
1835 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(chainparams,
GetTime(), coins_to_uncache);
1836 return MemPoolAccept(pool, active_chainstate).AcceptMultipleTransactionsAndCleanup(package,
args);
1838 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(chainparams,
GetTime(), coins_to_uncache, client_maxfeerate);
1839 return MemPoolAccept(pool, active_chainstate).AcceptPackage(package,
args);
1844 if (test_accept || result.m_state.IsInvalid()) {
1845 for (
const COutPoint& hashTx : coins_to_uncache) {
1864 nSubsidy >>= halvings;
1869 : m_dbview{
std::move(db_params),
std::move(options)},
1870 m_catcherview(&m_dbview) {}
1872void CoinsViews::InitCache()
1875 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1882 std::optional<uint256> from_snapshot_blockhash)
1883 : m_mempool(mempool),
1884 m_blockman(blockman),
1885 m_chainman(chainman),
1887 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1898const CBlockIndex* Chainstate::SnapshotBase()
const
1902 return m_cached_snapshot_base;
1907 if (!m_target_blockhash)
return nullptr;
1909 return m_cached_target_block;
1912void Chainstate::SetTargetBlock(
CBlockIndex* block)
1917 m_target_blockhash.reset();
1919 m_cached_target_block = block;
1922void Chainstate::SetTargetBlockHash(
uint256 block_hash)
1924 m_target_blockhash = block_hash;
1925 m_cached_target_block =
nullptr;
1929 size_t cache_size_bytes,
1936 .cache_bytes = cache_size_bytes,
1937 .memory_only = in_memory,
1938 .wipe_data = should_wipe,
1946void Chainstate::InitCoinsCache(
size_t cache_size_bytes)
1972 if (chain.Tip() ==
nullptr) {
1981 LogInfo(
"Leaving InitialBlockDownload (latching to false)");
1990 if (this->GetRole().historical) {
1995 LogWarning(
"Found invalid chain more than 6 blocks longer than our best chain. This could be due to database corruption or consensus incompatibility with peers.");
1998 _(
"Warning: Found invalid chain more than 6 blocks longer than our best chain. This could be due to database corruption or consensus incompatibility with peers."));
2011 SetBlockFailureFlags(pindexNew);
2016 LogInfo(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n", __func__,
2021 LogInfo(
"%s: current best=%s height=%d log2_work=%f date=%s\n", __func__,
2059 if (
VerifyScript(scriptSig,
m_tx_out.
scriptPubKey, witness,
m_flags,
CachingTransactionSignatureChecker(
ptxTo,
nIn,
m_tx_out.
nValue,
cacheStore, *
m_signature_cache, *
txdata), &error)) {
2060 return std::nullopt;
2063 return std::make_pair(error, std::move(debug_str));
2068 : m_signature_cache{signature_cache_bytes}
2079 LogInfo(
"Using %zu MiB out of %zu MiB requested for script execution cache, able to store %zu elements",
2080 approx_size_bytes >> 20, script_execution_cache_bytes >> 20, num_elems);
2106 std::vector<CScriptCheck>* pvChecks)
2111 pvChecks->reserve(tx.
vin.size());
2128 std::vector<CTxOut> spent_outputs;
2129 spent_outputs.reserve(tx.
vin.size());
2131 for (
const auto& txin : tx.
vin) {
2135 spent_outputs.emplace_back(coin.
out);
2137 txdata.
Init(tx, std::move(spent_outputs));
2141 for (
unsigned int i = 0; i < tx.
vin.size(); i++) {
2152 pvChecks->emplace_back(std::move(
check));
2153 }
else if (
auto result =
check(); result.has_value()) {
2168 if (cacheFullScriptStore && !pvChecks) {
2196 if (undo.nHeight == 0) {
2202 undo.nHeight = alternate.
nHeight;
2227 LogError(
"DisconnectBlock(): failure reading undo data\n");
2231 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
2232 LogError(
"DisconnectBlock(): block and undo data inconsistent\n");
2242 bool fEnforceBIP30 = !((pindex->
nHeight==91722 && pindex->
GetBlockHash() ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
2243 (pindex->
nHeight==91812 && pindex->
GetBlockHash() ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"}));
2246 for (
int i = block.
vtx.size() - 1; i >= 0; i--) {
2250 bool is_bip30_exception = (is_coinbase && !fEnforceBIP30);
2254 for (
size_t o = 0; o < tx.
vout.size(); o++) {
2255 if (!tx.
vout[o].scriptPubKey.IsUnspendable()) {
2260 if (!is_bip30_exception) {
2271 LogError(
"DisconnectBlock(): transaction and undo data inconsistent\n");
2274 for (
unsigned int j = tx.
vin.size(); j > 0;) {
2345 const auto time_start{SteadyClock::now()};
2361 if (!
CheckBlock(block, state, params.GetConsensus(), !fJustCheck, !fJustCheck)) {
2373 uint256 hashPrevBlock = pindex->
pprev ==
nullptr ?
uint256() : pindex->pprev->GetBlockHash();
2380 if (block_hash == params.GetConsensus().hashGenesisBlock) {
2386 const char* script_check_reason;
2388 script_check_reason =
"assumevalid=0 (always verify)";
2390 constexpr int64_t TWO_WEEKS_IN_SECONDS{60 * 60 * 24 * 7 * 2};
2398 script_check_reason =
"assumevalid hash not in headers";
2399 }
else if (it->second.GetAncestor(pindex->
nHeight) != pindex) {
2400 script_check_reason = (pindex->
nHeight > it->second.nHeight) ?
"block height above assumevalid height" :
"block not in assumevalid chain";
2402 script_check_reason =
"block not in best header chain";
2404 script_check_reason =
"best header chainwork below minimumchainwork";
2406 script_check_reason =
"block too recent relative to best header";
2422 script_check_reason =
nullptr;
2426 const auto time_1{SteadyClock::now()};
2427 m_chainman.time_check += time_1 - time_start;
2429 Ticks<MillisecondsDouble>(time_1 - time_start),
2471 static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2503 fEnforceBIP30 = fEnforceBIP30 && (!pindexBIP34height || !(pindexBIP34height->
GetBlockHash() == params.GetConsensus().BIP34Hash));
2508 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2509 for (
const auto& tx : block.
vtx) {
2510 for (
size_t o = 0; o < tx->
vout.size(); o++) {
2513 "tried to overwrite transaction");
2520 int nLockTimeFlags = 0;
2528 const auto time_2{SteadyClock::now()};
2531 Ticks<MillisecondsDouble>(time_2 - time_1),
2535 const bool fScriptChecks{!!script_check_reason};
2537 if (script_check_reason != m_last_script_check_reason_logged && role.validated && !role.historical) {
2538 if (fScriptChecks) {
2539 LogInfo(
"Enabling script verification at block #%d (%s): %s.",
2540 pindex->
nHeight, block_hash.ToString(), script_check_reason);
2542 LogInfo(
"Disabling script verification at block #%d (%s).",
2543 pindex->
nHeight, block_hash.ToString());
2545 m_last_script_check_reason_logged = script_check_reason;
2555 std::optional<CCheckQueueControl<CScriptCheck>> control;
2558 std::vector<PrecomputedTransactionData> txsdata(block.
vtx.size());
2560 std::vector<int> prevheights;
2563 int64_t nSigOpsCost = 0;
2564 blockundo.
vtxundo.reserve(block.
vtx.size() - 1);
2565 for (
unsigned int i = 0; i < block.
vtx.size(); i++)
2570 nInputs += tx.
vin.size();
2586 "accumulated fee in the block out of range");
2593 prevheights.resize(tx.
vin.size());
2594 for (
size_t j = 0; j < tx.
vin.size(); j++) {
2598 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2617 bool fCacheResults = fJustCheck;
2623 std::vector<CScriptCheck> vChecks;
2625 if (tx_ok) control->Add(std::move(vChecks));
2639 blockundo.
vtxundo.emplace_back();
2643 const auto time_3{SteadyClock::now()};
2645 LogDebug(
BCLog::BENCH,
" - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) [%.2fs (%.2fms/blk)]\n", (
unsigned)block.
vtx.size(),
2646 Ticks<MillisecondsDouble>(time_3 - time_2), Ticks<MillisecondsDouble>(time_3 - time_2) / block.
vtx.size(),
2647 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_3 - time_2) / (nInputs - 1),
2648 Ticks<SecondsDouble>(
m_chainman.time_connect),
2652 if (block.
vtx[0]->GetValueOut() > blockReward && state.
IsValid()) {
2654 strprintf(
"coinbase pays too much (actual=%d vs limit=%d)", block.
vtx[0]->GetValueOut(), blockReward));
2657 auto parallel_result = control->Complete();
2658 if (parallel_result.has_value() && state.
IsValid()) {
2666 const auto time_4{SteadyClock::now()};
2668 LogDebug(
BCLog::BENCH,
" - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n", nInputs - 1,
2669 Ticks<MillisecondsDouble>(time_4 - time_2),
2670 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_4 - time_2) / (nInputs - 1),
2671 Ticks<SecondsDouble>(
m_chainman.time_verify),
2678 if (!
m_blockman.WriteBlockUndo(blockundo, state, *pindex)) {
2682 const auto time_5{SteadyClock::now()};
2685 Ticks<MillisecondsDouble>(time_5 - time_4),
2697 const auto time_6{SteadyClock::now()};
2700 Ticks<MillisecondsDouble>(time_6 - time_5),
2710 Ticks<std::chrono::nanoseconds>(time_5 - time_start)
2719 return this->GetCoinsCacheSizeState(
2725 size_t max_coins_cache_size_bytes,
2726 size_t max_mempool_size_bytes)
2731 int64_t nTotalSpace =
2732 max_coins_cache_size_bytes + std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2734 if (cacheSize > nTotalSpace) {
2735 LogInfo(
"Cache size (%s) exceeds total space (%s)\n", cacheSize, nTotalSpace);
2746 int nManualPruneHeight)
2750 std::set<int> setFilesToPrune;
2751 bool full_flush_completed =
false;
2758 bool fFlushForPrune =
false;
2766 std::optional<std::string> limiting_lock;
2768 for (
const auto& prune_lock :
m_blockman.m_prune_locks) {
2769 if (prune_lock.second.height_first == std::numeric_limits<int>::max())
continue;
2772 last_prune = std::max(1, std::min(last_prune, lock_height));
2773 if (last_prune == lock_height) {
2774 limiting_lock = prune_lock.first;
2778 if (limiting_lock) {
2779 LogDebug(
BCLog::PRUNE,
"%s limited pruning to height %d\n", limiting_lock.value(), last_prune);
2782 if (nManualPruneHeight > 0) {
2787 std::min(last_prune, nManualPruneHeight),
2795 if (!setFilesToPrune.empty()) {
2796 fFlushForPrune =
true;
2798 m_blockman.m_block_tree_db->WriteFlag(
"prunedblockfiles",
true);
2811 bool should_write = (mode ==
FlushStateMode::ALWAYS) || fCacheLarge || fCacheCritical || fPeriodicWrite || fFlushForPrune;
2814 LogDebug(
BCLog::COINDB,
"Writing chainstate to disk: flush mode=%s, prune=%d, large=%d, critical=%d, periodic=%d",
2815 FlushStateModeNames[
size_t(mode)], fFlushForPrune, fCacheLarge, fCacheCritical, fPeriodicWrite);
2828 LogWarning(
"%s: Failed to flush block file.\n", __func__);
2839 if (fFlushForPrune) {
2845 if (!
CoinsTip().GetBestBlock().IsNull()) {
2846 if (coins_mem_usage >=
WARN_FLUSH_COINS_SIZE)
LogWarning(
"Flushing large (%d GiB) UTXO set to disk, it may take several minutes", coins_mem_usage >> 30);
2863 full_flush_completed =
true;
2865 int64_t{Ticks<std::chrono::microseconds>(
NodeClock::now() - nNow)},
2867 (uint64_t)coins_count,
2868 (uint64_t)coins_mem_usage,
2869 (
bool)fFlushForPrune);
2873 if (should_write ||
m_next_write == NodeClock::time_point::max()) {
2882 }
catch (
const std::runtime_error& e) {
2909 const std::string& func_name,
2910 const std::string&
prefix,
2922 chainman.GuessVerificationProgress(tip),
2925 !warning_messages.empty() ?
strprintf(
" warning='%s'", warning_messages) :
"");
2928void Chainstate::UpdateTip(
const CBlockIndex* pindexNew)
2931 const auto& coins_tip = this->
CoinsTip();
2937 constexpr int BACKGROUND_LOG_INTERVAL = 2000;
2938 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2949 std::vector<bilingual_str> warning_messages;
2952 for (
auto [bit, active] : bits) {
2957 warning_messages.push_back(warning);
2984 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2987 LogError(
"DisconnectTip(): Failed to read block\n");
2991 const auto time_start{SteadyClock::now()};
2995 if (DisconnectBlock(block, pindexDelete, view) !=
DISCONNECT_OK) {
2999 bool flushed = view.
Flush(
false);
3003 Ticks<MillisecondsDouble>(SteadyClock::now() - time_start));
3007 const int max_height_first{pindexDelete->
nHeight - 1};
3008 for (
auto& prune_lock :
m_blockman.m_prune_locks) {
3009 if (prune_lock.second.height_first <= max_height_first)
continue;
3011 prune_lock.second.height_first = max_height_first;
3012 LogDebug(
BCLog::PRUNE,
"%s prune lock moved back to %d\n", prune_lock.first, max_height_first);
3031 UpdateTip(pindexDelete->
pprev);
3089 std::shared_ptr<const CBlock> block_to_connect,
3098 const auto time_1{SteadyClock::now()};
3099 if (!block_to_connect) {
3100 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
3104 block_to_connect = std::move(pblockNew);
3109 const auto time_2{SteadyClock::now()};
3110 SteadyClock::time_point time_3;
3114 Ticks<MillisecondsDouble>(time_2 - time_1));
3117 bool rv =
ConnectBlock(*block_to_connect, state, pindexNew, view);
3127 time_3 = SteadyClock::now();
3128 m_chainman.time_connect_total += time_3 - time_2;
3131 Ticks<MillisecondsDouble>(time_3 - time_2),
3132 Ticks<SecondsDouble>(
m_chainman.time_connect_total),
3134 bool flushed = view.
Flush(
false);
3137 const auto time_4{SteadyClock::now()};
3140 Ticks<MillisecondsDouble>(time_4 - time_3),
3147 const auto time_5{SteadyClock::now()};
3148 m_chainman.time_chainstate += time_5 - time_4;
3150 Ticks<MillisecondsDouble>(time_5 - time_4),
3151 Ticks<SecondsDouble>(
m_chainman.time_chainstate),
3160 UpdateTip(pindexNew);
3162 const auto time_6{SteadyClock::now()};
3163 m_chainman.time_post_connect += time_6 - time_5;
3166 Ticks<MillisecondsDouble>(time_6 - time_5),
3167 Ticks<SecondsDouble>(
m_chainman.time_post_connect),
3170 Ticks<MillisecondsDouble>(time_6 - time_1),
3184 m_chainman.MaybeValidateSnapshot(*
this, current_cs);
3186 connectTrace.
BlockConnected(pindexNew, std::move(block_to_connect));
3211 bool fInvalidAncestor =
false;
3221 if (fFailedChain || fMissingData) {
3228 while (pindexTest != pindexFailed) {
3232 }
else if (fMissingData) {
3237 std::make_pair(pindexFailed->
pprev, pindexFailed));
3240 pindexFailed = pindexFailed->
pprev;
3243 fInvalidAncestor =
true;
3246 pindexTest = pindexTest->
pprev;
3248 if (!fInvalidAncestor)
3280 bool fBlocksDisconnected =
false;
3294 fBlocksDisconnected =
true;
3298 std::vector<CBlockIndex*> vpindexToConnect;
3299 bool fContinue =
true;
3304 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3305 vpindexToConnect.clear();
3306 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3309 vpindexToConnect.push_back(pindexIter);
3310 pindexIter = pindexIter->
pprev;
3315 for (
CBlockIndex* pindexConnect : vpindexToConnect | std::views::reverse) {
3316 if (!
ConnectTip(state, pindexConnect, pindexConnect == pindexMostWork ? pblock : std::shared_ptr<const CBlock>(), connectTrace, disconnectpool)) {
3323 fInvalidFound =
true;
3344 if (fBlocksDisconnected) {
3365 bool fNotify =
false;
3366 bool fInitialBlockDownload =
false;
3370 pindexHeader = m_best_header;
3372 if (pindexHeader != m_last_notified_header) {
3375 m_last_notified_header = pindexHeader;
3388 if (signals.CallbacksPending() > 10) {
3389 signals.SyncWithValidationInterfaceQueue();
3393bool Chainstate::ActivateBestChain(
BlockValidationState& state, std::shared_ptr<const CBlock> pblock)
3418 bool exited_ibd{
false};
3435 bool blocks_connected =
false;
3441 if (pindexMostWork ==
nullptr) {
3446 if (pindexMostWork ==
nullptr || pindexMostWork ==
m_chain.
Tip()) {
3450 bool fInvalidFound =
false;
3451 std::shared_ptr<const CBlock> nullBlockPtr;
3456 if (!
ActivateBestChainStep(state, pindexMostWork, pblock && pblock->GetHash() == pindexMostWork->
GetBlockHash() ? pblock : nullBlockPtr, fInvalidFound, connectTrace)) {
3460 blocks_connected =
true;
3462 if (fInvalidFound) {
3464 pindexMostWork =
nullptr;
3469 assert(trace.pblock && trace.pindex);
3480 if (!blocks_connected)
return true;
3485 if (was_in_ibd && !still_in_ibd) {
3518 bool reached_target;
3535 if (reached_target) {
3554 }
while (pindexNewTip != pindexMostWork);
3589 return ActivateBestChain(state, std::shared_ptr<const CBlock>());
3599 if (pindex->
nHeight == 0)
return false;
3602 bool pindex_was_in_chain =
false;
3603 int disconnected = 0;
3617 std::multimap<const arith_uint256, CBlockIndex*> highpow_outofchain_headers;
3621 for (
auto& entry :
m_blockman.m_block_index) {
3631 highpow_outofchain_headers.insert({candidate->
nChainWork, candidate});
3648 pindex_was_in_chain =
true;
3661 if (!
ret)
return false;
3684 auto candidate_it = highpow_outofchain_headers.lower_bound(invalid_walk_tip->
pprev->
nChainWork);
3686 const bool best_header_needs_update{
m_chainman.m_best_header->GetAncestor(invalid_walk_tip->
nHeight) == invalid_walk_tip};
3687 if (best_header_needs_update) {
3692 while (candidate_it != highpow_outofchain_headers.end()) {
3696 candidate->nStatus &= ~BLOCK_FAILED_VALID;
3701 candidate_it = highpow_outofchain_headers.erase(candidate_it);
3711 if (best_header_needs_update &&
3720 to_mark_failed = invalid_walk_tip;
3746 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3756 if (pindex_was_in_chain) {
3766 *to_mark_failed->
pprev,
3778void Chainstate::SetBlockFailureFlags(
CBlockIndex* invalid_block)
3782 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3783 if (invalid_block != &block_index && block_index.GetAncestor(invalid_block->
nHeight) == invalid_block) {
3784 block_index.nStatus = (block_index.nStatus & ~BLOCK_FAILED_VALID) |
BLOCK_FAILED_CHILD;
3796 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3798 block_index.nStatus &= ~BLOCK_FAILED_MASK;
3803 if (&block_index ==
m_chainman.m_best_invalid) {
3830 if (!target_block) {
3837 if (target_block->GetAncestor(pindex->
nHeight) == pindex) {
3847 pindexNew->
nTx = block.
vtx.size();
3853 auto prev_tx_sum = [](
CBlockIndex& block) {
return block.nTx + (block.pprev ? block.pprev->m_chain_tx_count : 0); };
3856 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",
3860 pindexNew->nFile = pos.
nFile;
3861 pindexNew->nDataPos = pos.
nPos;
3862 pindexNew->nUndoPos = 0;
3872 std::deque<CBlockIndex*> queue;
3873 queue.push_back(pindexNew);
3876 while (!queue.empty()) {
3884 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",
3889 for (
const auto& c : m_chainstates) {
3890 c->TryAddBlockIndexCandidate(pindex);
3892 std::pair<std::multimap<CBlockIndex*, CBlockIndex*>::iterator, std::multimap<CBlockIndex*, CBlockIndex*>::iterator> range =
m_blockman.
m_blocks_unlinked.equal_range(pindex);
3893 while (range.first != range.second) {
3894 std::multimap<CBlockIndex*, CBlockIndex*>::iterator it = range.first;
3895 queue.push_back(it->second);
3926 "hashMerkleRoot mismatch");
3935 "bad-txns-duplicate",
3936 "duplicate transaction");
3951 if (expect_witness_commitment) {
3956 assert(!block.
vtx.empty() && !block.
vtx[0]->vin.empty());
3957 const auto& witness_stack{block.
vtx[0]->vin[0].scriptWitness.stack};
3959 if (witness_stack.size() != 1 || witness_stack[0].size() != 32) {
3962 "bad-witness-nonce-size",
3963 strprintf(
"%s : invalid witness reserved value size", __func__));
3972 if (memcmp(hash_witness.
begin(), &block.
vtx[0]->vout[commitpos].scriptPubKey[6], 32)) {
3975 "bad-witness-merkle-match",
3976 strprintf(
"%s : witness merkle commitment mismatch", __func__));
3985 for (
const auto& tx : block.
vtx) {
3989 "unexpected-witness",
3990 strprintf(
"%s : unexpected witness data found", __func__));
4030 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase())
4032 for (
unsigned int i = 1; i < block.
vtx.size(); i++)
4033 if (block.
vtx[i]->IsCoinBase())
4038 for (
const auto& tx : block.
vtx) {
4050 unsigned int nSigOps = 0;
4051 for (
const auto& tx : block.
vtx)
4058 if (fCheckPOW && fCheckMerkleRoot)
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());
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(),
4125 [](
auto& tx) { return GetSerializeSize(TX_NO_WITNESS(tx)) == 64; });
4166 assert(pindexPrev !=
nullptr);
4167 const int nHeight = pindexPrev->nHeight + 1;
4175 if (block.
GetBlockTime() <= pindexPrev->GetMedianTimePast())
4214 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4217 bool enforce_locktime_median_time_past{
false};
4219 assert(pindexPrev !=
nullptr);
4220 enforce_locktime_median_time_past =
true;
4223 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past ?
4228 for (
const auto& tx : block.
vtx) {
4238 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4239 !std::equal(
expect.begin(),
expect.end(), block.
vtx[0]->vin[0].scriptSig.begin())) {
4275 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4277 if (miSelf !=
m_blockman.m_block_index.end()) {
4302 pindexPrev = &((*mi).second);
4312 if (!min_pow_checked) {
4347 blocks_left = std::max<int64_t>(0, blocks_left);
4348 const double progress{100.0 * last_accepted.nHeight / (last_accepted.nHeight + blocks_left)};
4349 LogInfo(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n", last_accepted.nHeight, progress);
4367 if (now < m_last_presync_update + std::chrono::milliseconds{250})
return;
4368 m_last_presync_update = now;
4372 if (initial_download) {
4374 blocks_left = std::max<int64_t>(0, blocks_left);
4375 const double progress{100.0 * height / (height + blocks_left)};
4376 LogInfo(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n", height, progress);
4383 const CBlock& block = *pblock;
4385 if (fNewBlock) *fNewBlock =
false;
4389 CBlockIndex *&pindex = ppindex ? *ppindex : pindexDummy;
4391 bool accepted_header{
AcceptBlockHeader(block, state, &pindex, min_pow_checked)};
4394 if (!accepted_header)
4418 if (fAlreadyHave)
return true;
4420 if (pindex->
nTx != 0)
return true;
4421 if (!fHasMoreOrSameWork)
return true;
4422 if (fTooFarAhead)
return true;
4433 if (!
CheckBlock(block, state, params.GetConsensus()) ||
4449 if (fNewBlock) *fNewBlock =
true;
4457 if (blockPos.IsNull()) {
4458 state.
Error(
strprintf(
"%s: Failed to find position to write new block to disk", __func__));
4463 }
catch (
const std::runtime_error& e) {
4487 if (new_block) *new_block =
false;
4502 ret =
AcceptBlock(block, state, &pindex, force_processing,
nullptr, new_block, min_pow_checked);
4517 LogError(
"%s: ActivateBestChain failed (%s)\n", __func__, state.
ToString());
4523 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
4524 LogError(
"%s: [background] ActivateBestChain failed (%s)\n", __func__, bg_state.
ToString());
4549 const bool check_pow,
4550 const bool check_merkle_root)
4561 state.
Invalid({},
"inconclusive-not-best-prevblk");
4602 index_dummy.pprev = tip;
4604 index_dummy.phashBlock = &block_hash;
4608 if(!chainstate.
ConnectBlock(block, state, &index_dummy, view_dummy,
true)) {
4653 target = target->pprev;
4663 LogInfo(
"Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f",
4670 if (!this->GetRole().historical) {
4698 int nCheckLevel,
int nCheckDepth)
4707 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
4710 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
4711 LogInfo(
"Verifying last %i blocks at level %i", nCheckDepth, nCheckLevel);
4715 int nGoodTransactions = 0;
4718 bool skipped_no_block_data{
false};
4719 bool skipped_l3_checks{
false};
4720 LogInfo(
"Verification progress: 0%%");
4725 const int percentageDone = std::max(1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (
double)nCheckDepth * (nCheckLevel >= 4 ? 50 : 100))));
4726 if (reportDone < percentageDone / 10) {
4728 LogInfo(
"Verification progress: %d%%", percentageDone);
4729 reportDone = percentageDone / 10;
4738 LogInfo(
"Block verification stopping at height %d (no data). This could be due to pruning or use of an assumeutxo snapshot.", pindex->
nHeight);
4739 skipped_no_block_data =
true;
4749 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensus_params)) {
4750 LogError(
"Verification error: found bad block at %d, hash=%s (%s)",
4755 if (nCheckLevel >= 2 && pindex) {
4767 if (nCheckLevel >= 3) {
4776 nGoodTransactions = 0;
4777 pindexFailure = pindex;
4779 nGoodTransactions += block.
vtx.size();
4782 skipped_l3_checks =
true;
4787 if (pindexFailure) {
4788 LogError(
"Verification error: coin database inconsistencies found (last %i blocks, %i good transactions before that)", chainstate.
m_chain.
Height() - pindexFailure->
nHeight + 1, nGoodTransactions);
4791 if (skipped_l3_checks) {
4792 LogWarning(
"Skipped verification of level >=3 (insufficient database cache size). Consider increasing -dbcache.");
4799 if (nCheckLevel >= 4 && !skipped_l3_checks) {
4801 const int percentageDone = std::max(1, std::min(99, 100 - (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (
double)nCheckDepth * 50)));
4802 if (reportDone < percentageDone / 10) {
4804 LogInfo(
"Verification progress: %d%%", percentageDone);
4805 reportDone = percentageDone / 10;
4814 if (!chainstate.
ConnectBlock(block, state, pindex, coins)) {
4822 LogInfo(
"Verification: No coin database inconsistencies in last %i blocks (%i transactions)", block_count, nGoodTransactions);
4824 if (skipped_l3_checks) {
4827 if (skipped_no_block_data) {
4845 if (!tx->IsCoinBase()) {
4846 for (
const CTxIn &txin : tx->vin) {
4864 if (hashHeads.empty())
return true;
4865 if (hashHeads.size() != 2) {
4866 LogError(
"ReplayBlocks(): unknown inconsistent state\n");
4877 if (!
m_blockman.m_block_index.contains(hashHeads[0])) {
4878 LogError(
"ReplayBlocks(): reorganization to unknown block requested\n");
4881 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
4883 if (!hashHeads[1].IsNull()) {
4884 if (!
m_blockman.m_block_index.contains(hashHeads[1])) {
4885 LogError(
"ReplayBlocks(): reorganization from unknown block requested\n");
4888 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
4890 assert(pindexFork !=
nullptr);
4894 const int nForkHeight{pindexFork ? pindexFork->
nHeight : 0};
4895 if (pindexOld != pindexFork) {
4897 while (pindexOld != pindexFork) {
4904 if (pindexOld->
nHeight % 10'000 == 0) {
4917 pindexOld = pindexOld->
pprev;
4923 if (nForkHeight < pindexNew->
nHeight) {
4955 block = block->pprev;
4961void Chainstate::ClearBlockIndexCandidates()
4973 if (!
ret)
return false;
4975 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
4977 std::vector<CBlockIndex*> vSortedByHeight{
m_blockman.GetAllBlockIndices()};
4978 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
4992 for (
const auto& chainstate : m_chainstates) {
4993 chainstate->TryAddBlockIndexCandidate(pindex);
4997 m_best_invalid = pindex;
5000 m_best_header = pindex;
5016 if (
m_blockman.m_block_index.contains(params.GenesisBlock().GetHash()))
5020 const CBlock& block = params.GenesisBlock();
5022 if (blockPos.IsNull()) {
5023 LogError(
"%s: writing genesis block to disk failed\n", __func__);
5028 }
catch (
const std::runtime_error& e) {
5029 LogError(
"%s: failed to write genesis block: %s\n", __func__, e.what());
5039 std::multimap<uint256, FlatFilePos>* blocks_with_unknown_parent)
5042 assert(!dbp == !blocks_with_unknown_parent);
5044 const auto start{SteadyClock::now()};
5052 uint64_t nRewind = blkdat.GetPos();
5053 while (!blkdat.eof()) {
5056 blkdat.SetPos(nRewind);
5059 unsigned int nSize = 0;
5063 blkdat.FindByte(std::byte(params.MessageStart()[0]));
5064 nRewind = blkdat.GetPos() + 1;
5066 if (buf != params.MessageStart()) {
5073 }
catch (
const std::exception&) {
5080 const uint64_t nBlockPos{blkdat.GetPos()};
5082 dbp->
nPos = nBlockPos;
5083 blkdat.SetLimit(nBlockPos + nSize);
5089 nRewind = nBlockPos + nSize;
5090 blkdat.SkipTo(nRewind);
5092 std::shared_ptr<CBlock> pblock{};
5100 if (dbp && blocks_with_unknown_parent) {
5101 blocks_with_unknown_parent->emplace(header.
hashPrevBlock, *dbp);
5110 blkdat.SetPos(nBlockPos);
5111 pblock = std::make_shared<CBlock>();
5113 nRewind = blkdat.GetPos();
5116 if (
AcceptBlock(pblock, state,
nullptr,
true, dbp,
nullptr,
true)) {
5122 }
else if (hash != params.GetConsensus().hashGenesisBlock && pindex->
nHeight % 1000 == 0) {
5149 if (
auto result{ActivateBestChains()}; !result) {
5157 if (!blocks_with_unknown_parent)
continue;
5160 std::deque<uint256> queue;
5161 queue.push_back(hash);
5162 while (!queue.empty()) {
5165 auto range = blocks_with_unknown_parent->equal_range(head);
5166 while (range.first != range.second) {
5167 std::multimap<uint256, FlatFilePos>::iterator it = range.first;
5168 std::shared_ptr<CBlock> pblockrecursive = std::make_shared<CBlock>();
5170 const auto& block_hash{pblockrecursive->GetHash()};
5174 if (
AcceptBlock(pblockrecursive, dummy,
nullptr,
true, &it->second,
nullptr,
true)) {
5176 queue.push_back(block_hash);
5180 blocks_with_unknown_parent->erase(it);
5184 }
catch (
const std::exception& e) {
5196 LogDebug(
BCLog::REINDEX,
"%s: unexpected data at file offset 0x%x - %s. continuing\n", __func__, (nRewind - 1), e.what());
5199 }
catch (
const std::runtime_error& e) {
5202 LogInfo(
"Loaded %i blocks from external file in %dms", nLoaded, Ticks<std::chrono::milliseconds>(SteadyClock::now() - start));
5237 best_hdr_chain.
SetTip(*m_best_header);
5239 std::multimap<const CBlockIndex*, const CBlockIndex*> forward;
5240 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
5242 if (!best_hdr_chain.
Contains(&block_index)) {
5244 assert(block_index.pprev);
5245 forward.emplace(block_index.pprev, &block_index);
5259 const CBlockIndex* pindexFirstNeverProcessed =
nullptr;
5260 const CBlockIndex* pindexFirstNotTreeValid =
nullptr;
5261 const CBlockIndex* pindexFirstNotTransactionsValid =
nullptr;
5262 const CBlockIndex* pindexFirstNotChainValid =
nullptr;
5263 const CBlockIndex* pindexFirstNotScriptsValid =
nullptr;
5270 const CBlockIndex *snap_first_missing{}, *snap_first_notx{}, *snap_first_notv{}, *snap_first_nocv{}, *snap_first_nosv{};
5271 auto snap_update_firsts = [&] {
5272 if (pindex == snap_base) {
5273 std::swap(snap_first_missing, pindexFirstMissing);
5274 std::swap(snap_first_notx, pindexFirstNeverProcessed);
5275 std::swap(snap_first_notv, pindexFirstNotTransactionsValid);
5276 std::swap(snap_first_nocv, pindexFirstNotChainValid);
5277 std::swap(snap_first_nosv, pindexFirstNotScriptsValid);
5281 while (pindex !=
nullptr) {
5283 if (pindexFirstInvalid ==
nullptr && pindex->nStatus &
BLOCK_FAILED_VALID) pindexFirstInvalid = pindex;
5284 if (pindexFirstMissing ==
nullptr && !(pindex->nStatus &
BLOCK_HAVE_DATA)) {
5285 pindexFirstMissing = pindex;
5287 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) pindexFirstNeverProcessed = pindex;
5290 if (pindex->
pprev !=
nullptr) {
5291 if (pindexFirstNotTransactionsValid ==
nullptr &&
5293 pindexFirstNotTransactionsValid = pindex;
5296 if (pindexFirstNotChainValid ==
nullptr &&
5298 pindexFirstNotChainValid = pindex;
5301 if (pindexFirstNotScriptsValid ==
nullptr &&
5303 pindexFirstNotScriptsValid = pindex;
5308 if (pindex->
pprev ==
nullptr) {
5311 for (
const auto& c : m_chainstates) {
5312 if (c->m_chain.Genesis() !=
nullptr) {
5313 assert(pindex == c->m_chain.Genesis());
5325 assert(pindexFirstMissing == pindexFirstNeverProcessed);
5331 if (snap_base && snap_base->GetAncestor(pindex->
nHeight) == pindex) {
5341 assert((pindexFirstNotTransactionsValid ==
nullptr || pindex == snap_base) == pindex->
HaveNumChainTxs());
5345 assert(pindexFirstNotTreeValid ==
nullptr);
5349 if (pindexFirstInvalid ==
nullptr) {
5356 if (!pindex->
pprev) {
5371 for (
const auto& c : m_chainstates) {
5372 if (c->m_chain.Tip() ==
nullptr)
continue;
5386 if (!
CBlockIndexWorkComparator()(pindex, c->m_chain.Tip()) && (pindexFirstNeverProcessed ==
nullptr || pindex == snap_base)) {
5390 if (pindexFirstInvalid ==
nullptr) {
5409 if (pindexFirstMissing ==
nullptr || pindex == c->m_chain.Tip() || pindex == c->SnapshotBase()) {
5415 if (!c->TargetBlock() || c->TargetBlock()->GetAncestor(pindex->
nHeight) == pindex) {
5429 bool foundInUnlinked =
false;
5430 while (rangeUnlinked.first != rangeUnlinked.second) {
5431 assert(rangeUnlinked.first->first == pindex->
pprev);
5432 if (rangeUnlinked.first->second == pindex) {
5433 foundInUnlinked =
true;
5436 rangeUnlinked.first++;
5438 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed !=
nullptr && pindexFirstInvalid ==
nullptr) {
5443 if (pindexFirstMissing ==
nullptr)
assert(!foundInUnlinked);
5444 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed ==
nullptr && pindexFirstMissing !=
nullptr) {
5455 for (
const auto& c : m_chainstates) {
5457 if (pindexFirstInvalid ==
nullptr) {
5458 if (!c->TargetBlock() || c->TargetBlock()->GetAncestor(pindex->
nHeight) == pindex) {
5470 snap_update_firsts();
5471 auto range{forward.equal_range(pindex)};
5472 if (range.first != range.second) {
5474 pindex = range.first->second;
5477 }
else if (best_hdr_chain.
Contains(pindex)) {
5480 pindex = best_hdr_chain[
nHeight];
5488 snap_update_firsts();
5490 if (pindex == pindexFirstInvalid) pindexFirstInvalid =
nullptr;
5491 if (pindex == pindexFirstMissing) pindexFirstMissing =
nullptr;
5492 if (pindex == pindexFirstNeverProcessed) pindexFirstNeverProcessed =
nullptr;
5493 if (pindex == pindexFirstNotTreeValid) pindexFirstNotTreeValid =
nullptr;
5494 if (pindex == pindexFirstNotTransactionsValid) pindexFirstNotTransactionsValid =
nullptr;
5495 if (pindex == pindexFirstNotChainValid) pindexFirstNotChainValid =
nullptr;
5496 if (pindex == pindexFirstNotScriptsValid) pindexFirstNotScriptsValid =
nullptr;
5500 auto rangePar{forward.equal_range(pindexPar)};
5501 while (rangePar.first->second != pindex) {
5502 assert(rangePar.first != rangePar.second);
5507 if (rangePar.first != rangePar.second) {
5509 pindex = rangePar.first->second;
5511 }
else if (pindexPar == best_hdr_chain[
nHeight - 1]) {
5513 pindex = best_hdr_chain[
nHeight];
5515 assert((pindex ==
nullptr) == (pindexPar == best_hdr_chain.
Tip()));
5527 assert(nNodes == forward.size() + best_hdr_chain.
Height() + 1);
5534 return strprintf(
"Chainstate [%s] @ height %d (%s)",
5539bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size)
5552 LogInfo(
"[%s] resized coinsdb cache to %.1f MiB",
5553 this->
ToString(), coinsdb_size * (1.0 / 1024 / 1024));
5554 LogInfo(
"[%s] resized coinstip cache to %.1f MiB",
5555 this->
ToString(), coinstip_size * (1.0 / 1024 / 1024));
5560 if (coinstip_size > old_coinstip_size) {
5574 if (pindex ==
nullptr) {
5583 const int64_t nNow{TicksSinceEpoch<std::chrono::seconds>(
NodeClock::now())};
5584 const auto block_time{
5585 (
Assume(m_best_header) && std::abs(nNow - pindex->
GetBlockTime()) <= Ticks<std::chrono::seconds>(2h) &&
5599 fTxTotal =
data.tx_count + (nNow -
data.nTime) *
data.dTxRate;
5610 assert(m_chainstates.empty());
5611 m_chainstates.emplace_back(std::make_unique<Chainstate>(mempool,
m_blockman, *
this));
5612 return *m_chainstates.back();
5624 bool existed = fs::remove(base_blockhash_path);
5626 LogWarning(
"[snapshot] snapshot chainstate dir being removed lacks %s file",
5629 }
catch (
const fs::filesystem_error& e) {
5630 LogWarning(
"[snapshot] failed to remove file %s: %s\n",
5636 LogInfo(
"Removing leveldb dir at %s\n", path_str);
5640 const bool destroyed =
DestroyDB(path_str);
5643 LogError(
"leveldb DestroyDB call failed on %s", path_str);
5670 if (!
GetParams().AssumeutxoForBlockhash(base_blockhash).has_value()) {
5672 std::string heights_formatted =
util::Join(available_heights,
", ", [&](
const auto& i) {
return util::ToString(i); });
5673 return util::Error{
Untranslated(
strprintf(
"assumeutxo block hash in snapshot metadata not recognized (hash: %s). The following snapshot heights are available: %s",
5675 heights_formatted))};
5679 if (!snapshot_start_block) {
5680 return util::Error{
Untranslated(
strprintf(
"The base block header (%s) must appear in the headers chain. Make sure all headers are syncing, and call loadtxoutset again",
5685 if (start_block_invalid) {
5689 if (!m_best_header || m_best_header->GetAncestor(snapshot_start_block->nHeight) != snapshot_start_block) {
5690 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.")};
5694 if (mempool && mempool->
size() > 0) {
5699 int64_t current_coinsdb_cache_size{0};
5700 int64_t current_coinstip_cache_size{0};
5708 static constexpr double IBD_CACHE_PERC = 0.01;
5709 static constexpr double SNAPSHOT_CACHE_PERC = 0.99;
5727 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
5728 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
5732 return std::make_unique<Chainstate>(
5733 nullptr,
m_blockman, *
this, base_blockhash));
5737 snapshot_chainstate->InitCoinsDB(
5738 static_cast<size_t>(current_coinsdb_cache_size * SNAPSHOT_CACHE_PERC),
5740 snapshot_chainstate->InitCoinsCache(
5741 static_cast<size_t>(current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
5745 this->MaybeRebalanceCaches();
5753 snapshot_chainstate.reset();
5757 "Manually remove it before restarting.\n"), fs::PathToString(*snapshot_datadir)));
5774 return cleanup_bad_snapshot(
Untranslated(
"work does not exceed active chainstate"));
5780 return cleanup_bad_snapshot(
Untranslated(
"could not write base blockhash"));
5784 Chainstate& chainstate{AddChainstate(std::move(snapshot_chainstate))};
5789 LogInfo(
"[snapshot] successfully activated snapshot %s", base_blockhash.
ToString());
5790 LogInfo(
"[snapshot] (%.2f MB)",
5791 chainstate.CoinsTip().DynamicMemoryUsage() / (1000 * 1000));
5793 this->MaybeRebalanceCaches();
5794 return snapshot_start_block;
5801 snapshot_loaded ?
"saving snapshot chainstate" :
"flushing coins cache",
5805 coins_cache.
Flush();
5810 const char*
what() const noexcept
override
5812 return "ComputeUTXOStats interrupted.";
5834 if (!snapshot_start_block) {
5841 int base_height = snapshot_start_block->
nHeight;
5844 if (!maybe_au_data) {
5846 "(%d) - refusing to load snapshot", base_height))};
5861 LogInfo(
"[snapshot] loading %d coins from snapshot %s", coins_left, base_blockhash.
ToString());
5862 int64_t coins_processed{0};
5864 while (coins_left > 0) {
5868 size_t coins_per_txid{0};
5871 if (coins_per_txid > coins_left) {
5875 for (
size_t i = 0; i < coins_per_txid; i++) {
5879 outpoint.
hash = txid;
5881 if (coin.
nHeight > base_height ||
5882 outpoint.
n >= std::numeric_limits<
decltype(outpoint.
n)>::max()
5885 coins_count - coins_left))};
5889 coins_count - coins_left))};
5896 if (coins_processed % 1000000 == 0) {
5897 LogInfo(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)",
5899 static_cast<float>(coins_processed) * 100 /
static_cast<float>(coins_count),
5907 if (coins_processed % 120000 == 0) {
5913 return snapshot_chainstate.GetCoinsCacheSizeState());
5926 }
catch (
const std::ios_base::failure&) {
5939 bool out_of_coins{
false};
5941 std::byte left_over_byte;
5942 coins_file >> left_over_byte;
5943 }
catch (
const std::ios_base::failure&) {
5945 out_of_coins =
true;
5947 if (!out_of_coins) {
5952 LogInfo(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s",
5966 std::optional<CCoinsStats> maybe_stats;
5974 if (!maybe_stats.has_value()) {
5995 constexpr int AFTER_GENESIS_START{1};
5997 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height(); ++i) {
5998 index = snapshot_chainstate.
m_chain[i];
6015 assert(index == snapshot_start_block);
6019 LogInfo(
"[snapshot] validated snapshot (%.2f MB)",
6046 !validated_cs.m_target_blockhash ||
6057 "%s failed to validate the -assumeutxo snapshot state. "
6058 "This indicates a hardware problem, or a bug in the software, or a "
6059 "bad software modification that allowed an invalid snapshot to be "
6060 "loaded. As a result of this, the node will shut down and stop using any "
6061 "state that was built on the snapshot, resetting the chain height "
6062 "from %d to %d. On the next "
6063 "restart, the node will resume syncing from %d "
6064 "without using any snapshot data. "
6065 "Please report this incident to %s, including how you obtained the snapshot. "
6066 "The invalid snapshot chainstate will be left on disk in case it is "
6067 "helpful in diagnosing the issue that caused this error."),
6073 LogError(
"[snapshot] deleting snapshot, reverting to validated chain, and stopping node\n");
6076 validated_cs.SetTargetBlock(
nullptr);
6080 auto rename_result = unvalidated_cs.InvalidateCoinsDBOnDisk();
6081 if (!rename_result) {
6092 if (!maybe_au_data) {
6093 LogWarning(
"[snapshot] assumeutxo data not found for height "
6094 "(%d) - refusing to validate snapshot", validated_cs.
m_chain.
Height());
6095 handle_invalid_snapshot();
6100 std::optional<CCoinsStats> validated_cs_stats;
6101 LogInfo(
"[snapshot] computing UTXO stats for background chainstate to validate "
6102 "snapshot - this could take a few minutes");
6105 CoinStatsHashType::HASH_SERIALIZED,
6106 &validated_coins_db,
6114 if (!validated_cs_stats) {
6115 LogWarning(
"[snapshot] failed to generate stats for validation coins db");
6119 handle_invalid_snapshot();
6130 LogWarning(
"[snapshot] hash mismatch: actual=%s, expected=%s",
6131 validated_cs_stats->hashSerialized.ToString(),
6133 handle_invalid_snapshot();
6137 LogInfo(
"[snapshot] snapshot beginning at %s has been fully validated",
6141 validated_cs.m_target_utxohash =
AssumeutxoHash{validated_cs_stats->hashSerialized};
6142 this->MaybeRebalanceCaches();
6153void ChainstateManager::MaybeRebalanceCaches()
6158 if (!historical_cs && !current_cs.m_from_snapshot_blockhash) {
6162 }
else if (!historical_cs) {
6164 LogInfo(
"[snapshot] allocating all cache to the snapshot chainstate");
6172 historical_cs->ResizeCoinsCaches(
6174 current_cs.ResizeCoinsCaches(
6177 current_cs.ResizeCoinsCaches(
6179 historical_cs->ResizeCoinsCaches(
6185void ChainstateManager::ResetChainstates()
6187 m_chainstates.clear();
6197 if (!opts.check_block_index.has_value()) opts.
check_block_index = opts.chainparams.DefaultConsistencyChecks();
6198 if (!opts.minimum_chain_work.has_value()) opts.minimum_chain_work =
UintToArith256(opts.chainparams.GetConsensus().nMinimumChainWork);
6199 if (!opts.assumed_valid_block.has_value()) opts.assumed_valid_block = opts.chainparams.GetConsensus().defaultAssumeValid;
6200 return std::move(opts);
6205 m_interrupt{interrupt},
6207 m_blockman{interrupt,
std::move(blockman_options)},
6208 m_validation_cache{m_options.script_execution_cache_bytes, m_options.signature_cache_bytes}
6219Chainstate* ChainstateManager::LoadAssumeutxoChainstate()
6227 if (!base_blockhash) {
6230 LogInfo(
"[snapshot] detected active snapshot chainstate (%s) - loading",
6231 fs::PathToString(*path));
6233 auto snapshot_chainstate{std::make_unique<Chainstate>(
nullptr,
m_blockman, *
this, base_blockhash)};
6234 LogInfo(
"[snapshot] switching active chainstate to %s", snapshot_chainstate->ToString());
6235 return &this->AddChainstate(std::move(snapshot_chainstate));
6238Chainstate& ChainstateManager::AddChainstate(std::unique_ptr<Chainstate> chainstate)
6243 assert(!prev_chainstate.m_target_blockhash);
6244 prev_chainstate.m_target_blockhash = chainstate->m_from_snapshot_blockhash;
6245 m_chainstates.push_back(std::move(chainstate));
6247 assert(&curr_chainstate == m_chainstates.back().get());
6251 assert(prev_chainstate.m_mempool->size() == 0);
6252 assert(!curr_chainstate.m_mempool);
6253 std::swap(curr_chainstate.m_mempool, prev_chainstate.m_mempool);
6254 return curr_chainstate;
6259 return (block_index.
nHeight==91842 && block_index.
GetBlockHash() ==
uint256{
"00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec"}) ||
6260 (block_index.
nHeight==91880 && block_index.
GetBlockHash() ==
uint256{
"00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721"});
6265 return (block_height==91722 && block_hash ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
6266 (block_height==91812 && block_hash ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"});
6278 const fs::path invalid_path{db_path +
"_INVALID"};
6281 LogInfo(
"[snapshot] renaming snapshot datadir %s to %s", db_path_str, invalid_path_str);
6287 fs::rename(db_path, invalid_path);
6288 }
catch (
const fs::filesystem_error& e) {
6289 LogError(
"While invalidating the coins db: Error renaming file '%s' -> '%s': %s",
6290 db_path_str, invalid_path_str, e.what());
6292 "Rename of '%s' -> '%s' failed. "
6293 "You should resolve this by manually moving or deleting the invalid "
6294 "snapshot directory %s, otherwise you will encounter the same error again "
6295 "on the next startup."),
6296 db_path_str, invalid_path_str, db_path_str)};
6301bool ChainstateManager::DeleteChainstate(
Chainstate& chainstate)
6307 LogError(
"Deletion of %s failed. Please remove it manually to continue reindexing.",
6308 fs::PathToString(db_path));
6313 assert(prev_chainstate->m_mempool->size() == 0);
6314 assert(!curr_chainstate.m_mempool);
6315 std::swap(curr_chainstate.m_mempool, prev_chainstate->m_mempool);
6324void ChainstateManager::RecalculateBestHeader()
6328 for (
auto& entry :
m_blockman.m_block_index) {
6329 if (!(entry.second.nStatus &
BLOCK_FAILED_MASK) && m_best_header->nChainWork < entry.second.nChainWork) {
6330 m_best_header = &entry.second;
6335bool ChainstateManager::ValidatedSnapshotCleanup(
Chainstate& validated_cs,
Chainstate& unvalidated_cs)
6343 const fs::path validated_path{validated_cs.
StoragePath()};
6344 const fs::path assumed_valid_path{unvalidated_cs.
StoragePath()};
6345 const fs::path delete_path{validated_path +
"_todelete"};
6353 this->ResetChainstates();
6354 assert(this->m_chainstates.size() == 0);
6356 LogInfo(
"[snapshot] deleting background chainstate directory (now unnecessary) (%s)",
6357 fs::PathToString(validated_path));
6359 auto rename_failed_abort = [
this](
6362 const fs::filesystem_error& err) {
6363 LogError(
"[snapshot] Error renaming path (%s) -> (%s): %s\n",
6364 fs::PathToString(p_old), fs::PathToString(p_new), err.what());
6366 "Rename of '%s' -> '%s' failed. "
6367 "Cannot clean up the background chainstate leveldb directory."),
6368 fs::PathToString(p_old), fs::PathToString(p_new)));
6372 fs::rename(validated_path, delete_path);
6373 }
catch (
const fs::filesystem_error& e) {
6374 rename_failed_abort(validated_path, delete_path, e);
6378 LogInfo(
"[snapshot] moving snapshot chainstate (%s) to "
6379 "default chainstate directory (%s)",
6380 fs::PathToString(assumed_valid_path), fs::PathToString(validated_path));
6383 fs::rename(assumed_valid_path, validated_path);
6384 }
catch (
const fs::filesystem_error& e) {
6385 rename_failed_abort(assumed_valid_path, validated_path, e);
6392 LogWarning(
"Deletion of %s failed. Please remove it manually, as the "
6393 "directory is now unnecessary.",
6394 fs::PathToString(delete_path));
6396 LogInfo(
"[snapshot] deleted background chainstate directory (%s)",
6397 fs::PathToString(validated_path));
6402std::pair<int, int> Chainstate::GetPruneRange(
int last_height_can_prune)
const
6413 prune_start =
Assert(SnapshotBase())->nHeight + 1;
6416 int max_prune = std::max<int>(
6425 int prune_end = std::min(last_height_can_prune, max_prune);
6427 return {prune_start, prune_end};
6430std::optional<std::pair<const CBlockIndex*, const CBlockIndex*>> ChainstateManager::GetHistoricalBlockRange()
const
6433 if (!chainstate)
return {};
6434 return std::make_pair(chainstate->
m_chain.
Tip(), chainstate->TargetBlock());
6443 std::vector<Chainstate*> chainstates;
6446 chainstates.reserve(m_chainstates.size());
6447 for (
const auto& chainstate : m_chainstates) {
6448 if (chainstate && chainstate->m_assumeutxo !=
Assumeutxo::INVALID && !chainstate->m_target_utxohash) {
6449 chainstates.push_back(chainstate.get());
6455 if (!chainstate->ActivateBestChain(state,
nullptr)) {
bool MoneyRange(const CAmount &nValue)
int64_t CAmount
Amount in satoshis (Can be negative)
static constexpr CAmount COIN
The amount of satoshis in one BTC.
arith_uint256 UintToArith256(const uint256 &a)
void InvalidateBlock(ChainstateManager &chainman, const uint256 block_hash)
arith_uint256 GetBlockProof(const CBlockIndex &block)
CBlockLocator GetLocator(const CBlockIndex *index)
Get a locator for a block index entry.
int64_t GetBlockProofEquivalentTime(const CBlockIndex &to, const CBlockIndex &from, const CBlockIndex &tip, const Consensus::Params ¶ms)
Return the time it would take to redo the work difference between from and to, assuming the current h...
const CBlockIndex * LastCommonAncestor(const CBlockIndex *pa, const CBlockIndex *pb)
Find the last common ancestor two blocks have.
@ BLOCK_VALID_CHAIN
Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends,...
@ BLOCK_VALID_MASK
All validity bits.
@ BLOCK_VALID_TRANSACTIONS
Only first tx is coinbase, 2 <= coinbase input script length <= 100, transactions valid,...
@ BLOCK_VALID_SCRIPTS
Scripts & signatures ok.
@ BLOCK_VALID_TREE
All parent headers found, difficulty matches, timestamp >= median previous.
@ BLOCK_HAVE_UNDO
undo data available in rev*.dat
@ BLOCK_HAVE_DATA
full block available in blk*.dat
@ BLOCK_FAILED_CHILD
descends from failed block
@ BLOCK_FAILED_VALID
stage after last reached validness failed
@ BLOCK_OPT_WITNESS
block data in blk*.dat was received with a witness-enforcing client
static constexpr int32_t SEQ_ID_BEST_CHAIN_FROM_DISK
Init values for CBlockIndex nSequenceId when loaded from disk.
static constexpr int32_t SEQ_ID_INIT_FROM_DISK
#define NONFATAL_UNREACHABLE()
NONFATAL_UNREACHABLE() is a macro that is used to mark unreachable code.
#define Assert(val)
Identity function.
#define STR_INTERNAL_BUG(msg)
#define Assume(val)
Assume is the identity function.
Non-refcounted RAII wrapper for FILE*.
std::string ToString() const
Wrapper around an AutoFile& that implements a ring buffer to deserialize from.
bool m_checked_merkle_root
std::vector< CTransactionRef > vtx
bool m_checked_witness_commitment
The block chain is a tree shaped structure starting with the genesis block at the root,...
bool IsValid(enum BlockStatus nUpTo) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
CBlockIndex * pprev
pointer to the index of the predecessor of this block
uint64_t m_chain_tx_count
(memory only) Number of transactions in the chain up to and including this block.
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
bool HaveNumChainTxs() const
Check whether this block and all previous blocks back to the genesis block or an assumeutxo snapshot ...
int32_t nSequenceId
(memory only) Sequential id assigned to distinguish order in which blocks are received.
uint256 GetBlockHash() const
int64_t GetBlockTime() const
int64_t GetMedianTimePast() const
FlatFilePos GetUndoPos() const EXCLUSIVE_LOCKS_REQUIRED(
bool RaiseValidity(enum BlockStatus nUpTo) EXCLUSIVE_LOCKS_REQUIRED(
Raise the validity level of this block index entry.
CBlockIndex * pskip
pointer to the index of some further predecessor of this block
unsigned int nTx
Number of transactions in this block.
int32_t nVersion
block header
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
int nHeight
height of the entry in the chain. The genesis block has height 0
const uint256 * phashBlock
pointer to the hash of the block, if any. Memory is owned by this CBlockIndex
Undo information for a CBlock.
std::vector< CTxUndo > vtxundo
An in-memory indexed chain of blocks.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
void SetTip(CBlockIndex &block)
Set/initialize a chain with a given tip.
CBlockIndex * Genesis() const
Returns the index entry for the genesis block of this chain, or nullptr if none.
CBlockIndex * Next(const CBlockIndex *pindex) const
Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip...
int Height() const
Return the maximal height in the chain.
const CBlockIndex * FindFork(const CBlockIndex *pindex) const
Find the last common block between this chain and a block index entry.
bool Contains(const CBlockIndex *pindex) const
Efficiently check whether a block is present in this chain.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
std::vector< int > GetAvailableSnapshotHeights() const
const ChainTxData & TxData() const
std::optional< AssumeutxoData > AssumeutxoForHeight(int height) const
CCoinsView that adds a memory cache for transactions to another CCoinsView.
bool Flush(bool will_reuse_cache=true)
Push the modifications applied to this cache to its base and wipe local state.
bool SpendCoin(const COutPoint &outpoint, Coin *moveto=nullptr)
Spend a coin.
void Uncache(const COutPoint &outpoint)
Removes the UTXO with the given outpoint from the cache, if it is not modified.
void AddCoin(const COutPoint &outpoint, Coin &&coin, bool possible_overwrite)
Add a coin.
unsigned int GetCacheSize() const
Calculate the size of the cache (in number of transaction outputs)
uint256 GetBestBlock() const override
Retrieve the block hash whose state this CCoinsView currently represents.
void SetBestBlock(const uint256 &hashBlock)
bool HaveCoinInCache(const COutPoint &outpoint) const
Check if we have the given utxo already loaded in this cache.
size_t DynamicMemoryUsage() const
Calculate the size of the cache (in bytes)
bool Sync()
Push the modifications applied to this cache to its base while retaining the contents of this cache (...
void EmplaceCoinInternalDANGER(COutPoint &&outpoint, Coin &&coin)
Emplace a coin into cacheCoins without performing any checks, marking the emplaced coin as dirty.
bool HaveCoin(const COutPoint &outpoint) const override
Just check whether a given outpoint is unspent.
const Coin & AccessCoin(const COutPoint &output) const
Return a reference to Coin in the cache, or coinEmpty if not found.
CCoinsView backed by the coin database (chainstate/)
void ResizeCache(size_t new_cache_size) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Dynamically alter the underlying leveldb cache size.
Abstract view on the open txout dataset.
virtual std::optional< Coin > GetCoin(const COutPoint &outpoint) const
Retrieve the Coin (unspent transaction output) for a given outpoint.
virtual std::vector< uint256 > GetHeadBlocks() const
Retrieve the range of blocks that may have been only partially written.
CCoinsView that brings transactions from a mempool into view.
Fee rate in satoshis per virtualbyte: CAmount / vB the feerate is represented internally as FeeFrac.
A hasher class for Bitcoin's 256-bit hash (double SHA-256).
void Finalize(std::span< unsigned char > output)
CHash256 & Write(std::span< const unsigned char > input)
An outpoint - a combination of a transaction hash and an index n into its vout.
A hasher class for SHA-256.
void Finalize(unsigned char hash[OUTPUT_SIZE])
CSHA256 & Write(const unsigned char *data, size_t len)
Closure representing one script verification Note that this stores references to the spending transac...
SignatureCache * m_signature_cache
PrecomputedTransactionData * txdata
script_verify_flags m_flags
std::optional< std::pair< ScriptError, std::string > > operator()()
const CTransaction * ptxTo
Serialized script, used inside transaction inputs and outputs.
The basic transaction that is broadcasted on the network and contained in blocks.
const std::vector< CTxOut > vout
const Wtxid & GetWitnessHash() const LIFETIMEBOUND
const Txid & GetHash() const LIFETIMEBOUND
const std::vector< CTxIn > vin
An input of a transaction.
CTxMemPool::txiter TxHandle
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it.
void check(const CCoinsViewCache &active_coins_tip, int64_t spendheight) const EXCLUSIVE_LOCKS_REQUIRED(void removeRecursive(const CTransaction &tx, MemPoolRemovalReason reason) EXCLUSIVE_LOCKS_REQUIRED(cs)
If sanity-checking is turned on, check makes sure the pool is consistent (does not contain two transa...
void UpdateTransactionsFromBlock(const std::vector< Txid > &vHashesToUpdate) EXCLUSIVE_LOCKS_REQUIRED(cs
UpdateTransactionsFromBlock is called when adding transactions from a disconnected block back to the ...
void AddTransactionsUpdated(unsigned int n)
CTransactionRef get(const Txid &hash) const
size_t DynamicMemoryUsage() const
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...
bool exists(const Txid &txid) const
std::set< txiter, CompareIteratorByHash > setEntries
void removeForBlock(const std::vector< CTransactionRef > &vtx, unsigned int nBlockHeight) EXCLUSIVE_LOCKS_REQUIRED(cs)
indexed_transaction_set::nth_index< 0 >::type::const_iterator txiter
unsigned long size() const
An output of a transaction.
Undo information for a CTransaction.
std::vector< Coin > vprevout
VerifyDBResult VerifyDB(Chainstate &chainstate, const Consensus::Params &consensus_params, CCoinsView &coinsview, int nCheckLevel, int nCheckDepth) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
kernel::Notifications & m_notifications
CVerifyDB(kernel::Notifications ¬ifications)
Chainstate stores and provides an API to update our local knowledge of the current best chain.
void InitCoinsCache(size_t cache_size_bytes) EXCLUSIVE_LOCKS_REQUIRED(bool CanFlushToDisk() const EXCLUSIVE_LOCKS_REQUIRED(
Initialize the in-memory coins cache (to be done after the health of the on-disk database is verified...
Mutex m_chainstate_mutex
The ChainState Mutex A lock that must be held when modifying this ChainState - held in ActivateBestCh...
CChain m_chain
The current chain of blockheaders we consult and build on.
CTxMemPool * GetMempool()
bool RollforwardBlock(const CBlockIndex *pindex, CCoinsViewCache &inputs) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Apply the effects of a block on the utxo cache, ignoring that it may already have been applied.
size_t m_coinstip_cache_size_bytes
The cache size of the in-memory coins view.
void UpdateTip(const CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(NodeClock::time_poin m_next_write)
Check warning conditions and do some notifications on new chain tip set.
CCoinsViewCache & CoinsTip() EXCLUSIVE_LOCKS_REQUIRED(
bool LoadChainTip() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Update the chain tip based on database information, i.e.
size_t m_coinsdb_cache_size_bytes
The cache size of the on-disk coins view.
bool PreciousBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(bool InvalidateBlock(BlockValidationState &state, CBlockIndex *pindex) LOCKS_EXCLUDED(void SetBlockFailureFlags(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(voi ResetBlockFailureFlags)(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as precious and reorganize.
void InvalidBlockFound(CBlockIndex *pindex, const BlockValidationState &state) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
void CheckForkWarningConditions() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
const CBlockIndex *SnapshotBase() const EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *TargetBlock() const EXCLUSIVE_LOCKS_REQUIRED(void SetTargetBlock(CBlockIndex *block) EXCLUSIVE_LOCKS_REQUIRED(void SetTargetBlockHash(uint256 block_hash) EXCLUSIVE_LOCKS_REQUIRED(boo ReachedTarget)() const EXCLUSIVE_LOCKS_REQUIRED(
The base of the snapshot this chainstate was created from.
void ForceFlushStateToDisk()
Unconditionally flush all changes to disk.
bool ConnectTip(BlockValidationState &state, CBlockIndex *pindexNew, std::shared_ptr< const CBlock > block_to_connect, ConnectTrace &connectTrace, DisconnectedBlockTransactions &disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Connect a new block to m_chain.
bool LoadGenesisBlock()
Ensures we have a genesis block in the block tree, possibly writing one to disk.
kernel::ChainstateRole GetRole() const EXCLUSIVE_LOCKS_REQUIRED(void InitCoinsDB(size_t cache_size_bytes, bool in_memory, bool should_wipe)
Return the current role of the chainstate.
const std::optional< uint256 > m_from_snapshot_blockhash
The blockhash which is the base of the snapshot this chainstate was created from.
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.
CTxMemPool * m_mempool
Optional mempool that is kept in sync with the chain.
CCoinsViewDB & CoinsDB() EXCLUSIVE_LOCKS_REQUIRED(
bool DisconnectTip(BlockValidationState &state, DisconnectedBlockTransactions *disconnectpool) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Disconnect m_chain's tip.
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...
std::set< CBlockIndex *, node::CBlockIndexWorkComparator > setBlockIndexCandidates
The set of all CBlockIndex entries that have as much work as our current tip or more,...
ChainstateManager & m_chainman
The chainstate manager that owns this chainstate.
std::unique_ptr< CoinsViews > m_coins_views
Manages the UTXO set, which is a reflection of the contents of m_chain.
bool ReplayBlocks()
Replay blocks that aren't fully applied to the database.
void PruneBlockIndexCandidates()
Delete all entries in setBlockIndexCandidates that are worse than the current tip.
void TryAddBlockIndexCandidate(CBlockIndex *pindex) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
void PruneAndFlush()
Prune blockfiles from the disk if necessary and then flush chainstate changes if we pruned.
bool ResizeCoinsCaches(size_t coinstip_size, size_t coinsdb_size) EXCLUSIVE_LOCKS_REQUIRED(bool FlushStateToDisk(BlockValidationState &state, FlushStateMode mode, int nManualPruneHeight=0)
Resize the CoinsViews caches dynamically and flush state to disk.
node::BlockManager & m_blockman
Reference to a BlockManager instance which itself is shared across all Chainstate instances.
void 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...
bool ActivateBestChainStep(BlockValidationState &state, CBlockIndex *pindexMostWork, const std::shared_ptr< const CBlock > &pblock, bool &fInvalidFound, ConnectTrace &connectTrace) EXCLUSIVE_LOCKS_REQUIRED(cs_main
Return the [start, end] (inclusive) of block heights we can prune.
void ClearBlockIndexCandidates() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex * FindForkInGlobalIndex(const CBlockLocator &locator) const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Find the last common block of this chain and a locator.
void InvalidChainFound(CBlockIndex *pindexNew) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Chainstate(CTxMemPool *mempool, node::BlockManager &blockman, ChainstateManager &chainman, std::optional< uint256 > from_snapshot_blockhash=std::nullopt)
fs::path StoragePath() const
Return path to chainstate leveldb directory.
bool NeedsRedownload() const EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Whether the chain state needs to be redownloaded due to lack of witness data.
CoinsCacheSizeState GetCoinsCacheSizeState() EXCLUSIVE_LOCKS_REQUIRED(CoinsCacheSizeState GetCoinsCacheSizeState(size_t max_coins_cache_size_bytes, size_t max_mempool_size_bytes) EXCLUSIVE_LOCKS_REQUIRED(std::string ToString() EXCLUSIVE_LOCKS_REQUIRED(RecursiveMutex * MempoolMutex() const LOCK_RETURNED(m_mempool -> cs)
Dictates whether we need to flush the cache to disk or not.
Interface for managing multiple Chainstate objects, where each chainstate is associated with chainsta...
util::Result< void > PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
Chainstate * HistoricalChainstate() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Return historical chainstate targeting a specific block, if any.
const uint256 & AssumedValidBlock() const
ValidationCache m_validation_cache
double GuessVerificationProgress(const CBlockIndex *pindex) const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Guess verification progress (as a fraction between 0.0=genesis and 1.0=current tip).
Chainstate *LoadAssumeutxoChainstate() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &AddChainstate(std::unique_ptr< Chainstate > chainstate) EXCLUSIVE_LOCKS_REQUIRED(void ResetChainstates() EXCLUSIVE_LOCKS_REQUIRED(bool DeleteChainstate(Chainstate &chainstate) EXCLUSIVE_LOCKS_REQUIRED(bool ValidatedSnapshotCleanup(Chainstate &validated_cs, Chainstate &unvalidated_cs) EXCLUSIVE_LOCKS_REQUIRED(std::optional< std::pair< const CBlockIndex *, const CBlockIndex * > > GetHistoricalBlockRange() const EXCLUSIVE_LOCKS_REQUIRED(util::Result< void > ActivateBestChains() LOCKS_EXCLUDED(void RecalculateBestHeader() 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...
size_t m_total_coinstip_cache
The total number of bytes available for us to use across all in-memory coins caches.
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...
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.
std::unique_ptr< Chainstate > RemoveChainstate(Chainstate &chainstate) EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Remove a chainstate.
kernel::Notifications & GetNotifications() const
bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network)
void ReceivedBlockTransactions(const CBlock &block, CBlockIndex *pindexNew, const FlatFilePos &pos) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Mark a block as having its data received and checked (up to BLOCK_VALID_TRANSACTIONS).
bool ShouldCheckBlockIndex() const
RecursiveMutex & GetMutex() const LOCK_RETURNED(
Alias for cs_main.
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Chainstate & ActiveChainstate() const
Alternatives to CurrentChainstate() used by older code to query latest chainstate information without...
SnapshotCompletionResult MaybeValidateSnapshot(Chainstate &validated_cs, Chainstate &unvalidated_cs) EXCLUSIVE_LOCKS_REQUIRED(Chainstate & CurrentChainstate() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
Try to validate an assumeutxo snapshot by using a validated historical chainstate targeted at the sna...
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.
size_t m_total_coinsdb_cache
The total number of bytes available for us to use across all leveldb coins databases.
std::atomic< bool > m_cached_finished_ibd
Whether initial block download has ended and IsInitialBlockDownload should return false from now on.
void CheckBlockIndex() const
Make various assertions about the state of the block index.
const util::SignalInterrupt & m_interrupt
void LoadExternalBlockFile(AutoFile &file_in, FlatFilePos *dbp=nullptr, std::multimap< uint256, FlatFilePos > *blocks_with_unknown_parent=nullptr)
Import blocks from an external file.
int ActiveHeight() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
VersionBitsCache m_versionbitscache
Track versionbit status.
std::function< void()> snapshot_download_completed
Function to restart active indexes; set dynamically to avoid a circular dependency on base/index....
const CChainParams & GetParams() const
bool ProcessNewBlockHeaders(std::span< const CBlockHeader > headers, bool min_pow_checked, BlockValidationState &state, const CBlockIndex **ppindex=nullptr) LOCKS_EXCLUDED(cs_main)
Process incoming block headers.
const Consensus::Params & GetConsensus() const
ChainstateManager(const util::SignalInterrupt &interrupt, Options options, node::BlockManager::Options blockman_options)
const arith_uint256 & MinimumChainWork() const
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 &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(util::Result< CBlockIndex * ActivateSnapshot)(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Instantiate a new chainstate.
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
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...
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 ...
arith_uint256 nLastPreciousChainwork
chainwork for the last block that preciousblock has been applied to.
bool NotifyHeaderTip() LOCKS_EXCLUDED(GetMutex())
void MaybeRebalanceCaches() EXCLUSIVE_LOCKS_REQUIRED(void UpdateUncommittedBlockStructures(CBlock &block, const CBlockIndex *pindexPrev) const
Check to see if caches are out of balance and if so, call ResizeCoinsCaches() as needed.
int32_t nBlockReverseSequenceId
Decreasing counter (used by subsequent preciousblock calls).
node::BlockManager m_blockman
A single BlockManager instance is shared across each constructed chainstate to avoid duplicating bloc...
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).
CTxOut out
unspent transaction output
bool IsSpent() const
Either this coin never existed (see e.g.
uint32_t nHeight
at which height this containing transaction was included in the active block chain
unsigned int fCoinBase
whether containing transaction was a coinbase
CoinsViews(DBParams db_params, CoinsViewOptions options)
This constructor initializes CCoinsViewDB and CCoinsViewErrorCatcher instances, but it does not creat...
Used to track blocks whose transactions were applied to the UTXO state as a part of a single Activate...
std::vector< PerBlockConnectTrace > & GetBlocksConnected()
std::vector< PerBlockConnectTrace > blocksConnected
void BlockConnected(CBlockIndex *pindex, std::shared_ptr< const CBlock > pblock)
std::pair< uint32_t, size_t > setup_bytes(size_t bytes)
setup_bytes is a convenience function which accounts for internal memory usage when deciding how many...
void insert(Element e)
insert loops at most depth_limit times trying to insert a hash at various locations in the table via ...
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.
DisconnectedBlockTransactions.
std::list< CTransactionRef > take()
Clear all data structures and return the list of transactions.
void removeForBlock(const std::vector< CTransactionRef > &vtx)
Remove any entries that are in this block.
std::vector< CTransactionRef > AddTransactionsFromBlock(const std::vector< CTransactionRef > &vtx)
Add transactions from the block, iterating through vtx in reverse order.
Tp rand_uniform_delay(const Tp &time, typename Tp::duration range) noexcept
Return the time point advanced by a uniform random duration.
Convenience class for initializing and passing the script execution cache and signature cache.
ValidationCache(size_t script_execution_cache_bytes, size_t signature_cache_bytes)
CuckooCache::cache< uint256, SignatureCacheHasher > m_script_execution_cache
CSHA256 ScriptExecutionCacheHasher() const
Return a copy of the pre-initialized hasher.
CSHA256 m_script_execution_cache_hasher
Pre-initialized hasher to avoid having to recreate it for every hash calculation.
SignatureCache m_signature_cache
void BlockConnected(const kernel::ChainstateRole &, const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockChecked(const std::shared_ptr< const CBlock > &, const BlockValidationState &)
void ChainStateFlushed(const kernel::ChainstateRole &, const CBlockLocator &)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void ActiveTipChange(const CBlockIndex &, bool)
std::string GetRejectReason() const
std::string GetDebugMessage() const
bool Error(const std::string &reject_reason)
bool Invalid(Result result, const std::string &reject_reason="", const std::string &debug_message="")
std::string ToString() const
std::vector< std::pair< int, bool > > CheckUnknownActivations(const CBlockIndex *pindex, const CChainParams &chainparams) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Check for unknown activations Returns a vector containing the bit number used for signalling and a bo...
void Clear() EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
256-bit unsigned big integer.
constexpr bool IsNull() const
std::string ToString() const
constexpr unsigned char * begin()
A base class defining functions for notifying about certain kernel events.
virtual void headerTip(SynchronizationState state, int64_t height, int64_t timestamp, bool presync)
virtual void fatalError(const bilingual_str &message)
The fatal error notification is sent to notify the user when an error occurs in kernel code that can'...
virtual void warningSet(Warning id, const bilingual_str &message)
virtual void progress(const bilingual_str &title, int progress_percent, bool resume_possible)
virtual InterruptResult blockTip(SynchronizationState state, const CBlockIndex &index, double verification_progress)
virtual void warningUnset(Warning id)
Maintains a tree of blocks (stored in m_block_index) which is consulted to determine where the most-w...
const kernel::BlockManagerOpts m_opts
RecursiveMutex cs_LastBlockFile
bool FlushChainstateBlockFile(int tip_height)
void FindFilesToPrune(std::set< int > &setFilesToPrune, int last_prune, const Chainstate &chain, ChainstateManager &chainman)
Prune block and undo files (blk???.dat and rev???.dat) so that the disk space used is less than a use...
void UpdateBlockInfo(const CBlock &block, unsigned int nHeight, const FlatFilePos &pos)
Update blockfile info while processing a block during reindex.
CBlockIndex * LookupBlockIndex(const uint256 &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool ReadBlockUndo(CBlockUndo &blockundo, const CBlockIndex &index) const
std::atomic_bool m_blockfiles_indexed
Whether all blockfiles have been added to the block tree database.
std::vector< CBlockIndex * > GetAllBlockIndices() EXCLUSIVE_LOCKS_REQUIRED(std::multimap< CBlockIndex *, CBlockIndex * > m_blocks_unlinked
All pairs A->B, where A (or one of its ancestors) misses transactions, but B has transactions.
std::set< CBlockIndex * > m_dirty_blockindex
Dirty block index entries.
bool LoadingBlocks() const
void UnlinkPrunedFiles(const std::set< int > &setFilesToPrune) const
Actually unlink the specified files.
void 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.
bool ReadBlock(CBlock &block, const FlatFilePos &pos, const std::optional< uint256 > &expected_hash) const
Functions for disk access for blocks.
bool m_check_for_pruning
Global flag to indicate we should check to see if there are block/undo files that should be deleted.
bool IsPruneMode() const
Whether running in -prune mode.
bool CheckBlockDataAvailability(const CBlockIndex &upper_block, const CBlockIndex &lower_block) EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex &GetFirstBlock(const CBlockIndex &upper_block LIFETIMEBOUND, uint32_t status_mask, const CBlockIndex *lower_block LIFETIMEBOUND=nullptr) const EXCLUSIVE_LOCKS_REQUIRED(boo m_have_pruned)
Check if all blocks in the [upper_block, lower_block] range have data available.
bool WriteBlockUndo(const CBlockUndo &blockundo, BlockValidationState &state, CBlockIndex &block) EXCLUSIVE_LOCKS_REQUIRED(FlatFilePos WriteBlock(const CBlock &block, int nHeight)
Store block on disk and update block file statistics.
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, const Chainstate &chain)
std::optional< int > m_snapshot_height
The height of the base block of an assumeutxo snapshot, if one is in use.
std::string ToString() const
constexpr const std::byte * begin() const
const uint256 & ToUint256() const LIFETIMEBOUND
std::string GetHex() const
Helper class that manages an interrupt flag, and allows a thread or signal to interrupt another threa...
std::string FormatFullVersion()
const Coin & AccessByTxid(const CCoinsViewCache &view, const Txid &txid)
Utility function to find any unspent output with a given txid.
void AddCoins(CCoinsViewCache &cache, const CTransaction &tx, int nHeight, bool check_for_overwrite)
Utility function to add all of a transaction's outputs to a cache.
uint256 BlockMerkleRoot(const CBlock &block, bool *mutated)
uint256 BlockWitnessMerkleRoot(const CBlock &block)
static constexpr int NO_WITNESS_COMMITMENT
Index marker for when no witness commitment is present in a coinbase transaction.
static constexpr size_t MINIMUM_WITNESS_COMMITMENT
Minimum size of a witness commitment structure.
static int64_t GetBlockWeight(const CBlock &block)
@ BLOCK_HEADER_LOW_WORK
the block header may be on a too-little-work chain
@ BLOCK_INVALID_HEADER
invalid proof of work or time too old
@ BLOCK_CACHED_INVALID
this block was cached as being invalid and we didn't store the reason why
@ BLOCK_CONSENSUS
invalid by consensus rules (excluding any below reasons)
@ BLOCK_MISSING_PREV
We don't have the previous block the checked one is built on.
@ BLOCK_INVALID_PREV
A block this one builds on is invalid.
@ BLOCK_MUTATED
the block's data didn't match the data committed to by the PoW
@ BLOCK_TIME_FUTURE
block timestamp was > 2 hours in the future (or our clock is bad)
int GetWitnessCommitmentIndex(const CBlock &block)
Compute at which vout of the block's coinbase transaction the witness commitment occurs,...
@ TX_MISSING_INPUTS
transaction was missing some of its inputs
@ TX_MEMPOOL_POLICY
violated mempool's fee/size/descendant/RBF/etc limits
@ TX_PREMATURE_SPEND
transaction spends a coinbase too early, or violates locktime/sequence locks
@ TX_INPUTS_NOT_STANDARD
inputs (covered by txid) failed policy rules
@ TX_WITNESS_STRIPPED
Transaction is missing a witness.
@ TX_CONFLICT
Tx already in mempool or conflicts with a tx in the chain (if it conflicts with another tx in mempool...
@ TX_NOT_STANDARD
otherwise didn't meet our local policy rules
@ TX_WITNESS_MUTATED
Transaction might have a witness prior to SegWit activation, or witness may have been malleated (whic...
@ TX_NO_MEMPOOL
this node does not have a mempool so can't validate the transaction
@ TX_CONSENSUS
invalid by consensus rules
@ TX_RECONSIDERABLE
fails some policy, but might be acceptable if submitted in a (different) package
static constexpr unsigned int LOCKTIME_VERIFY_SEQUENCE
Flags for nSequence and nLockTime locks.
static constexpr int64_t MAX_TIMEWARP
Maximum number of seconds that the timestamp of the first block of a difficulty adjustment period is ...
static const unsigned int MAX_BLOCK_WEIGHT
The maximum allowed weight for a block, see BIP 141 (network rule)
static const unsigned int MAX_BLOCK_SERIALIZED_SIZE
The maximum allowed size for a serialized block, in bytes (only for buffer size limits)
static const int64_t MAX_BLOCK_SIGOPS_COST
The maximum allowed number of signature check operations in a block (network rule)
static const int COINBASE_MATURITY
Coinbase transaction outputs can only be spent after this number of new blocks (network rule)
static const int WITNESS_SCALE_FACTOR
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
bool DestroyDB(const std::string &path_str)
bool DeploymentActiveAfter(const CBlockIndex *pindexPrev, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, VersionBitsCache &versionbitscache)
Determine if a deployment is active for the next block.
bool DeploymentActiveAt(const CBlockIndex &index, const Consensus::Params ¶ms, Consensus::BuriedDeployment dep, VersionBitsCache &versionbitscache)
Determine if a deployment is active for this block.
static const unsigned int MAX_DISCONNECTED_TX_POOL_BYTES
Maximum bytes for transactions to store for processing during reorg.
bool CheckEphemeralSpends(const Package &package, CFeeRate dust_relay_rate, const CTxMemPool &tx_pool, TxValidationState &out_child_state, Wtxid &out_child_wtxid)
Must be called for each transaction(package) if any dust is in the package.
bool PreCheckEphemeralTx(const CTransaction &tx, CFeeRate dust_relay_rate, CAmount base_fee, CAmount mod_fee, TxValidationState &state)
These utility functions ensure that ephemeral dust is safely created and spent without unduly risking...
static bool exists(const path &p)
static std::string PathToString(const path &path)
Convert path object to a byte string.
bool CheckDiskSpace(const fs::path &dir, uint64_t additional_bytes)
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, const CScriptWitness *witness, script_verify_flags flags, const BaseSignatureChecker &checker, ScriptError *serror)
@ SCRIPT_VERIFY_NULLDUMMY
@ SCRIPT_VERIFY_CHECKSEQUENCEVERIFY
@ SCRIPT_VERIFY_CHECKLOCKTIMEVERIFY
is a home for simple enum and struct type definitions that can be used internally by functions in the...
#define LogPrintLevel_(category, level, should_ratelimit,...)
#define LogDebug(category,...)
@ REORG
Removed for reorganization.
std::array< uint8_t, 4 > MessageStartChars
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...
std::function< FILE *(const fs::path &, const char *)> FopenFn
bool IsInterrupted(const T &result)
@ UNKNOWN_NEW_RULES_ACTIVATED
@ LARGE_WORK_INVALID_CHAIN
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 fs::path SNAPSHOT_BLOCKHASH_FILENAME
The file in the snapshot chainstate dir which stores the base blockhash.
bool WriteSnapshotBaseBlockhash(Chainstate &snapshot_chainstate)
std::optional< fs::path > FindAssumeutxoChainstateDir(const fs::path &data_dir)
Return a path to the snapshot-based chainstate dir, if one exists.
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.
std::unordered_map< uint256, CBlockIndex, BlockHasher > BlockMap
std::optional< uint256 > ReadSnapshotBaseBlockhash(fs::path chaindir)
bilingual_str ErrorString(const Result< T > &result)
std::string ToString(const T &t)
Locale-independent version of std::to_string.
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
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.
std::shared_ptr< Chain::Notifications > m_notifications
bool IsChildWithParents(const Package &package)
Context-free check that a package is exactly one child and its parents; not all parents need to be pr...
bool IsWellFormedPackage(const Package &txns, PackageValidationState &state, bool require_sorted)
Context-free package policy checks:
uint256 GetPackageHash(const std::vector< CTransactionRef > &transactions)
Get the hash of the concatenated wtxids of transactions, with wtxids treated as a little-endian numbe...
std::vector< CTransactionRef > Package
A package is an ordered list of transactions.
@ PCKG_POLICY
The package itself is invalid (e.g. too many transactions).
@ PCKG_MEMPOOL_ERROR
Mempool logic error.
@ PCKG_TX
At least one tx is invalid.
std::optional< std::pair< DiagramCheckError, std::string > > ImprovesFeerateDiagram(CTxMemPool::ChangeSet &changeset)
The replacement transaction must improve the feerate diagram of the mempool.
std::optional< std::string > PaysForRBF(CAmount original_fees, CAmount replacement_fees, size_t replacement_vsize, CFeeRate relay_fee, const Txid &txid)
The replacement transaction must pay more fees than the original transactions.
std::optional< std::string > EntriesAndTxidsDisjoint(const CTxMemPool::setEntries &ancestors, const std::set< Txid > &direct_conflicts, const Txid &txid)
Check the intersection between two sets of transactions (a set of mempool entries and a set of txids)...
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.
@ FAILURE
New diagram wasn't strictly superior
bool SpendsNonAnchorWitnessProg(const CTransaction &tx, const CCoinsViewCache &prevouts)
Check whether this transaction spends any witness program but P2A, including not-yet-defined ones.
bool AreInputsStandard(const CTransaction &tx, const CCoinsViewCache &mapInputs)
Check transaction inputs.
bool IsWitnessStandard(const CTransaction &tx, const CCoinsViewCache &mapInputs)
Check if the transaction is over standard P2WSH resources limit: 3600bytes witnessScript size,...
bool IsStandardTx(const CTransaction &tx, const std::optional< unsigned > &max_datacarrier_bytes, bool permit_bare_multisig, const CFeeRate &dust_relay_fee, std::string &reason)
Check for standard transaction types.
static constexpr script_verify_flags STANDARD_SCRIPT_VERIFY_FLAGS
Standard script verification flags that standard transactions will comply with.
static constexpr unsigned int STANDARD_LOCKTIME_VERIFY_FLAGS
Used as the flags parameter to sequence and nLocktime checks in non-consensus code.
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 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
static constexpr script_verify_flags STANDARD_NOT_MANDATORY_VERIFY_FLAGS
For convenience, standard but not mandatory verify flags.
unsigned int GetNextWorkRequired(const CBlockIndex *pindexLast, const CBlockHeader *pblock, const Consensus::Params ¶ms)
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.
static constexpr TransactionSerParams TX_NO_WITNESS
static constexpr TransactionSerParams TX_WITH_WITNESS
static CTransactionRef MakeTransactionRef(Tx &&txIn)
std::shared_ptr< const CTransaction > CTransactionRef
uint256 GetRandHash() noexcept
Generate a random uint256.
std::string ScriptErrorString(const ScriptError serror)
enum ScriptError_t ScriptError
@ SCRIPT_ERR_UNKNOWN_ERROR
uint64_t ReadCompactSize(Stream &is, bool range_check=true)
Decode a CompactSize-encoded variable-length integer.
uint64_t GetSerializeSize(const T &t)
bool CheckSignetBlockSolution(const CBlock &block, const Consensus::Params &consensusParams)
Extract signature and check whether a block has a valid solution.
unsigned char * UCharCast(char *c)
Holds configuration for use during UTXO snapshot load and validation.
AssumeutxoHash hash_serialized
The expected hash of the deserialized UTXO set.
uint64_t m_chain_tx_count
Used to populate the m_chain_tx_count value, which is used during BlockManager::LoadBlockIndex().
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
std::vector< uint256 > vHave
A mutable version of CTransaction.
std::vector< CTxOut > vout
Holds various statistics on transactions within a chain.
User-controlled performance and debug options.
Parameters that influence chain consensus.
bool enforce_BIP94
Enforce BIP94 timewarp attack mitigation.
int64_t DifficultyAdjustmentInterval() const
bool signet_blocks
If true, witness commitments contain a payload equal to a Bitcoin Script solution to the signet chall...
int nSubsidyHalvingInterval
std::map< uint256, script_verify_flags > script_flag_exceptions
Hashes of blocks that.
int64_t nPowTargetSpacing
std::chrono::seconds PowTargetSpacing() const
Application-specific storage settings.
fs::path path
Location in the filesystem where leveldb data will be stored.
Data structure storing a fee and size, ordered by increasing fee/size.
Validation result for a transaction evaluated by MemPoolAccept (single or package).
const ResultType m_result_type
Result type.
const TxValidationState m_state
Contains information about why the transaction failed.
@ INVALID
‍Fully validated, valid.
static MempoolAcceptResult Failure(TxValidationState state)
static MempoolAcceptResult FeeFailure(TxValidationState state, CFeeRate effective_feerate, const std::vector< Wtxid > &wtxids_fee_calculations)
static MempoolAcceptResult MempoolTxDifferentWitness(const Wtxid &other_wtxid)
static MempoolAcceptResult MempoolTx(int64_t vsize, CAmount fees)
static MempoolAcceptResult Success(std::list< CTransactionRef > &&replaced_txns, int64_t vsize, CAmount fees, CFeeRate effective_feerate, const std::vector< Wtxid > &wtxids_fee_calculations)
static time_point now() noexcept
Return current system time or mocked time, if set.
static time_point now() noexcept
Return current system time or mocked time, if set.
Validation result for package mempool acceptance.
std::shared_ptr< const CBlock > pblock
PerBlockConnectTrace()=default
void Init(const T &tx, std::vector< CTxOut > &&spent_outputs, bool force=false)
Initialize this PrecomputedTransactionData with transaction data.
bool m_spent_outputs_ready
Whether m_spent_outputs is initialized.
std::vector< CTxOut > m_spent_outputs
const char * what() const noexcept override
An options struct for BlockManager, more ergonomically referred to as BlockManager::Options due to th...
const fs::path blocks_dir
An options struct for ChainstateManager, more ergonomically referred to as ChainstateManager::Options...
ValidationSignals * signals
std::optional< int32_t > check_block_index
std::chrono::seconds max_tip_age
If the tip is older than this, the node is considered to be in initial block download.
const CChainParams & chainparams
CoinsViewOptions coins_view
Information about chainstate that notifications are sent from.
bool validated
Whether this is a notification from a chainstate that's been fully validated starting from the genesi...
#define AssertLockNotHeld(cs)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define EXCLUSIVE_LOCKS_REQUIRED(...)
#define LOCKS_EXCLUDED(...)
#define LOG_TIME_MILLIS_WITH_CATEGORY(end_msg, log_category)
#define LOG_TIME_MILLIS_WITH_CATEGORY_MSG_ONCE(end_msg, log_category)
#define TRACEPOINT(context,...)
consteval auto _(util::TranslatedLiteral str)
bilingual_str Untranslated(std::string original)
Mark a bilingual_str as untranslated.
std::optional< std::pair< std::string, CTransactionRef > > SingleTRUCChecks(const CTxMemPool &pool, const CTransactionRef &ptx, const std::vector< CTxMemPoolEntry::CTxMemPoolEntryRef > &mempool_parents, const std::set< Txid > &direct_conflicts, int64_t vsize)
Must be called for every transaction, even if not TRUC.
std::optional< std::string > PackageTRUCChecks(const CTxMemPool &pool, const CTransactionRef &ptx, int64_t vsize, const Package &package, const std::vector< CTxMemPoolEntry::CTxMemPoolEntryRef > &mempool_parents)
Must be called for every transaction that is submitted within a package, even if not TRUC.
static constexpr decltype(CTransaction::version) TRUC_VERSION
bool CheckTransaction(const CTransaction &tx, TxValidationState &state)
bool EvaluateSequenceLocks(const CBlockIndex &block, std::pair< int, int64_t > lockPair)
std::pair< int, int64_t > CalculateSequenceLocks(const CTransaction &tx, int flags, std::vector< int > &prevHeights, const CBlockIndex &block)
Calculates the block height and previous block's median time past at which the transaction will be co...
int64_t GetTransactionSigOpCost(const CTransaction &tx, const CCoinsViewCache &inputs, script_verify_flags flags)
Compute total signature operation cost of a transaction.
unsigned int GetLegacySigOpCount(const CTransaction &tx)
Auxiliary functions for transaction validation (ideally should not be exposed)
bool SequenceLocks(const CTransaction &tx, int flags, std::vector< int > &prevHeights, const CBlockIndex &block)
Check if transaction is final per BIP 68 sequence numbers and can be included in a block.
bool 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 TestLockPointValidity(CChain &active_chain, const LockPoints &lp)
Test whether the LockPoints height and time are still valid on the current chain.
static const uint32_t MEMPOOL_HEIGHT
Fake height value used in Coin to signify they are only in the memory pool (since 0....
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
std::string FormatISO8601DateTime(int64_t nTime)
ISO 8601 formatting is preferred.
constexpr int64_t count_seconds(std::chrono::seconds t)
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds
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.
static void LimitMempoolSize(CTxMemPool &pool, CCoinsViewCache &coins_cache) EXCLUSIVE_LOCKS_REQUIRED(
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).
script_verify_flags GetBlockScriptFlags(const CBlockIndex &block_index, const ChainstateManager &chainman)
std::optional< LockPoints > CalculateLockPointsAtTip(CBlockIndex *tip, const CCoinsView &coins_view, const CTransaction &tx)
bool CheckInputScripts(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &inputs, script_verify_flags 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.
static constexpr size_t WARN_FLUSH_COINS_SIZE
Size threshold for warning about slow UTXO set flush to disk.
bool CheckFinalTxAtTip(const CBlockIndex &active_chain_tip, const CTransaction &tx)
CAmount GetBlockSubsidy(int nHeight, const Consensus::Params &consensusParams)
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.
int ApplyTxInUndo(Coin &&undo, CCoinsViewCache &view, const COutPoint &out)
Restore the UTXO in a Coin at a given COutPoint.
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 FatalError(Notifications ¬ifications, BlockValidationState &state, const bilingual_str &message)
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 void UpdateTipLog(const ChainstateManager &chainman, const CCoinsViewCache &coins_tip, const CBlockIndex *tip, const std::string &func_name, const std::string &prefix, const std::string &warning_messages) EXCLUSIVE_LOCKS_REQUIRED(
bool HasValidProofOfWork(const std::vector< CBlockHeader > &headers, const Consensus::Params &consensusParams)
Check with the proof of work on each blockheader matches the value in nBits.
static ChainstateManager::Options && Flatten(ChainstateManager::Options &&opts)
Apply default chain params to nullopt members.
static bool CheckInputsFromMempoolAndCache(const CTransaction &tx, TxValidationState &state, const CCoinsViewCache &view, const CTxMemPool &pool, script_verify_flags 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...
static constexpr auto DATABASE_WRITE_INTERVAL_MAX
static bool CheckWitnessMalleation(const CBlock &block, bool expect_witness_commitment, BlockValidationState &state)
CheckWitnessMalleation performs checks for block malleation with regard to its witnesses.
void UpdateCoins(const CTransaction &tx, CCoinsViewCache &inputs, CTxUndo &txundo, int nHeight)
static bool DeleteCoinsDBFromDisk(const fs::path db_path, bool is_snapshot) EXCLUSIVE_LOCKS_REQUIRED(
static bool CheckMerkleRoot(const CBlock &block, BlockValidationState &state)
static constexpr int PRUNE_LOCK_BUFFER
The number of blocks to keep below the deepest prune lock.
arith_uint256 CalculateClaimedHeadersWork(std::span< const CBlockHeader > headers)
Return the sum of the claimed work on a given set of headers.
const std::vector< std::string > CHECKLEVEL_DOC
Documentation for argument 'checklevel'.
bool CheckBlock(const CBlock &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW, bool fCheckMerkleRoot)
Functions for validating blocks and updating the block tree.
static constexpr std::chrono::hours MAX_FEE_ESTIMATION_TIP_AGE
Maximum age of our tip for us to be considered current for fee estimation.
void PruneBlockFilesManual(Chainstate &active_chainstate, int nManualPruneHeight)
Prune block files up to a given height.
static void FlushSnapshotToDisk(CCoinsViewCache &coins_cache, bool snapshot_loaded)
static bool IsCurrentForFeeEstimation(Chainstate &active_chainstate) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
static constexpr auto DATABASE_WRITE_INTERVAL_MIN
Time window to wait between writing blocks/block index and chainstate to disk.
BlockValidationState TestBlockValidity(Chainstate &chainstate, const CBlock &block, const bool check_pow, const bool check_merkle_root)
Verify a block, including transactions.
static bool CheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, const Consensus::Params &consensusParams, bool fCheckPOW=true)
bool IsBIP30Repeat(const CBlockIndex &block_index)
Identifies blocks that overwrote an existing coinbase output in the UTXO set (see BIP30)
static void SnapshotUTXOHashBreakpoint(const util::SignalInterrupt &interrupt)
static SynchronizationState GetSynchronizationState(bool init, bool blockfiles_indexed)
static bool ContextualCheckBlockHeader(const CBlockHeader &block, BlockValidationState &state, BlockManager &blockman, const ChainstateManager &chainman, const CBlockIndex *pindexPrev) EXCLUSIVE_LOCKS_REQUIRED(
Context-dependent validity checks.
bool IsBIP30Unspendable(const uint256 &block_hash, int block_height)
Identifies blocks which coinbase output was subsequently overwritten in the UTXO set (see BIP30)
TRACEPOINT_SEMAPHORE(validation, block_connected)
static void LimitValidationInterfaceQueue(ValidationSignals &signals) LOCKS_EXCLUDED(cs_main)
static constexpr int MAX_SCRIPTCHECK_THREADS
Maximum number of dedicated script-checking threads allowed.
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...
Assumeutxo
Chainstate assumeutxo validity.
@ VALIDATED
Every block in the chain has been validated.
@ UNVALIDATED
Blocks after an assumeutxo snapshot have been validated but the snapshot itself has not been validate...
@ INVALID
The assumeutxo snapshot failed validation.
SynchronizationState
Current sync state passed to tip changed callbacks.
constexpr std::array FlushStateModeNames
constexpr int64_t LargeCoinsCacheThreshold(int64_t total_space) noexcept
@ LARGE
The cache is at >= 90% capacity.
@ CRITICAL
The coins cache is in immediate need of a flush.