6#include <bitcoin-build-config.h>
102 "level 0 reads the blocks from disk",
103 "level 1 verifies block validity",
104 "level 2 verifies undo data",
105 "level 3 checks disconnection of tip blocks",
106 "level 4 tries to reconnect the blocks",
107 "each level includes the checks of the previous levels",
145 std::vector<CScriptCheck>* pvChecks =
nullptr)
158 const int nBlockHeight = active_chain_tip.nHeight + 1;
165 const int64_t nBlockTime{active_chain_tip.GetMedianTimePast()};
167 return IsFinalTx(tx, nBlockHeight, nBlockTime);
181std::optional<std::vector<int>> CalculatePrevHeights(
186 std::vector<int> prev_heights;
187 prev_heights.resize(tx.
vin.size());
188 for (
size_t i = 0; i < tx.
vin.size(); ++i) {
189 if (
auto coin{coins.
GetCoin(tx.
vin[i].prevout)}) {
194 LogInfo(
"ERROR: %s: Missing input %d in transaction \'%s\'\n", __func__, i, tx.
GetHash().
GetHex());
209 auto prev_heights{CalculatePrevHeights(*tip, coins_view, tx)};
210 if (!prev_heights.has_value())
return std::nullopt;
213 next_tip.
pprev = tip;
231 int max_input_height{0};
232 for (
const int height : prev_heights.value()) {
234 if (height != next_tip.
nHeight) {
235 max_input_height = std::max(max_input_height, height);
270 int expired = pool.Expire(GetTime<std::chrono::seconds>() - pool.m_opts.expiry);
275 std::vector<COutPoint> vNoSpendsRemaining;
276 pool.TrimToSize(pool.m_opts.max_size_bytes, &vNoSpendsRemaining);
277 for (
const COutPoint& removed : vNoSpendsRemaining)
278 coins_cache.Uncache(removed);
284 if (active_chainstate.m_chainman.IsInitialBlockDownload()) {
289 if (active_chainstate.m_chain.Height() < active_chainstate.m_chainman.m_best_header->nHeight - 1) {
303 std::vector<Txid> vHashUpdate;
310 const auto queuedTx = disconnectpool.
take();
311 auto it = queuedTx.rbegin();
312 while (it != queuedTx.rend()) {
314 if (!fAddToMempool || (*it)->IsCoinBase() ||
316 true,
false).m_result_type !=
322 vHashUpdate.push_back((*it)->GetHash());
363 it->UpdateLockPoints(*new_lock_points);
370 if (it->GetSpendsCoinbase()) {
376 if (coin.IsCoinBase() && mempool_spend_height - coin.nHeight <
COINBASE_MATURITY) {
412 if (coin.
IsSpent())
return false;
442 m_viewmempool(&active_chainstate.CoinsTip(), m_pool),
443 m_active_chainstate(active_chainstate)
451 const int64_t m_accept_time;
452 const bool m_bypass_limits;
460 std::vector<COutPoint>& m_coins_to_uncache;
462 const bool m_test_accept;
466 const bool m_allow_replacement;
468 const bool m_allow_sibling_eviction;
471 const bool m_package_submission;
475 const bool m_package_feerates;
480 const std::optional<CFeeRate> m_client_maxfeerate;
483 static ATMPArgs SingleAccept(
const CChainParams& chainparams, int64_t accept_time,
484 bool bypass_limits, std::vector<COutPoint>& coins_to_uncache,
486 return ATMPArgs{ chainparams,
500 static ATMPArgs PackageTestAccept(
const CChainParams& chainparams, int64_t accept_time,
501 std::vector<COutPoint>& coins_to_uncache) {
502 return ATMPArgs{ chainparams,
516 static ATMPArgs PackageChildWithParents(
const CChainParams& chainparams, int64_t accept_time,
517 std::vector<COutPoint>& coins_to_uncache,
const std::optional<CFeeRate>& client_maxfeerate) {
518 return ATMPArgs{ chainparams,
532 static ATMPArgs SingleInPackageAccept(
const ATMPArgs& package_args) {
533 return ATMPArgs{ package_args.m_chainparams,
534 package_args.m_accept_time,
536 package_args.m_coins_to_uncache,
537 package_args.m_test_accept,
542 package_args.m_client_maxfeerate,
552 std::vector<COutPoint>& coins_to_uncache,
554 bool allow_replacement,
555 bool allow_sibling_eviction,
556 bool package_submission,
557 bool package_feerates,
558 std::optional<CFeeRate> client_maxfeerate)
559 : m_chainparams{chainparams},
560 m_accept_time{accept_time},
561 m_bypass_limits{bypass_limits},
562 m_coins_to_uncache{coins_to_uncache},
563 m_test_accept{test_accept},
564 m_allow_replacement{allow_replacement},
565 m_allow_sibling_eviction{allow_sibling_eviction},
566 m_package_submission{package_submission},
567 m_package_feerates{package_feerates},
568 m_client_maxfeerate{client_maxfeerate}
572 if (m_package_feerates) {
573 Assume(m_package_submission);
574 Assume(!m_allow_sibling_eviction);
576 if (m_allow_sibling_eviction)
Assume(m_allow_replacement);
587 ClearSubPackageState();
600 ClearSubPackageState();
628 explicit Workspace(
const CTransactionRef& ptx) : m_ptx(ptx), m_hash(ptx->GetHash()) {}
631 std::set<Txid> m_conflicts;
636 std::vector<CTxMemPoolEntry::CTxMemPoolEntryRef> m_parents;
641 bool m_sibling_eviction{
false};
676 bool PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
677 std::vector<Workspace>& workspaces,
698 std::map<Wtxid, MempoolAcceptResult>& results)
706 CAmount mempoolRejectFee = m_pool.GetMinFee().GetFee(package_size);
707 if (mempoolRejectFee > 0 && package_fee < mempoolRejectFee) {
711 if (package_fee < m_pool.m_opts.min_relay_feerate.GetFee(package_size)) {
713 strprintf(
"%d < %d", package_fee, m_pool.m_opts.min_relay_feerate.GetFee(package_size)));
720 return m_active_chainstate.m_chainman.m_validation_cache;
752 struct SubPackageState {
754 CAmount m_total_modified_fees{0};
756 int64_t m_total_vsize{0};
763 std::list<CTransactionRef> m_replaced_transactions;
765 std::unique_ptr<CTxMemPool::ChangeSet> m_changeset;
770 size_t m_conflicting_size{0};
773 struct SubPackageState m_subpackage;
778 m_subpackage = SubPackageState{};
781 CleanupTemporaryCoins();
785bool MemPoolAccept::PreChecks(ATMPArgs&
args, Workspace& ws)
791 const Txid& hash = ws.m_hash;
794 const int64_t nAcceptTime =
args.m_accept_time;
795 const bool bypass_limits =
args.m_bypass_limits;
796 std::vector<COutPoint>& coins_to_uncache =
args.m_coins_to_uncache;
811 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)) {
829 }
else if (m_pool.exists(tx.
GetHash())) {
839 if (ptxConflicting) {
840 if (!
args.m_allow_replacement) {
844 ws.m_conflicts.insert(ptxConflicting->
GetHash());
848 m_view.SetBackend(m_viewmempool);
854 coins_to_uncache.push_back(txin.
prevout);
860 if (!m_view.HaveCoin(txin.
prevout)) {
875 m_view.GetBestBlock();
880 m_view.SetBackend(m_dummy);
882 assert(m_active_chainstate.m_blockman.LookupBlockIndex(m_view.GetBestBlock()) == m_active_chainstate.m_chain.Tip());
889 const std::optional<LockPoints> lock_points{
CalculateLockPointsAtTip(m_active_chainstate.m_chain.Tip(), m_view, tx)};
912 bool fSpendsCoinbase =
false;
914 const Coin &coin = m_view.AccessCoin(txin.
prevout);
916 fSpendsCoinbase =
true;
923 const uint64_t entry_sequence = bypass_limits ? 0 : m_pool.GetSequence();
924 if (!m_subpackage.m_changeset) {
925 m_subpackage.m_changeset = m_pool.GetChangeSet();
927 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());
930 ws.m_modified_fees = ws.m_tx_handle->GetModifiedFee();
932 ws.m_vsize = ws.m_tx_handle->GetTxSize();
935 if (m_pool.m_opts.require_standard) {
936 if (!
PreCheckEphemeralTx(*ptx, m_pool.m_opts.dust_relay_feerate, ws.m_base_fees, ws.m_modified_fees, state)) {
951 if (!bypass_limits && ws.m_ptx->version !=
TRUC_VERSION && ws.m_modified_fees < m_pool.m_opts.min_relay_feerate.GetFee(ws.m_vsize)) {
955 strprintf(
"%d < %d", ws.m_modified_fees, m_pool.m_opts.min_relay_feerate.GetFee(ws.m_vsize)));
960 if (!bypass_limits && !
args.m_package_feerates && !
CheckFeeRate(ws.m_vsize, ws.m_modified_fees, state))
return false;
962 ws.m_iters_conflicting = m_pool.GetIterSet(ws.m_conflicts);
964 ws.m_parents = m_pool.GetParents(*ws.m_tx_handle);
966 if (!
args.m_bypass_limits) {
968 if (
const auto err{
SingleTRUCChecks(m_pool, ws.m_ptx, ws.m_parents, ws.m_conflicts, ws.m_vsize)}) {
970 if (
args.m_allow_sibling_eviction && err->second !=
nullptr) {
975 ws.m_conflicts.insert(err->second->GetHash());
979 ws.m_iters_conflicting.insert(m_pool.GetIter(err->second->GetHash()).value());
980 ws.m_sibling_eviction =
true;
992 m_subpackage.m_rbf |= !ws.m_conflicts.empty();
996bool MemPoolAccept::ReplacementChecks(Workspace& ws)
1002 const Txid& hash = ws.m_hash;
1005 CFeeRate newFeeRate(ws.m_modified_fees, ws.m_vsize);
1012 strprintf(
"too many potential replacements%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1018 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1019 m_subpackage.m_conflicting_size += it->GetTxSize();
1022 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees, ws.m_modified_fees, ws.m_vsize,
1023 m_pool.m_opts.incremental_relay_feerate, hash)}) {
1026 strprintf(
"insufficient fee%s", ws.m_sibling_eviction ?
" (including sibling eviction)" :
""), *err_string);
1030 for (
auto it : all_conflicts) {
1031 m_subpackage.m_changeset->StageRemoval(it);
1035 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1049bool MemPoolAccept::PackageMempoolChecks(
const std::vector<CTransactionRef>& txns,
1050 std::vector<Workspace>& workspaces,
1051 const int64_t total_vsize,
1057 assert(std::all_of(txns.cbegin(), txns.cend(), [
this](
const auto& tx)
1058 { return !m_pool.exists(tx->GetHash());}));
1060 assert(txns.size() == workspaces.size());
1063 if (!m_subpackage.m_rbf)
return true;
1078 for (
const auto& ws : workspaces) {
1079 if (!ws.m_parents.empty()) {
1086 for (Workspace& ws : workspaces) {
1088 direct_conflict_iters.merge(ws.m_iters_conflicting);
1091 const auto& parent_ws = workspaces[0];
1092 const auto& child_ws = workspaces[1];
1100 "package RBF failed: too many potential replacements", *err_string);
1104 m_subpackage.m_changeset->StageRemoval(it);
1105 m_subpackage.m_conflicting_fees += it->GetModifiedFee();
1106 m_subpackage.m_conflicting_size += it->GetTxSize();
1110 const Txid& child_hash = child_ws.m_ptx->GetHash();
1111 if (
const auto err_string{
PaysForRBF(m_subpackage.m_conflicting_fees,
1112 m_subpackage.m_total_modified_fees,
1113 m_subpackage.m_total_vsize,
1114 m_pool.m_opts.incremental_relay_feerate, child_hash)}) {
1116 "package RBF failed: insufficient anti-DoS fees", *err_string);
1121 const CFeeRate parent_feerate(parent_ws.m_modified_fees, parent_ws.m_vsize);
1122 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1123 if (package_feerate <= parent_feerate) {
1125 "package RBF failed: package feerate is less than or equal to parent feerate",
1126 strprintf(
"package feerate %s <= parent feerate is %s", package_feerate.ToString(), parent_feerate.ToString()));
1130 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1138 "package RBF failed: " + err_tup.value().second,
"");
1141 LogDebug(
BCLog::TXPACKAGES,
"package RBF checks passed: parent %s (wtxid=%s), child %s (wtxid=%s), package hash (%s)\n",
1142 txns.front()->GetHash().ToString(), txns.front()->GetWitnessHash().ToString(),
1143 txns.back()->GetHash().ToString(), txns.back()->GetWitnessHash().ToString(),
1150bool MemPoolAccept::PolicyScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1161 if (!
CheckInputScripts(tx, state, m_view, scriptVerifyFlags,
true,
false, ws.m_precomputed_txdata, GetValidationCache())) {
1173bool MemPoolAccept::ConsensusScriptChecks(
const ATMPArgs&
args, Workspace& ws)
1178 const Txid& hash = ws.m_hash;
1198 ws.m_precomputed_txdata, m_active_chainstate.CoinsTip(), GetValidationCache())) {
1199 LogError(
"BUG! PLEASE REPORT THIS! CheckInputScripts failed against latest-block but not STANDARD flags %s, %s", hash.
ToString(), state.
ToString());
1206void MemPoolAccept::FinalizeSubpackage(
const ATMPArgs&
args)
1211 if (!m_subpackage.m_changeset->GetRemovals().empty())
Assume(
args.m_allow_replacement);
1215 std::string log_string =
strprintf(
"replacing mempool tx %s (wtxid=%s, fees=%s, vsize=%s). ",
1216 it->GetTx().GetHash().ToString(),
1217 it->GetTx().GetWitnessHash().ToString(),
1220 FeeFrac feerate{m_subpackage.m_total_modified_fees, int32_t(m_subpackage.m_total_vsize)};
1222 const bool replaced_with_tx{m_subpackage.m_changeset->GetTxCount() == 1};
1223 if (replaced_with_tx) {
1224 const CTransaction& tx = m_subpackage.m_changeset->GetAddedTxn(0);
1226 log_string +=
strprintf(
"New tx %s (wtxid=%s, fees=%s, vsize=%s)",
1232 tx_or_package_hash =
GetPackageHash(m_subpackage.m_changeset->GetAddedTxns());
1233 log_string +=
strprintf(
"New package %s with %lu txs, fees=%s, vsize=%s",
1234 tx_or_package_hash.ToString(),
1235 m_subpackage.m_changeset->GetTxCount(),
1242 it->GetTx().GetHash().data(),
1245 std::chrono::duration_cast<std::chrono::duration<std::uint64_t>>(it->GetTime()).count(),
1246 tx_or_package_hash.data(),
1251 m_subpackage.m_replaced_transactions.push_back(it->GetSharedTx());
1253 m_subpackage.m_changeset->Apply();
1254 m_subpackage.m_changeset.reset();
1257bool MemPoolAccept::SubmitPackage(
const ATMPArgs&
args, std::vector<Workspace>& workspaces,
1259 std::map<Wtxid, MempoolAcceptResult>& results)
1265 assert(std::all_of(workspaces.cbegin(), workspaces.cend(), [
this](
const auto& ws) { return !m_pool.exists(ws.m_ptx->GetHash()); }));
1267 bool all_submitted =
true;
1268 FinalizeSubpackage(
args);
1273 for (Workspace& ws : workspaces) {
1274 if (!ConsensusScriptChecks(
args, ws)) {
1278 all_submitted =
false;
1280 strprintf(
"BUG! PolicyScriptChecks succeeded but ConsensusScriptChecks failed: %s",
1281 ws.m_ptx->GetHash().ToString()));
1283 if (!m_subpackage.m_changeset) m_subpackage.m_changeset = m_pool.GetChangeSet();
1284 m_subpackage.m_changeset->StageRemoval(m_pool.GetIter(ws.m_ptx->GetHash()).value());
1287 if (!all_submitted) {
1288 Assume(m_subpackage.m_changeset);
1292 m_subpackage.m_changeset->Apply();
1293 m_subpackage.m_changeset.reset();
1297 std::vector<Wtxid> all_package_wtxids;
1298 all_package_wtxids.reserve(workspaces.size());
1299 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1300 [](
const auto& ws) { return ws.m_ptx->GetWitnessHash(); });
1302 if (!m_subpackage.m_replaced_transactions.empty()) {
1303 LogDebug(
BCLog::MEMPOOL,
"replaced %u mempool transactions with %u new one(s) for %s additional fees, %d delta bytes\n",
1304 m_subpackage.m_replaced_transactions.size(), workspaces.size(),
1305 m_subpackage.m_total_modified_fees - m_subpackage.m_conflicting_fees,
1306 m_subpackage.m_total_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1310 for (Workspace& ws : workspaces) {
1311 auto iter = m_pool.GetIter(ws.m_ptx->GetHash());
1312 Assume(iter.has_value());
1313 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1314 CFeeRate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1315 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1316 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1317 results.emplace(ws.m_ptx->GetWitnessHash(),
1319 ws.m_base_fees, effective_feerate, effective_feerate_wtxids));
1320 if (!m_pool.m_opts.signals)
continue;
1323 ws.m_vsize, (*iter)->GetHeight(),
1324 args.m_bypass_limits,
args.m_package_submission,
1326 m_pool.HasNoInputsOf(tx));
1327 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1329 return all_submitted;
1338 const std::vector<Wtxid> single_wtxid{ws.m_ptx->GetWitnessHash()};
1340 if (!PreChecks(
args, ws)) {
1348 if (m_subpackage.m_rbf && !ReplacementChecks(ws)) {
1357 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1364 if (ws.m_conflicts.size()) {
1365 auto ancestors = m_subpackage.m_changeset->CalculateMemPoolAncestors(ws.m_tx_handle);
1379 m_subpackage.m_total_vsize = ws.m_vsize;
1380 m_subpackage.m_total_modified_fees = ws.m_modified_fees;
1383 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1388 if (m_pool.m_opts.require_standard) {
1390 if (!
CheckEphemeralSpends({ptx}, m_pool.m_opts.dust_relay_feerate, m_pool, ws.m_state, dummy_wtxid)) {
1401 const CFeeRate effective_feerate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1403 if (
args.m_test_accept) {
1405 ws.m_base_fees, effective_feerate, single_wtxid);
1408 FinalizeSubpackage(
args);
1411 if (!
args.m_package_submission && !
args.m_bypass_limits) {
1415 CleanupTemporaryCoins();
1417 if (!m_pool.exists(ws.m_hash)) {
1424 if (m_pool.m_opts.signals) {
1426 auto iter = m_pool.GetIter(tx.
GetHash());
1427 Assume(iter.has_value());
1429 ws.m_vsize, (*iter)->GetHeight(),
1430 args.m_bypass_limits,
args.m_package_submission,
1432 m_pool.HasNoInputsOf(tx));
1433 m_pool.m_opts.signals->TransactionAddedToMempool(tx_info, m_pool.GetAndIncrementSequence());
1436 if (!m_subpackage.m_replaced_transactions.empty()) {
1437 LogDebug(
BCLog::MEMPOOL,
"replaced %u mempool transactions with 1 new transaction for %s additional fees, %d delta bytes\n",
1438 m_subpackage.m_replaced_transactions.size(),
1439 ws.m_modified_fees - m_subpackage.m_conflicting_fees,
1440 ws.m_vsize -
static_cast<int>(m_subpackage.m_conflicting_size));
1444 effective_feerate, single_wtxid);
1456 std::vector<Workspace> workspaces{};
1457 workspaces.reserve(txns.size());
1458 std::transform(txns.cbegin(), txns.cend(), std::back_inserter(workspaces),
1459 [](
const auto& tx) { return Workspace(tx); });
1460 std::map<Wtxid, MempoolAcceptResult> results;
1463 for (Workspace& ws : workspaces) {
1464 if (!PreChecks(
args, ws)) {
1473 if (
args.m_client_maxfeerate &&
CFeeRate(ws.m_modified_fees, ws.m_vsize) >
args.m_client_maxfeerate.value()) {
1491 m_viewmempool.PackageAddTransaction(ws.m_ptx);
1496 for (Workspace& ws : workspaces) {
1497 if (
auto err{
PackageTRUCChecks(m_pool, ws.m_ptx, ws.m_vsize, txns, ws.m_parents)}) {
1512 m_subpackage.m_total_vsize = std::accumulate(workspaces.cbegin(), workspaces.cend(), int64_t{0},
1513 [](int64_t
sum,
auto& ws) { return sum + ws.m_vsize; });
1514 m_subpackage.m_total_modified_fees = std::accumulate(workspaces.cbegin(), workspaces.cend(),
CAmount{0},
1515 [](
CAmount sum,
auto& ws) { return sum + ws.m_modified_fees; });
1516 const CFeeRate package_feerate(m_subpackage.m_total_modified_fees, m_subpackage.m_total_vsize);
1517 std::vector<Wtxid> all_package_wtxids;
1518 all_package_wtxids.reserve(workspaces.size());
1519 std::transform(workspaces.cbegin(), workspaces.cend(), std::back_inserter(all_package_wtxids),
1520 [](
const auto& ws) { return ws.m_ptx->GetWitnessHash(); });
1522 if (
args.m_package_feerates &&
1523 !
CheckFeeRate(m_subpackage.m_total_vsize, m_subpackage.m_total_modified_fees, placeholder_state)) {
1530 if (!PackageMempoolChecks(txns, workspaces, m_subpackage.m_total_vsize, package_state)) {
1535 if (!m_subpackage.m_changeset->CheckMemPoolPolicyLimits()) {
1541 if (m_pool.m_opts.require_standard) {
1544 if (!
CheckEphemeralSpends(txns, m_pool.m_opts.dust_relay_feerate, m_pool, child_state, child_wtxid)) {
1551 for (Workspace& ws : workspaces) {
1552 ws.m_package_feerate = package_feerate;
1553 if (!PolicyScriptChecks(
args, ws)) {
1559 if (
args.m_test_accept) {
1560 const auto effective_feerate =
args.m_package_feerates ? ws.m_package_feerate :
1561 CFeeRate{ws.m_modified_fees,
static_cast<int32_t
>(ws.m_vsize)};
1562 const auto effective_feerate_wtxids =
args.m_package_feerates ? all_package_wtxids :
1563 std::vector<Wtxid>{ws.m_ptx->GetWitnessHash()};
1564 results.emplace(ws.m_ptx->GetWitnessHash(),
1566 ws.m_vsize, ws.m_base_fees, effective_feerate,
1567 effective_feerate_wtxids));
1573 if (!SubmitPackage(
args, workspaces, package_state, results)) {
1581void MemPoolAccept::CleanupTemporaryCoins()
1602 for (
const auto& outpoint : m_viewmempool.GetNonBaseCoins()) {
1605 m_view.Uncache(outpoint);
1608 m_viewmempool.Reset();
1616 if (subpackage.size() > 1) {
1617 return AcceptMultipleTransactionsInternal(subpackage,
args);
1619 const auto& tx = subpackage.front();
1620 ATMPArgs single_args = ATMPArgs::SingleInPackageAccept(
args);
1621 const auto single_res = AcceptSingleTransactionInternal(tx, single_args);
1632 ClearSubPackageState();
1639 Assert(!package.empty());
1665 std::map<Wtxid, MempoolAcceptResult> results_final;
1669 std::map<Wtxid, MempoolAcceptResult> individual_results_nonfinal;
1671 bool quit_early{
false};
1672 std::vector<CTransactionRef> txns_package_eval;
1673 for (
const auto& tx : package) {
1675 const auto& txid = tx->
GetHash();
1679 if (m_pool.exists(wtxid)) {
1689 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1691 }
else if (m_pool.exists(txid)) {
1699 const auto& entry{*
Assert(m_pool.GetEntry(txid))};
1705 const auto single_package_res = AcceptSubPackage({tx},
args);
1706 const auto& single_res = single_package_res.m_tx_results.at(wtxid);
1710 assert(m_pool.exists(wtxid));
1711 results_final.emplace(wtxid, single_res);
1712 }
else if (package.size() == 1 ||
1726 individual_results_nonfinal.emplace(wtxid, single_res);
1728 individual_results_nonfinal.emplace(wtxid, single_res);
1729 txns_package_eval.push_back(tx);
1734 auto multi_submission_result = quit_early || txns_package_eval.empty() ?
PackageMempoolAcceptResult(package_state_quit_early, {}) :
1735 AcceptSubPackage(txns_package_eval,
args);
1741 ClearSubPackageState();
1748 for (
const auto& tx : package) {
1750 if (multi_submission_result.m_tx_results.count(wtxid) > 0) {
1752 Assume(results_final.count(wtxid) == 0);
1755 const auto& txresult = multi_submission_result.m_tx_results.at(wtxid);
1762 results_final.emplace(wtxid, txresult);
1764 }
else if (
const auto it{results_final.find(wtxid)}; it != results_final.end()) {
1768 Assume(individual_results_nonfinal.count(wtxid) == 0);
1770 if (!m_pool.exists(tx->
GetHash())) {
1775 results_final.erase(wtxid);
1778 }
else if (
const auto it{individual_results_nonfinal.find(wtxid)}; it != individual_results_nonfinal.end()) {
1781 results_final.emplace(wtxid, it->second);
1784 Assume(results_final.size() == package.size());
1791 int64_t accept_time,
bool bypass_limits,
bool test_accept)
1798 std::vector<COutPoint> coins_to_uncache;
1800 auto args = MemPoolAccept::ATMPArgs::SingleAccept(chainparams, accept_time, bypass_limits, coins_to_uncache, test_accept);
1801 MempoolAcceptResult result = MemPoolAccept(pool, active_chainstate).AcceptSingleTransactionAndCleanup(tx,
args);
1809 for (
const COutPoint& hashTx : coins_to_uncache)
1812 tx->GetHash().data(),
1823 const Package& package,
bool test_accept,
const std::optional<CFeeRate>& client_maxfeerate)
1826 assert(!package.empty());
1827 assert(std::all_of(package.cbegin(), package.cend(), [](
const auto& tx){return tx != nullptr;}));
1829 std::vector<COutPoint> coins_to_uncache;
1834 auto args = MemPoolAccept::ATMPArgs::PackageTestAccept(chainparams,
GetTime(), coins_to_uncache);
1835 return MemPoolAccept(pool, active_chainstate).AcceptMultipleTransactionsAndCleanup(package,
args);
1837 auto args = MemPoolAccept::ATMPArgs::PackageChildWithParents(chainparams,
GetTime(), coins_to_uncache, client_maxfeerate);
1838 return MemPoolAccept(pool, active_chainstate).AcceptPackage(package,
args);
1843 if (test_accept || result.m_state.IsInvalid()) {
1844 for (
const COutPoint& hashTx : coins_to_uncache) {
1863 nSubsidy >>= halvings;
1868 : m_dbview{
std::move(db_params),
std::move(options)},
1869 m_catcherview(&m_dbview) {}
1871void CoinsViews::InitCache()
1874 m_cacheview = std::make_unique<CCoinsViewCache>(&m_catcherview);
1881 std::optional<uint256> from_snapshot_blockhash)
1882 : m_mempool(mempool),
1883 m_blockman(blockman),
1884 m_chainman(chainman),
1885 m_from_snapshot_blockhash(from_snapshot_blockhash) {}
1887const CBlockIndex* Chainstate::SnapshotBase()
const
1891 return m_cached_snapshot_base;
1895 size_t cache_size_bytes,
1898 fs::path leveldb_name)
1907 .cache_bytes = cache_size_bytes,
1908 .memory_only = in_memory,
1909 .wipe_data = should_wipe,
1917void Chainstate::InitCoinsCache(
size_t cache_size_bytes)
1943 if (chain.Tip() ==
nullptr) {
1952 LogInfo(
"Leaving InitialBlockDownload (latching to false)");
1966 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.");
1969 _(
"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."));
1982 SetBlockFailureFlags(pindexNew);
1987 LogInfo(
"%s: invalid block=%s height=%d log2_work=%f date=%s\n", __func__,
1992 LogInfo(
"%s: current best=%s height=%d log2_work=%f date=%s\n", __func__,
2030 if (
VerifyScript(scriptSig,
m_tx_out.
scriptPubKey, witness,
m_flags,
CachingTransactionSignatureChecker(
ptxTo,
nIn,
m_tx_out.
nValue,
cacheStore, *
m_signature_cache, *
txdata), &error)) {
2031 return std::nullopt;
2034 return std::make_pair(error, std::move(debug_str));
2039 : m_signature_cache{signature_cache_bytes}
2050 LogInfo(
"Using %zu MiB out of %zu MiB requested for script execution cache, able to store %zu elements",
2051 approx_size_bytes >> 20, script_execution_cache_bytes >> 20, num_elems);
2077 std::vector<CScriptCheck>* pvChecks)
2082 pvChecks->reserve(tx.
vin.size());
2099 std::vector<CTxOut> spent_outputs;
2100 spent_outputs.reserve(tx.
vin.size());
2102 for (
const auto& txin : tx.
vin) {
2106 spent_outputs.emplace_back(coin.
out);
2108 txdata.
Init(tx, std::move(spent_outputs));
2112 for (
unsigned int i = 0; i < tx.
vin.size(); i++) {
2123 pvChecks->emplace_back(std::move(
check));
2124 }
else if (
auto result =
check(); result.has_value()) {
2139 if (cacheFullScriptStore && !pvChecks) {
2167 if (undo.nHeight == 0) {
2173 undo.nHeight = alternate.
nHeight;
2198 LogError(
"DisconnectBlock(): failure reading undo data\n");
2202 if (blockUndo.
vtxundo.size() + 1 != block.
vtx.size()) {
2203 LogError(
"DisconnectBlock(): block and undo data inconsistent\n");
2213 bool fEnforceBIP30 = !((pindex->
nHeight==91722 && pindex->
GetBlockHash() ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
2214 (pindex->
nHeight==91812 && pindex->
GetBlockHash() ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"}));
2217 for (
int i = block.
vtx.size() - 1; i >= 0; i--) {
2221 bool is_bip30_exception = (is_coinbase && !fEnforceBIP30);
2225 for (
size_t o = 0; o < tx.
vout.size(); o++) {
2226 if (!tx.
vout[o].scriptPubKey.IsUnspendable()) {
2231 if (!is_bip30_exception) {
2242 LogError(
"DisconnectBlock(): transaction and undo data inconsistent\n");
2245 for (
unsigned int j = tx.
vin.size(); j > 0;) {
2316 const auto time_start{SteadyClock::now()};
2332 if (!
CheckBlock(block, state, params.GetConsensus(), !fJustCheck, !fJustCheck)) {
2344 uint256 hashPrevBlock = pindex->
pprev ==
nullptr ?
uint256() : pindex->pprev->GetBlockHash();
2351 if (block_hash == params.GetConsensus().hashGenesisBlock) {
2357 const char* script_check_reason;
2359 script_check_reason =
"assumevalid=0 (always verify)";
2361 constexpr int64_t TWO_WEEKS_IN_SECONDS{60 * 60 * 24 * 7 * 2};
2369 script_check_reason =
"assumevalid hash not in headers";
2370 }
else if (it->second.GetAncestor(pindex->
nHeight) != pindex) {
2371 script_check_reason = (pindex->
nHeight > it->second.nHeight) ?
"block height above assumevalid height" :
"block not in assumevalid chain";
2373 script_check_reason =
"block not in best header chain";
2375 script_check_reason =
"best header chainwork below minimumchainwork";
2377 script_check_reason =
"block too recent relative to best header";
2393 script_check_reason =
nullptr;
2397 const auto time_1{SteadyClock::now()};
2398 m_chainman.time_check += time_1 - time_start;
2400 Ticks<MillisecondsDouble>(time_1 - time_start),
2442 static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702;
2474 fEnforceBIP30 = fEnforceBIP30 && (!pindexBIP34height || !(pindexBIP34height->
GetBlockHash() == params.GetConsensus().BIP34Hash));
2479 if (fEnforceBIP30 || pindex->
nHeight >= BIP34_IMPLIES_BIP30_LIMIT) {
2480 for (
const auto& tx : block.
vtx) {
2481 for (
size_t o = 0; o < tx->
vout.size(); o++) {
2484 "tried to overwrite transaction");
2491 int nLockTimeFlags = 0;
2499 const auto time_2{SteadyClock::now()};
2502 Ticks<MillisecondsDouble>(time_2 - time_1),
2506 const bool fScriptChecks{!!script_check_reason};
2508 if (fScriptChecks) {
2509 LogInfo(
"Enabling script verification at block #%d (%s): %s.",
2510 pindex->
nHeight, block_hash.ToString(), script_check_reason);
2512 LogInfo(
"Disabling script verification at block #%d (%s).",
2513 pindex->
nHeight, block_hash.ToString());
2515 m_last_script_check_reason_logged = script_check_reason;
2525 std::optional<CCheckQueueControl<CScriptCheck>> control;
2528 std::vector<PrecomputedTransactionData> txsdata(block.
vtx.size());
2530 std::vector<int> prevheights;
2533 int64_t nSigOpsCost = 0;
2534 blockundo.
vtxundo.reserve(block.
vtx.size() - 1);
2535 for (
unsigned int i = 0; i < block.
vtx.size(); i++)
2540 nInputs += tx.
vin.size();
2556 "accumulated fee in the block out of range");
2563 prevheights.resize(tx.
vin.size());
2564 for (
size_t j = 0; j < tx.
vin.size(); j++) {
2568 if (!
SequenceLocks(tx, nLockTimeFlags, prevheights, *pindex)) {
2587 bool fCacheResults = fJustCheck;
2593 std::vector<CScriptCheck> vChecks;
2595 if (tx_ok) control->Add(std::move(vChecks));
2609 blockundo.
vtxundo.emplace_back();
2613 const auto time_3{SteadyClock::now()};
2615 LogDebug(
BCLog::BENCH,
" - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) [%.2fs (%.2fms/blk)]\n", (
unsigned)block.
vtx.size(),
2616 Ticks<MillisecondsDouble>(time_3 - time_2), Ticks<MillisecondsDouble>(time_3 - time_2) / block.
vtx.size(),
2617 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_3 - time_2) / (nInputs - 1),
2618 Ticks<SecondsDouble>(
m_chainman.time_connect),
2622 if (block.
vtx[0]->GetValueOut() > blockReward && state.
IsValid()) {
2624 strprintf(
"coinbase pays too much (actual=%d vs limit=%d)", block.
vtx[0]->GetValueOut(), blockReward));
2627 auto parallel_result = control->Complete();
2628 if (parallel_result.has_value() && state.
IsValid()) {
2636 const auto time_4{SteadyClock::now()};
2638 LogDebug(
BCLog::BENCH,
" - Verify %u txins: %.2fms (%.3fms/txin) [%.2fs (%.2fms/blk)]\n", nInputs - 1,
2639 Ticks<MillisecondsDouble>(time_4 - time_2),
2640 nInputs <= 1 ? 0 : Ticks<MillisecondsDouble>(time_4 - time_2) / (nInputs - 1),
2641 Ticks<SecondsDouble>(
m_chainman.time_verify),
2648 if (!
m_blockman.WriteBlockUndo(blockundo, state, *pindex)) {
2652 const auto time_5{SteadyClock::now()};
2655 Ticks<MillisecondsDouble>(time_5 - time_4),
2667 const auto time_6{SteadyClock::now()};
2670 Ticks<MillisecondsDouble>(time_6 - time_5),
2680 Ticks<std::chrono::nanoseconds>(time_5 - time_start)
2689 return this->GetCoinsCacheSizeState(
2695 size_t max_coins_cache_size_bytes,
2696 size_t max_mempool_size_bytes)
2701 int64_t nTotalSpace =
2702 max_coins_cache_size_bytes + std::max<int64_t>(int64_t(max_mempool_size_bytes) - nMempoolUsage, 0);
2704 if (cacheSize > nTotalSpace) {
2705 LogInfo(
"Cache size (%s) exceeds total space (%s)\n", cacheSize, nTotalSpace);
2716 int nManualPruneHeight)
2720 std::set<int> setFilesToPrune;
2721 bool full_flush_completed =
false;
2728 bool fFlushForPrune =
false;
2736 std::optional<std::string> limiting_lock;
2738 for (
const auto& prune_lock :
m_blockman.m_prune_locks) {
2739 if (prune_lock.second.height_first == std::numeric_limits<int>::max())
continue;
2742 last_prune = std::max(1, std::min(last_prune, lock_height));
2743 if (last_prune == lock_height) {
2744 limiting_lock = prune_lock.first;
2748 if (limiting_lock) {
2749 LogDebug(
BCLog::PRUNE,
"%s limited pruning to height %d\n", limiting_lock.value(), last_prune);
2752 if (nManualPruneHeight > 0) {
2757 std::min(last_prune, nManualPruneHeight),
2765 if (!setFilesToPrune.empty()) {
2766 fFlushForPrune =
true;
2768 m_blockman.m_block_tree_db->WriteFlag(
"prunedblockfiles",
true);
2781 bool should_write = (mode ==
FlushStateMode::ALWAYS) || fCacheLarge || fCacheCritical || fPeriodicWrite || fFlushForPrune;
2784 LogDebug(
BCLog::COINDB,
"Writing chainstate to disk: flush mode=%s, prune=%d, large=%d, critical=%d, periodic=%d",
2785 FlushStateModeNames[
size_t(mode)], fFlushForPrune, fCacheLarge, fCacheCritical, fPeriodicWrite);
2798 LogWarning(
"%s: Failed to flush block file.\n", __func__);
2809 if (fFlushForPrune) {
2815 if (!
CoinsTip().GetBestBlock().IsNull()) {
2816 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);
2833 full_flush_completed =
true;
2835 int64_t{Ticks<std::chrono::microseconds>(
NodeClock::now() - nNow)},
2837 (uint64_t)coins_count,
2838 (uint64_t)coins_mem_usage,
2839 (
bool)fFlushForPrune);
2843 if (should_write ||
m_next_write == NodeClock::time_point::max()) {
2852 }
catch (
const std::runtime_error& e) {
2879 const std::string& func_name,
2880 const std::string&
prefix,
2892 chainman.GuessVerificationProgress(tip),
2895 !warning_messages.empty() ?
strprintf(
" warning='%s'", warning_messages) :
"");
2898void Chainstate::UpdateTip(
const CBlockIndex* pindexNew)
2901 const auto& coins_tip = this->
CoinsTip();
2907 constexpr int BACKGROUND_LOG_INTERVAL = 2000;
2908 if (pindexNew->
nHeight % BACKGROUND_LOG_INTERVAL == 0) {
2919 std::vector<bilingual_str> warning_messages;
2922 for (
auto [bit, active] : bits) {
2927 warning_messages.push_back(warning);
2954 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
2957 LogError(
"DisconnectTip(): Failed to read block\n");
2961 const auto time_start{SteadyClock::now()};
2965 if (DisconnectBlock(block, pindexDelete, view) !=
DISCONNECT_OK) {
2969 bool flushed = view.
Flush(
false);
2973 Ticks<MillisecondsDouble>(SteadyClock::now() - time_start));
2977 const int max_height_first{pindexDelete->
nHeight - 1};
2978 for (
auto& prune_lock :
m_blockman.m_prune_locks) {
2979 if (prune_lock.second.height_first <= max_height_first)
continue;
2981 prune_lock.second.height_first = max_height_first;
2982 LogDebug(
BCLog::PRUNE,
"%s prune lock moved back to %d\n", prune_lock.first, max_height_first);
3001 UpdateTip(pindexDelete->
pprev);
3059 std::shared_ptr<const CBlock> block_to_connect,
3068 const auto time_1{SteadyClock::now()};
3069 if (!block_to_connect) {
3070 std::shared_ptr<CBlock> pblockNew = std::make_shared<CBlock>();
3074 block_to_connect = std::move(pblockNew);
3079 const auto time_2{SteadyClock::now()};
3080 SteadyClock::time_point time_3;
3084 Ticks<MillisecondsDouble>(time_2 - time_1));
3087 bool rv =
ConnectBlock(*block_to_connect, state, pindexNew, view);
3097 time_3 = SteadyClock::now();
3098 m_chainman.time_connect_total += time_3 - time_2;
3101 Ticks<MillisecondsDouble>(time_3 - time_2),
3102 Ticks<SecondsDouble>(
m_chainman.time_connect_total),
3104 bool flushed = view.
Flush(
false);
3107 const auto time_4{SteadyClock::now()};
3110 Ticks<MillisecondsDouble>(time_4 - time_3),
3117 const auto time_5{SteadyClock::now()};
3118 m_chainman.time_chainstate += time_5 - time_4;
3120 Ticks<MillisecondsDouble>(time_5 - time_4),
3121 Ticks<SecondsDouble>(
m_chainman.time_chainstate),
3130 UpdateTip(pindexNew);
3132 const auto time_6{SteadyClock::now()};
3133 m_chainman.time_post_connect += time_6 - time_5;
3136 Ticks<MillisecondsDouble>(time_6 - time_5),
3137 Ticks<SecondsDouble>(
m_chainman.time_post_connect),
3140 Ticks<MillisecondsDouble>(time_6 - time_1),
3149 m_chainman.MaybeCompleteSnapshotValidation();
3152 connectTrace.
BlockConnected(pindexNew, std::move(block_to_connect));
3177 bool fInvalidAncestor =
false;
3187 if (fFailedChain || fMissingData) {
3194 while (pindexTest != pindexFailed) {
3198 }
else if (fMissingData) {
3203 std::make_pair(pindexFailed->
pprev, pindexFailed));
3206 pindexFailed = pindexFailed->
pprev;
3209 fInvalidAncestor =
true;
3212 pindexTest = pindexTest->
pprev;
3214 if (!fInvalidAncestor)
3246 bool fBlocksDisconnected =
false;
3260 fBlocksDisconnected =
true;
3264 std::vector<CBlockIndex*> vpindexToConnect;
3265 bool fContinue =
true;
3270 int nTargetHeight = std::min(
nHeight + 32, pindexMostWork->
nHeight);
3271 vpindexToConnect.clear();
3272 vpindexToConnect.reserve(nTargetHeight -
nHeight);
3275 vpindexToConnect.push_back(pindexIter);
3276 pindexIter = pindexIter->
pprev;
3281 for (
CBlockIndex* pindexConnect : vpindexToConnect | std::views::reverse) {
3282 if (!
ConnectTip(state, pindexConnect, pindexConnect == pindexMostWork ? pblock : std::shared_ptr<const CBlock>(), connectTrace, disconnectpool)) {
3289 fInvalidFound =
true;
3310 if (fBlocksDisconnected) {
3331 bool fNotify =
false;
3332 bool fInitialBlockDownload =
false;
3336 pindexHeader = m_best_header;
3338 if (pindexHeader != m_last_notified_header) {
3341 m_last_notified_header = pindexHeader;
3354 if (signals.CallbacksPending() > 10) {
3355 signals.SyncWithValidationInterfaceQueue();
3359bool Chainstate::ActivateBestChain(
BlockValidationState& state, std::shared_ptr<const CBlock> pblock)
3378 LogError(
"m_disabled is set - this chainstate should not be in operation. "
3379 "Please report this as a bug. %s", CLIENT_BUGREPORT);
3385 bool exited_ibd{
false};
3402 bool blocks_connected =
false;
3408 if (pindexMostWork ==
nullptr) {
3413 if (pindexMostWork ==
nullptr || pindexMostWork ==
m_chain.
Tip()) {
3417 bool fInvalidFound =
false;
3418 std::shared_ptr<const CBlock> nullBlockPtr;
3423 if (!
ActivateBestChainStep(state, pindexMostWork, pblock && pblock->GetHash() == pindexMostWork->
GetBlockHash() ? pblock : nullBlockPtr, fInvalidFound, connectTrace)) {
3427 blocks_connected =
true;
3429 if (fInvalidFound) {
3431 pindexMostWork =
nullptr;
3436 assert(trace.pblock && trace.pindex);
3451 if (!blocks_connected)
return true;
3456 if (was_in_ibd && !still_in_ibd) {
3489 bool disabled{
false};
3503 disabled = m_disabled;
3526 }
while (pindexNewTip != pindexMostWork);
3561 return ActivateBestChain(state, std::shared_ptr<const CBlock>());
3571 if (pindex->
nHeight == 0)
return false;
3574 bool pindex_was_in_chain =
false;
3575 int disconnected = 0;
3589 std::multimap<const arith_uint256, CBlockIndex*> highpow_outofchain_headers;
3593 for (
auto& entry :
m_blockman.m_block_index) {
3603 highpow_outofchain_headers.insert({candidate->
nChainWork, candidate});
3620 pindex_was_in_chain =
true;
3633 if (!
ret)
return false;
3656 auto candidate_it = highpow_outofchain_headers.lower_bound(invalid_walk_tip->
pprev->
nChainWork);
3658 const bool best_header_needs_update{
m_chainman.m_best_header->GetAncestor(invalid_walk_tip->
nHeight) == invalid_walk_tip};
3659 if (best_header_needs_update) {
3664 while (candidate_it != highpow_outofchain_headers.end()) {
3668 candidate->nStatus &= ~BLOCK_FAILED_VALID;
3673 candidate_it = highpow_outofchain_headers.erase(candidate_it);
3683 if (best_header_needs_update &&
3692 to_mark_failed = invalid_walk_tip;
3718 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3728 if (pindex_was_in_chain) {
3738 *to_mark_failed->
pprev,
3750void Chainstate::SetBlockFailureFlags(
CBlockIndex* invalid_block)
3754 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3755 if (invalid_block != &block_index && block_index.GetAncestor(invalid_block->
nHeight) == invalid_block) {
3756 block_index.nStatus = (block_index.nStatus & ~BLOCK_FAILED_VALID) |
BLOCK_FAILED_CHILD;
3768 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
3770 block_index.nStatus &= ~BLOCK_FAILED_MASK;
3775 if (&block_index ==
m_chainman.m_best_invalid) {
3793 if (is_active_chainstate) {
3797 }
else if (!m_disabled) {
3802 if (snapshot_base->GetAncestor(pindex->
nHeight) == pindex) {
3812 pindexNew->
nTx = block.
vtx.size();
3818 auto prev_tx_sum = [](
CBlockIndex& block) {
return block.nTx + (block.pprev ? block.pprev->m_chain_tx_count : 0); };
3820 pindexNew == GetSnapshotBaseBlock())) {
3821 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",
3825 pindexNew->nFile = pos.
nFile;
3826 pindexNew->nDataPos = pos.
nPos;
3827 pindexNew->nUndoPos = 0;
3837 std::deque<CBlockIndex*> queue;
3838 queue.push_back(pindexNew);
3841 while (!queue.empty()) {
3849 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",
3855 c->TryAddBlockIndexCandidate(pindex);
3857 std::pair<std::multimap<CBlockIndex*, CBlockIndex*>::iterator, std::multimap<CBlockIndex*, CBlockIndex*>::iterator> range =
m_blockman.
m_blocks_unlinked.equal_range(pindex);
3858 while (range.first != range.second) {
3859 std::multimap<CBlockIndex*, CBlockIndex*>::iterator it = range.first;
3860 queue.push_back(it->second);
3891 "hashMerkleRoot mismatch");
3900 "bad-txns-duplicate",
3901 "duplicate transaction");
3916 if (expect_witness_commitment) {
3921 assert(!block.
vtx.empty() && !block.
vtx[0]->vin.empty());
3922 const auto& witness_stack{block.
vtx[0]->vin[0].scriptWitness.stack};
3924 if (witness_stack.size() != 1 || witness_stack[0].size() != 32) {
3927 "bad-witness-nonce-size",
3928 strprintf(
"%s : invalid witness reserved value size", __func__));
3937 if (memcmp(hash_witness.
begin(), &block.
vtx[0]->vout[commitpos].scriptPubKey[6], 32)) {
3940 "bad-witness-merkle-match",
3941 strprintf(
"%s : witness merkle commitment mismatch", __func__));
3950 for (
const auto& tx : block.
vtx) {
3954 "unexpected-witness",
3955 strprintf(
"%s : unexpected witness data found", __func__));
3995 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase())
3997 for (
unsigned int i = 1; i < block.
vtx.size(); i++)
3998 if (block.
vtx[i]->IsCoinBase())
4003 for (
const auto& tx : block.
vtx) {
4015 unsigned int nSigOps = 0;
4016 for (
const auto& tx : block.
vtx)
4023 if (fCheckPOW && fCheckMerkleRoot)
4032 static const std::vector<unsigned char>
nonce(32, 0x00);
4035 tx.
vin[0].scriptWitness.stack.resize(1);
4036 tx.
vin[0].scriptWitness.stack[0] =
nonce;
4043 std::vector<unsigned char> commitment;
4045 std::vector<unsigned char>
ret(32, 0x00);
4053 out.scriptPubKey[1] = 0x24;
4054 out.scriptPubKey[2] = 0xaa;
4055 out.scriptPubKey[3] = 0x21;
4056 out.scriptPubKey[4] = 0xa9;
4057 out.scriptPubKey[5] = 0xed;
4058 memcpy(&
out.scriptPubKey[6], witnessroot.
begin(), 32);
4059 commitment = std::vector<unsigned char>(
out.scriptPubKey.begin(),
out.scriptPubKey.end());
4070 return std::all_of(headers.cbegin(), headers.cend(),
4071 [&](
const auto& header) { return CheckProofOfWork(header.GetHash(), header.nBits, consensusParams);});
4082 if (block.
vtx.empty() || !block.
vtx[0]->IsCoinBase()) {
4089 return std::any_of(block.
vtx.begin(), block.
vtx.end(),
4090 [](
auto& tx) { return GetSerializeSize(TX_NO_WITNESS(tx)) == 64; });
4131 assert(pindexPrev !=
nullptr);
4132 const int nHeight = pindexPrev->nHeight + 1;
4140 if (block.
GetBlockTime() <= pindexPrev->GetMedianTimePast())
4179 const int nHeight = pindexPrev ==
nullptr ? 0 : pindexPrev->
nHeight + 1;
4182 bool enforce_locktime_median_time_past{
false};
4184 assert(pindexPrev !=
nullptr);
4185 enforce_locktime_median_time_past =
true;
4188 const int64_t nLockTimeCutoff{enforce_locktime_median_time_past ?
4193 for (
const auto& tx : block.
vtx) {
4203 if (block.
vtx[0]->vin[0].scriptSig.size() <
expect.size() ||
4204 !std::equal(
expect.begin(),
expect.end(), block.
vtx[0]->vin[0].scriptSig.begin())) {
4240 BlockMap::iterator miSelf{
m_blockman.m_block_index.find(hash)};
4242 if (miSelf !=
m_blockman.m_block_index.end()) {
4267 pindexPrev = &((*mi).second);
4277 if (!min_pow_checked) {
4312 blocks_left = std::max<int64_t>(0, blocks_left);
4313 const double progress{100.0 * last_accepted.nHeight / (last_accepted.nHeight + blocks_left)};
4314 LogInfo(
"Synchronizing blockheaders, height: %d (~%.2f%%)\n", last_accepted.nHeight, progress);
4332 if (now < m_last_presync_update + std::chrono::milliseconds{250})
return;
4333 m_last_presync_update = now;
4337 if (initial_download) {
4339 blocks_left = std::max<int64_t>(0, blocks_left);
4340 const double progress{100.0 * height / (height + blocks_left)};
4341 LogInfo(
"Pre-synchronizing blockheaders, height: %d (~%.2f%%)\n", height, progress);
4348 const CBlock& block = *pblock;
4350 if (fNewBlock) *fNewBlock =
false;
4354 CBlockIndex *&pindex = ppindex ? *ppindex : pindexDummy;
4356 bool accepted_header{
AcceptBlockHeader(block, state, &pindex, min_pow_checked)};
4359 if (!accepted_header)
4383 if (fAlreadyHave)
return true;
4385 if (pindex->
nTx != 0)
return true;
4386 if (!fHasMoreOrSameWork)
return true;
4387 if (fTooFarAhead)
return true;
4398 if (!
CheckBlock(block, state, params.GetConsensus()) ||
4414 if (fNewBlock) *fNewBlock =
true;
4422 if (blockPos.IsNull()) {
4423 state.
Error(
strprintf(
"%s: Failed to find position to write new block to disk", __func__));
4428 }
catch (
const std::runtime_error& e) {
4452 if (new_block) *new_block =
false;
4467 ret =
AcceptBlock(block, state, &pindex, force_processing,
nullptr, new_block, min_pow_checked);
4482 LogError(
"%s: ActivateBestChain failed (%s)\n", __func__, state.
ToString());
4488 if (bg_chain && !bg_chain->ActivateBestChain(bg_state, block)) {
4489 LogError(
"%s: [background] ActivateBestChain failed (%s)\n", __func__, bg_state.
ToString());
4514 const bool check_pow,
4515 const bool check_merkle_root)
4526 state.
Invalid({},
"inconclusive-not-best-prevblk");
4567 index_dummy.pprev = tip;
4569 index_dummy.phashBlock = &block_hash;
4573 if(!chainstate.
ConnectBlock(block, state, &index_dummy, view_dummy,
true)) {
4618 target = target->pprev;
4628 LogInfo(
"Loaded best chain: hashBestChain=%s height=%d date=%s progress=%f",
4663 int nCheckLevel,
int nCheckDepth)
4672 if (nCheckDepth <= 0 || nCheckDepth > chainstate.
m_chain.
Height()) {
4675 nCheckLevel = std::max(0, std::min(4, nCheckLevel));
4676 LogInfo(
"Verifying last %i blocks at level %i", nCheckDepth, nCheckLevel);
4680 int nGoodTransactions = 0;
4683 bool skipped_no_block_data{
false};
4684 bool skipped_l3_checks{
false};
4685 LogInfo(
"Verification progress: 0%%");
4690 const int percentageDone = std::max(1, std::min(99, (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (
double)nCheckDepth * (nCheckLevel >= 4 ? 50 : 100))));
4691 if (reportDone < percentageDone / 10) {
4693 LogInfo(
"Verification progress: %d%%", percentageDone);
4694 reportDone = percentageDone / 10;
4703 LogInfo(
"Block verification stopping at height %d (no data). This could be due to pruning or use of an assumeutxo snapshot.", pindex->
nHeight);
4704 skipped_no_block_data =
true;
4714 if (nCheckLevel >= 1 && !
CheckBlock(block, state, consensus_params)) {
4715 LogError(
"Verification error: found bad block at %d, hash=%s (%s)",
4720 if (nCheckLevel >= 2 && pindex) {
4732 if (nCheckLevel >= 3) {
4741 nGoodTransactions = 0;
4742 pindexFailure = pindex;
4744 nGoodTransactions += block.
vtx.size();
4747 skipped_l3_checks =
true;
4752 if (pindexFailure) {
4753 LogError(
"Verification error: coin database inconsistencies found (last %i blocks, %i good transactions before that)", chainstate.
m_chain.
Height() - pindexFailure->
nHeight + 1, nGoodTransactions);
4756 if (skipped_l3_checks) {
4757 LogWarning(
"Skipped verification of level >=3 (insufficient database cache size). Consider increasing -dbcache.");
4764 if (nCheckLevel >= 4 && !skipped_l3_checks) {
4766 const int percentageDone = std::max(1, std::min(99, 100 - (
int)(((
double)(chainstate.
m_chain.
Height() - pindex->
nHeight)) / (
double)nCheckDepth * 50)));
4767 if (reportDone < percentageDone / 10) {
4769 LogInfo(
"Verification progress: %d%%", percentageDone);
4770 reportDone = percentageDone / 10;
4779 if (!chainstate.
ConnectBlock(block, state, pindex, coins)) {
4787 LogInfo(
"Verification: No coin database inconsistencies in last %i blocks (%i transactions)", block_count, nGoodTransactions);
4789 if (skipped_l3_checks) {
4792 if (skipped_no_block_data) {
4810 if (!tx->IsCoinBase()) {
4811 for (
const CTxIn &txin : tx->vin) {
4829 if (hashHeads.empty())
return true;
4830 if (hashHeads.size() != 2) {
4831 LogError(
"ReplayBlocks(): unknown inconsistent state\n");
4842 if (
m_blockman.m_block_index.count(hashHeads[0]) == 0) {
4843 LogError(
"ReplayBlocks(): reorganization to unknown block requested\n");
4846 pindexNew = &(
m_blockman.m_block_index[hashHeads[0]]);
4848 if (!hashHeads[1].IsNull()) {
4849 if (
m_blockman.m_block_index.count(hashHeads[1]) == 0) {
4850 LogError(
"ReplayBlocks(): reorganization from unknown block requested\n");
4853 pindexOld = &(
m_blockman.m_block_index[hashHeads[1]]);
4855 assert(pindexFork !=
nullptr);
4859 const int nForkHeight{pindexFork ? pindexFork->
nHeight : 0};
4860 if (pindexOld != pindexFork) {
4862 while (pindexOld != pindexFork) {
4869 if (pindexOld->
nHeight % 10'000 == 0) {
4882 pindexOld = pindexOld->
pprev;
4888 if (nForkHeight < pindexNew->
nHeight) {
4920 block = block->pprev;
4926void Chainstate::ClearBlockIndexCandidates()
4938 if (!
ret)
return false;
4940 m_blockman.ScanAndUnlinkAlreadyPrunedFiles();
4942 std::vector<CBlockIndex*> vSortedByHeight{
m_blockman.GetAllBlockIndices()};
4943 std::sort(vSortedByHeight.begin(), vSortedByHeight.end(),
4953 if (pindex == GetSnapshotBaseBlock() ||
4958 chainstate->TryAddBlockIndexCandidate(pindex);
4962 m_best_invalid = pindex;
4965 m_best_header = pindex;
4981 if (
m_blockman.m_block_index.count(params.GenesisBlock().GetHash()))
4985 const CBlock& block = params.GenesisBlock();
4987 if (blockPos.IsNull()) {
4988 LogError(
"%s: writing genesis block to disk failed\n", __func__);
4993 }
catch (
const std::runtime_error& e) {
4994 LogError(
"%s: failed to write genesis block: %s\n", __func__, e.what());
5004 std::multimap<uint256, FlatFilePos>* blocks_with_unknown_parent)
5007 assert(!dbp == !blocks_with_unknown_parent);
5009 const auto start{SteadyClock::now()};
5017 uint64_t nRewind = blkdat.GetPos();
5018 while (!blkdat.eof()) {
5021 blkdat.SetPos(nRewind);
5024 unsigned int nSize = 0;
5028 blkdat.FindByte(std::byte(params.MessageStart()[0]));
5029 nRewind = blkdat.GetPos() + 1;
5031 if (buf != params.MessageStart()) {
5038 }
catch (
const std::exception&) {
5045 const uint64_t nBlockPos{blkdat.GetPos()};
5047 dbp->
nPos = nBlockPos;
5048 blkdat.SetLimit(nBlockPos + nSize);
5054 nRewind = nBlockPos + nSize;
5055 blkdat.SkipTo(nRewind);
5057 std::shared_ptr<CBlock> pblock{};
5065 if (dbp && blocks_with_unknown_parent) {
5066 blocks_with_unknown_parent->emplace(header.
hashPrevBlock, *dbp);
5075 blkdat.SetPos(nBlockPos);
5076 pblock = std::make_shared<CBlock>();
5078 nRewind = blkdat.GetPos();
5081 if (
AcceptBlock(pblock, state,
nullptr,
true, dbp,
nullptr,
true)) {
5087 }
else if (hash != params.GetConsensus().hashGenesisBlock && pindex->
nHeight % 1000 == 0) {
5114 bool activation_failure =
false;
5115 for (
auto c :
GetAll()) {
5117 if (!c->ActivateBestChain(state, pblock)) {
5119 activation_failure =
true;
5123 if (activation_failure) {
5130 if (!blocks_with_unknown_parent)
continue;
5133 std::deque<uint256> queue;
5134 queue.push_back(hash);
5135 while (!queue.empty()) {
5138 auto range = blocks_with_unknown_parent->equal_range(head);
5139 while (range.first != range.second) {
5140 std::multimap<uint256, FlatFilePos>::iterator it = range.first;
5141 std::shared_ptr<CBlock> pblockrecursive = std::make_shared<CBlock>();
5143 const auto& block_hash{pblockrecursive->GetHash()};
5147 if (
AcceptBlock(pblockrecursive, dummy,
nullptr,
true, &it->second,
nullptr,
true)) {
5149 queue.push_back(block_hash);
5153 blocks_with_unknown_parent->erase(it);
5157 }
catch (
const std::exception& e) {
5169 LogDebug(
BCLog::REINDEX,
"%s: unexpected data at file offset 0x%x - %s. continuing\n", __func__, (nRewind - 1), e.what());
5172 }
catch (
const std::runtime_error& e) {
5175 LogInfo(
"Loaded %i blocks from external file in %dms", nLoaded, Ticks<std::chrono::milliseconds>(SteadyClock::now() - start));
5210 best_hdr_chain.
SetTip(*m_best_header);
5212 std::multimap<const CBlockIndex*, const CBlockIndex*> forward;
5213 for (
auto& [
_, block_index] :
m_blockman.m_block_index) {
5215 if (!best_hdr_chain.
Contains(&block_index)) {
5217 assert(block_index.pprev);
5218 forward.emplace(block_index.pprev, &block_index);
5232 const CBlockIndex* pindexFirstNeverProcessed =
nullptr;
5233 const CBlockIndex* pindexFirstNotTreeValid =
nullptr;
5234 const CBlockIndex* pindexFirstNotTransactionsValid =
nullptr;
5235 const CBlockIndex* pindexFirstNotChainValid =
nullptr;
5236 const CBlockIndex* pindexFirstNotScriptsValid =
nullptr;
5242 const CBlockIndex* snap_base{GetSnapshotBaseBlock()};
5243 const CBlockIndex *snap_first_missing{}, *snap_first_notx{}, *snap_first_notv{}, *snap_first_nocv{}, *snap_first_nosv{};
5244 auto snap_update_firsts = [&] {
5245 if (pindex == snap_base) {
5246 std::swap(snap_first_missing, pindexFirstMissing);
5247 std::swap(snap_first_notx, pindexFirstNeverProcessed);
5248 std::swap(snap_first_notv, pindexFirstNotTransactionsValid);
5249 std::swap(snap_first_nocv, pindexFirstNotChainValid);
5250 std::swap(snap_first_nosv, pindexFirstNotScriptsValid);
5254 while (pindex !=
nullptr) {
5256 if (pindexFirstInvalid ==
nullptr && pindex->nStatus &
BLOCK_FAILED_VALID) pindexFirstInvalid = pindex;
5257 if (pindexFirstMissing ==
nullptr && !(pindex->nStatus &
BLOCK_HAVE_DATA)) {
5258 pindexFirstMissing = pindex;
5260 if (pindexFirstNeverProcessed ==
nullptr && pindex->
nTx == 0) pindexFirstNeverProcessed = pindex;
5263 if (pindex->
pprev !=
nullptr) {
5264 if (pindexFirstNotTransactionsValid ==
nullptr &&
5266 pindexFirstNotTransactionsValid = pindex;
5269 if (pindexFirstNotChainValid ==
nullptr &&
5271 pindexFirstNotChainValid = pindex;
5274 if (pindexFirstNotScriptsValid ==
nullptr &&
5276 pindexFirstNotScriptsValid = pindex;
5281 if (pindex->
pprev ==
nullptr) {
5284 for (
const Chainstate* c : {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
5285 if (c && c->m_chain.Genesis() !=
nullptr) {
5286 assert(pindex == c->m_chain.Genesis());
5298 assert(pindexFirstMissing == pindexFirstNeverProcessed);
5304 if (snap_base && snap_base->GetAncestor(pindex->
nHeight) == pindex) {
5314 assert((pindexFirstNotTransactionsValid ==
nullptr || pindex == snap_base) == pindex->
HaveNumChainTxs());
5318 assert(pindexFirstNotTreeValid ==
nullptr);
5322 if (pindexFirstInvalid ==
nullptr) {
5329 if (!pindex->
pprev) {
5344 for (
const Chainstate* c : {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
5345 if (!c || c->m_chain.Tip() ==
nullptr)
continue;
5359 if (!
CBlockIndexWorkComparator()(pindex, c->m_chain.Tip()) && (pindexFirstNeverProcessed ==
nullptr || pindex == snap_base)) {
5363 if (pindexFirstInvalid ==
nullptr) {
5382 if (pindexFirstMissing ==
nullptr || pindex == c->m_chain.Tip() || pindex == c->SnapshotBase()) {
5402 bool foundInUnlinked =
false;
5403 while (rangeUnlinked.first != rangeUnlinked.second) {
5404 assert(rangeUnlinked.first->first == pindex->
pprev);
5405 if (rangeUnlinked.first->second == pindex) {
5406 foundInUnlinked =
true;
5409 rangeUnlinked.first++;
5411 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed !=
nullptr && pindexFirstInvalid ==
nullptr) {
5416 if (pindexFirstMissing ==
nullptr)
assert(!foundInUnlinked);
5417 if (pindex->
pprev && (pindex->nStatus &
BLOCK_HAVE_DATA) && pindexFirstNeverProcessed ==
nullptr && pindexFirstMissing !=
nullptr) {
5428 for (
const Chainstate* c : {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
5432 if (pindexFirstInvalid ==
nullptr) {
5433 if (is_active || snap_base->GetAncestor(pindex->
nHeight) == pindex) {
5445 snap_update_firsts();
5446 auto range{forward.equal_range(pindex)};
5447 if (range.first != range.second) {
5449 pindex = range.first->second;
5452 }
else if (best_hdr_chain.
Contains(pindex)) {
5455 pindex = best_hdr_chain[
nHeight];
5463 snap_update_firsts();
5465 if (pindex == pindexFirstInvalid) pindexFirstInvalid =
nullptr;
5466 if (pindex == pindexFirstMissing) pindexFirstMissing =
nullptr;
5467 if (pindex == pindexFirstNeverProcessed) pindexFirstNeverProcessed =
nullptr;
5468 if (pindex == pindexFirstNotTreeValid) pindexFirstNotTreeValid =
nullptr;
5469 if (pindex == pindexFirstNotTransactionsValid) pindexFirstNotTransactionsValid =
nullptr;
5470 if (pindex == pindexFirstNotChainValid) pindexFirstNotChainValid =
nullptr;
5471 if (pindex == pindexFirstNotScriptsValid) pindexFirstNotScriptsValid =
nullptr;
5475 auto rangePar{forward.equal_range(pindexPar)};
5476 while (rangePar.first->second != pindex) {
5477 assert(rangePar.first != rangePar.second);
5482 if (rangePar.first != rangePar.second) {
5484 pindex = rangePar.first->second;
5486 }
else if (pindexPar == best_hdr_chain[
nHeight - 1]) {
5488 pindex = best_hdr_chain[
nHeight];
5490 assert((pindex ==
nullptr) == (pindexPar == best_hdr_chain.
Tip()));
5502 assert(nNodes == forward.size() + best_hdr_chain.
Height() + 1);
5509 return strprintf(
"Chainstate [%s] @ height %d (%s)",
5514bool Chainstate::ResizeCoinsCaches(
size_t coinstip_size,
size_t coinsdb_size)
5527 LogInfo(
"[%s] resized coinsdb cache to %.1f MiB",
5528 this->
ToString(), coinsdb_size * (1.0 / 1024 / 1024));
5529 LogInfo(
"[%s] resized coinstip cache to %.1f MiB",
5530 this->
ToString(), coinstip_size * (1.0 / 1024 / 1024));
5535 if (coinstip_size > old_coinstip_size) {
5549 if (pindex ==
nullptr) {
5558 const int64_t nNow{TicksSinceEpoch<std::chrono::seconds>(
NodeClock::now())};
5559 const auto block_time{
5560 (
Assume(m_best_header) && std::abs(nNow - pindex->
GetBlockTime()) <= Ticks<std::chrono::seconds>(2h) &&
5574 fTxTotal =
data.tx_count + (nNow -
data.nTime) *
data.dTxRate;
5585 if (m_active_chainstate && m_active_chainstate->m_from_snapshot_blockhash) {
5587 return m_active_chainstate->m_from_snapshot_blockhash;
5589 return std::nullopt;
5595 std::vector<Chainstate*>
out;
5597 for (
Chainstate*
cs : {m_ibd_chainstate.get(), m_snapshot_chainstate.get()}) {
5607 assert(!m_ibd_chainstate);
5608 assert(!m_active_chainstate);
5610 m_ibd_chainstate = std::make_unique<Chainstate>(mempool,
m_blockman, *
this);
5611 m_active_chainstate = m_ibd_chainstate.get();
5612 return *m_active_chainstate;
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);
5671 if (!
GetParams().AssumeutxoForBlockhash(base_blockhash).has_value()) {
5673 std::string heights_formatted =
util::Join(available_heights,
", ", [&](
const auto& i) {
return util::ToString(i); });
5674 return util::Error{
Untranslated(
strprintf(
"assumeutxo block hash in snapshot metadata not recognized (hash: %s). The following snapshot heights are available: %s",
5676 heights_formatted))};
5680 if (!snapshot_start_block) {
5681 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",
5686 if (start_block_invalid) {
5690 if (!m_best_header || m_best_header->GetAncestor(snapshot_start_block->nHeight) != snapshot_start_block) {
5691 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 auto mempool{m_active_chainstate->GetMempool()};
5695 if (mempool && mempool->
size() > 0) {
5700 int64_t current_coinsdb_cache_size{0};
5701 int64_t current_coinstip_cache_size{0};
5709 static constexpr double IBD_CACHE_PERC = 0.01;
5710 static constexpr double SNAPSHOT_CACHE_PERC = 0.99;
5722 current_coinsdb_cache_size = this->
ActiveChainstate().m_coinsdb_cache_size_bytes;
5723 current_coinstip_cache_size = this->
ActiveChainstate().m_coinstip_cache_size_bytes;
5728 static_cast<size_t>(current_coinstip_cache_size * IBD_CACHE_PERC),
5729 static_cast<size_t>(current_coinsdb_cache_size * IBD_CACHE_PERC));
5733 return std::make_unique<Chainstate>(
5734 nullptr,
m_blockman, *
this, base_blockhash));
5738 snapshot_chainstate->InitCoinsDB(
5739 static_cast<size_t>(current_coinsdb_cache_size * SNAPSHOT_CACHE_PERC),
5740 in_memory,
false,
"chainstate");
5741 snapshot_chainstate->InitCoinsCache(
5742 static_cast<size_t>(current_coinstip_cache_size * SNAPSHOT_CACHE_PERC));
5746 this->MaybeRebalanceCaches();
5754 snapshot_chainstate.reset();
5758 "Manually remove it before restarting.\n"),
fs::PathToString(*snapshot_datadir)));
5775 return cleanup_bad_snapshot(
Untranslated(
"work does not exceed active chainstate"));
5781 return cleanup_bad_snapshot(
Untranslated(
"could not write base blockhash"));
5785 assert(!m_snapshot_chainstate);
5786 m_snapshot_chainstate.swap(snapshot_chainstate);
5787 const bool chaintip_loaded = m_snapshot_chainstate->LoadChainTip();
5792 Assert(m_active_chainstate->m_mempool->size() == 0);
5793 Assert(!m_snapshot_chainstate->m_mempool);
5794 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
5795 m_active_chainstate->m_mempool =
nullptr;
5796 m_active_chainstate = m_snapshot_chainstate.get();
5799 LogInfo(
"[snapshot] successfully activated snapshot %s", base_blockhash.
ToString());
5800 LogInfo(
"[snapshot] (%.2f MB)",
5801 m_snapshot_chainstate->CoinsTip().DynamicMemoryUsage() / (1000 * 1000));
5803 this->MaybeRebalanceCaches();
5804 return snapshot_start_block;
5811 snapshot_loaded ?
"saving snapshot chainstate" :
"flushing coins cache",
5815 coins_cache.
Flush();
5820 const char*
what() const noexcept
override
5822 return "ComputeUTXOStats interrupted.";
5844 if (!snapshot_start_block) {
5851 int base_height = snapshot_start_block->
nHeight;
5854 if (!maybe_au_data) {
5856 "(%d) - refusing to load snapshot", base_height))};
5871 LogInfo(
"[snapshot] loading %d coins from snapshot %s", coins_left, base_blockhash.
ToString());
5872 int64_t coins_processed{0};
5874 while (coins_left > 0) {
5878 size_t coins_per_txid{0};
5881 if (coins_per_txid > coins_left) {
5885 for (
size_t i = 0; i < coins_per_txid; i++) {
5889 outpoint.
hash = txid;
5891 if (coin.
nHeight > base_height ||
5892 outpoint.
n >= std::numeric_limits<
decltype(outpoint.
n)>::max()
5895 coins_count - coins_left))};
5899 coins_count - coins_left))};
5906 if (coins_processed % 1000000 == 0) {
5907 LogInfo(
"[snapshot] %d coins loaded (%.2f%%, %.2f MB)",
5909 static_cast<float>(coins_processed) * 100 /
static_cast<float>(coins_count),
5917 if (coins_processed % 120000 == 0) {
5923 return snapshot_chainstate.GetCoinsCacheSizeState());
5936 }
catch (
const std::ios_base::failure&) {
5949 bool out_of_coins{
false};
5951 std::byte left_over_byte;
5952 coins_file >> left_over_byte;
5953 }
catch (
const std::ios_base::failure&) {
5955 out_of_coins =
true;
5957 if (!out_of_coins) {
5962 LogInfo(
"[snapshot] loaded %d (%.2f MB) coins from snapshot %s",
5976 std::optional<CCoinsStats> maybe_stats;
5984 if (!maybe_stats.has_value()) {
6005 constexpr int AFTER_GENESIS_START{1};
6007 for (
int i = AFTER_GENESIS_START; i <= snapshot_chainstate.
m_chain.
Height(); ++i) {
6008 index = snapshot_chainstate.
m_chain[i];
6025 assert(index == snapshot_start_block);
6029 LogInfo(
"[snapshot] validated snapshot (%.2f MB)",
6051 if (m_ibd_chainstate.get() == &this->ActiveChainstate() ||
6052 !this->IsUsable(m_snapshot_chainstate.get()) ||
6053 !this->IsUsable(m_ibd_chainstate.get()) ||
6054 !m_ibd_chainstate->m_chain.Tip()) {
6060 const int snapshot_base_height = *
Assert(this->GetSnapshotBaseHeight());
6063 if (index_new.
nHeight < snapshot_base_height) {
6073 "%s failed to validate the -assumeutxo snapshot state. "
6074 "This indicates a hardware problem, or a bug in the software, or a "
6075 "bad software modification that allowed an invalid snapshot to be "
6076 "loaded. As a result of this, the node will shut down and stop using any "
6077 "state that was built on the snapshot, resetting the chain height "
6078 "from %d to %d. On the next "
6079 "restart, the node will resume syncing from %d "
6080 "without using any snapshot data. "
6081 "Please report this incident to %s, including how you obtained the snapshot. "
6082 "The invalid snapshot chainstate will be left on disk in case it is "
6083 "helpful in diagnosing the issue that caused this error."),
6084 CLIENT_NAME, snapshot_tip_height, snapshot_base_height, snapshot_base_height, CLIENT_BUGREPORT
6088 LogError(
"[snapshot] deleting snapshot, reverting to validated chain, and stopping node\n");
6090 m_active_chainstate = m_ibd_chainstate.get();
6091 m_snapshot_chainstate->m_disabled =
true;
6095 auto rename_result = m_snapshot_chainstate->InvalidateCoinsDBOnDisk();
6096 if (!rename_result) {
6104 LogWarning(
"[snapshot] supposed base block %s does not match the "
6105 "snapshot base block %s (height %d). Snapshot is not valid.",
6106 index_new.
ToString(), snapshot_blockhash.
ToString(), snapshot_base_height);
6107 handle_invalid_snapshot();
6113 int curr_height = m_ibd_chainstate->m_chain.Height();
6115 assert(snapshot_base_height == curr_height);
6120 CCoinsViewDB& ibd_coins_db = m_ibd_chainstate->CoinsDB();
6121 m_ibd_chainstate->ForceFlushStateToDisk();
6124 if (!maybe_au_data) {
6125 LogWarning(
"[snapshot] assumeutxo data not found for height "
6126 "(%d) - refusing to validate snapshot", curr_height);
6127 handle_invalid_snapshot();
6132 std::optional<CCoinsStats> maybe_ibd_stats;
6133 LogInfo(
"[snapshot] computing UTXO stats for background chainstate to validate "
6134 "snapshot - this could take a few minutes");
6137 CoinStatsHashType::HASH_SERIALIZED,
6146 if (!maybe_ibd_stats) {
6147 LogWarning(
"[snapshot] failed to generate stats for validation coins db");
6151 handle_invalid_snapshot();
6154 const auto& ibd_stats = *maybe_ibd_stats;
6163 LogWarning(
"[snapshot] hash mismatch: actual=%s, expected=%s",
6164 ibd_stats.hashSerialized.ToString(),
6166 handle_invalid_snapshot();
6170 LogInfo(
"[snapshot] snapshot beginning at %s has been fully validated",
6173 m_ibd_chainstate->m_disabled =
true;
6174 this->MaybeRebalanceCaches();
6182 assert(m_active_chainstate);
6183 return *m_active_chainstate;
6189 return m_snapshot_chainstate && m_active_chainstate == m_snapshot_chainstate.get();
6192void ChainstateManager::MaybeRebalanceCaches()
6195 bool ibd_usable = this->
IsUsable(m_ibd_chainstate.get());
6196 bool snapshot_usable = this->
IsUsable(m_snapshot_chainstate.get());
6197 assert(ibd_usable || snapshot_usable);
6199 if (ibd_usable && !snapshot_usable) {
6204 else if (snapshot_usable && !ibd_usable) {
6206 LogInfo(
"[snapshot] allocating all cache to the snapshot chainstate");
6210 else if (ibd_usable && snapshot_usable) {
6215 m_ibd_chainstate->ResizeCoinsCaches(
6217 m_snapshot_chainstate->ResizeCoinsCaches(
6220 m_snapshot_chainstate->ResizeCoinsCaches(
6222 m_ibd_chainstate->ResizeCoinsCaches(
6228void ChainstateManager::ResetChainstates()
6230 m_ibd_chainstate.reset();
6231 m_snapshot_chainstate.reset();
6232 m_active_chainstate =
nullptr;
6242 if (!opts.check_block_index.has_value()) opts.
check_block_index = opts.chainparams.DefaultConsistencyChecks();
6243 if (!opts.minimum_chain_work.has_value()) opts.minimum_chain_work =
UintToArith256(opts.chainparams.GetConsensus().nMinimumChainWork);
6244 if (!opts.assumed_valid_block.has_value()) opts.assumed_valid_block = opts.chainparams.GetConsensus().defaultAssumeValid;
6245 return std::move(opts);
6250 m_interrupt{interrupt},
6252 m_blockman{interrupt,
std::move(blockman_options)},
6253 m_validation_cache{m_options.script_execution_cache_bytes, m_options.signature_cache_bytes}
6264bool ChainstateManager::DetectSnapshotChainstate()
6266 assert(!m_snapshot_chainstate);
6272 if (!base_blockhash) {
6275 LogInfo(
"[snapshot] detected active snapshot chainstate (%s) - loading",
6276 fs::PathToString(*path));
6278 this->ActivateExistingSnapshot(*base_blockhash);
6284 assert(!m_snapshot_chainstate);
6285 m_snapshot_chainstate =
6286 std::make_unique<Chainstate>(
nullptr,
m_blockman, *
this, base_blockhash);
6287 LogInfo(
"[snapshot] switching active chainstate to %s", m_snapshot_chainstate->ToString());
6290 Assert(m_active_chainstate->m_mempool->size() == 0);
6291 Assert(!m_snapshot_chainstate->m_mempool);
6292 m_snapshot_chainstate->m_mempool = m_active_chainstate->m_mempool;
6293 m_active_chainstate->m_mempool =
nullptr;
6294 m_active_chainstate = m_snapshot_chainstate.get();
6295 return *m_snapshot_chainstate;
6300 return (block_index.
nHeight==91842 && block_index.
GetBlockHash() ==
uint256{
"00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec"}) ||
6301 (block_index.
nHeight==91880 && block_index.
GetBlockHash() ==
uint256{
"00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721"});
6306 return (block_height==91722 && block_hash ==
uint256{
"00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e"}) ||
6307 (block_height==91812 && block_hash ==
uint256{
"00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f"});
6314 assert(
cs.m_from_snapshot_blockhash);
6315 auto storage_path_maybe =
cs.CoinsDB().StoragePath();
6317 assert(storage_path_maybe);
6318 return *storage_path_maybe;
6328 auto invalid_path = snapshot_datadir +
"_INVALID";
6331 LogInfo(
"[snapshot] renaming snapshot datadir %s to %s", dbpath, target);
6337 fs::rename(snapshot_datadir, invalid_path);
6338 }
catch (
const fs::filesystem_error& e) {
6342 LogError(
"While invalidating the coins db: Error renaming file '%s' -> '%s': %s",
6343 src_str, dest_str, e.what());
6345 "Rename of '%s' -> '%s' failed. "
6346 "You should resolve this by manually moving or deleting the invalid "
6347 "snapshot directory %s, otherwise you will encounter the same error again "
6348 "on the next startup."),
6349 src_str, dest_str, src_str)};
6354bool ChainstateManager::DeleteSnapshotChainstate()
6357 Assert(m_snapshot_chainstate);
6358 Assert(m_ibd_chainstate);
6362 LogError(
"Deletion of %s failed. Please remove it manually to continue reindexing.",
6363 fs::PathToString(snapshot_datadir));
6366 m_active_chainstate = m_ibd_chainstate.get();
6367 m_active_chainstate->m_mempool = m_snapshot_chainstate->m_mempool;
6368 m_snapshot_chainstate.reset();
6382const CBlockIndex* ChainstateManager::GetSnapshotBaseBlock()
const
6384 return m_active_chainstate ? m_active_chainstate->SnapshotBase() :
nullptr;
6387std::optional<int> ChainstateManager::GetSnapshotBaseHeight()
const
6389 const CBlockIndex* base = this->GetSnapshotBaseBlock();
6390 return base ? std::make_optional(base->
nHeight) :
std::nullopt;
6393void ChainstateManager::RecalculateBestHeader()
6397 for (
auto& entry :
m_blockman.m_block_index) {
6398 if (!(entry.second.nStatus &
BLOCK_FAILED_MASK) && m_best_header->nChainWork < entry.second.nChainWork) {
6399 m_best_header = &entry.second;
6404bool ChainstateManager::ValidatedSnapshotCleanup()
6408 if (!(chainstate && chainstate->HasCoinsViews())) {
6411 return chainstate->CoinsDB().StoragePath();
6413 std::optional<fs::path> ibd_chainstate_path_maybe = get_storage_path(m_ibd_chainstate);
6414 std::optional<fs::path> snapshot_chainstate_path_maybe = get_storage_path(m_snapshot_chainstate);
6423 if (!ibd_chainstate_path_maybe || !snapshot_chainstate_path_maybe) {
6424 LogError(
"[snapshot] snapshot chainstate cleanup cannot happen with "
6425 "in-memory chainstates. You are testing, right?");
6429 const auto& snapshot_chainstate_path = *snapshot_chainstate_path_maybe;
6430 const auto& ibd_chainstate_path = *ibd_chainstate_path_maybe;
6438 this->ResetChainstates();
6443 LogInfo(
"[snapshot] deleting background chainstate directory (now unnecessary) (%s)",
6444 fs::PathToString(ibd_chainstate_path));
6446 fs::path tmp_old{ibd_chainstate_path +
"_todelete"};
6448 auto rename_failed_abort = [
this](
6451 const fs::filesystem_error& err) {
6452 LogError(
"[snapshot] Error renaming path (%s) -> (%s): %s\n",
6453 fs::PathToString(p_old), fs::PathToString(p_new), err.what());
6455 "Rename of '%s' -> '%s' failed. "
6456 "Cannot clean up the background chainstate leveldb directory."),
6457 fs::PathToString(p_old), fs::PathToString(p_new)));
6461 fs::rename(ibd_chainstate_path, tmp_old);
6462 }
catch (
const fs::filesystem_error& e) {
6463 rename_failed_abort(ibd_chainstate_path, tmp_old, e);
6467 LogInfo(
"[snapshot] moving snapshot chainstate (%s) to "
6468 "default chainstate directory (%s)",
6469 fs::PathToString(snapshot_chainstate_path), fs::PathToString(ibd_chainstate_path));
6472 fs::rename(snapshot_chainstate_path, ibd_chainstate_path);
6473 }
catch (
const fs::filesystem_error& e) {
6474 rename_failed_abort(snapshot_chainstate_path, ibd_chainstate_path, e);
6481 LogWarning(
"Deletion of %s failed. Please remove it manually, as the "
6482 "directory is now unnecessary.",
6483 fs::PathToString(tmp_old));
6485 LogInfo(
"[snapshot] deleted background chainstate directory (%s)",
6486 fs::PathToString(ibd_chainstate_path));
6491Chainstate& ChainstateManager::GetChainstateForIndexing()
6496 return (this->
GetAll().size() > 1) ? *m_ibd_chainstate : *m_active_chainstate;
6499std::pair<int, int> ChainstateManager::GetPruneRange(
const Chainstate& chainstate,
int last_height_can_prune)
6506 if (this->
GetAll().size() > 1 && m_snapshot_chainstate.get() == &chainstate) {
6509 prune_start = *
Assert(GetSnapshotBaseHeight()) + 1;
6512 int max_prune = std::max<int>(
6521 int prune_end = std::min(last_height_can_prune, max_prune);
6523 return {prune_start, prune_end};
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 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.
std::string ToString() const
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)
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.
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...
ChainstateRole GetRole() const EXCLUSIVE_LOCKS_REQUIRED(void InitCoinsDB(size_t cache_size_bytes, bool in_memory, bool should_wipe, fs::path leveldb_name="chainstate")
Return the current role of the chainstate.
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.
const CBlockIndex *SnapshotBase() const EXCLUSIVE_LOCKS_REQUIRED(std::set< CBlockIndex *, node::CBlockIndexWorkComparator > setBlockIndexCandidates
The base of the snapshot this chainstate was created from.
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
Try to make some progress towards making pindexMostWork the active block.
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)
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.
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
util::Result< void > PopulateAndValidateSnapshot(Chainstate &snapshot_chainstate, AutoFile &coins_file, const node::SnapshotMetadata &metadata)
Internal helper for ActivateSnapshot().
const uint256 & AssumedValidBlock() const
SnapshotCompletionResult MaybeCompleteSnapshotValidation() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetSnapshotBaseBlock() const EXCLUSIVE_LOCKS_REQUIRED(Chainstate ActiveChainstate)() const
Once the background validation chainstate has reached the height which is the base of the UTXO snapsh...
ValidationCache m_validation_cache
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).
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.
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.
bool IsSnapshotValidated() const EXCLUSIVE_LOCKS_REQUIRED(
Is there a snapshot in use and has it been fully validated?
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
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.
bool BackgroundSyncInProgress() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
The state of a background sync (for net processing)
size_t m_total_coinsdb_cache
The total number of bytes available for us to use across all leveldb coins databases.
std::atomic< bool > m_cached_finished_ibd
Whether initial block download has ended and IsInitialBlockDownload should return false from now on.
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.
bool IsSnapshotActive() const
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
util::Result< CBlockIndex * > ActivateSnapshot(AutoFile &coins_file, const node::SnapshotMetadata &metadata, bool in_memory)
Construct and activate a Chainstate on the basis of UTXO snapshot data.
bool LoadBlockIndex() EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Load the block tree and coins database from disk, initializing state if we're running with -reindex.
std::optional< uint256 > SnapshotBlockhash() const
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())
Chainstate &InitializeChainstate(CTxMemPool *mempool) EXCLUSIVE_LOCKS_REQUIRED(std::vector< Chainstate * GetAll)()
Instantiate a new chainstate.
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.
bool DetectSnapshotChainstate() EXCLUSIVE_LOCKS_REQUIRED(void ResetChainstates() EXCLUSIVE_LOCKS_REQUIRED(bool DeleteSnapshotChainstate() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &ActivateExistingSnapshot(uint256 base_blockhash) EXCLUSIVE_LOCKS_REQUIRED(bool ValidatedSnapshotCleanup() EXCLUSIVE_LOCKS_REQUIRED(Chainstate &GetChainstateForIndexing() EXCLUSIVE_LOCKS_REQUIRED(std::pair< int, int > GetPruneRange(const Chainstate &chainstate, int last_height_can_prune) EXCLUSIVE_LOCKS_REQUIRED(std::optional< int > GetSnapshotBaseHeight() const EXCLUSIVE_LOCKS_REQUIRED(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...
bool IsUsable(const Chainstate *const cs) const EXCLUSIVE_LOCKS_REQUIRED(
Return true if a chainstate is considered usable.
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 BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockChecked(const std::shared_ptr< const CBlock > &, const BlockValidationState &)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void ChainStateFlushed(ChainstateRole, const CBlockLocator &)
void ActiveTipChange(const CBlockIndex &, bool)
void BlockConnected(ChainstateRole, const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
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.
void FindFilesToPruneManual(std::set< int > &setFilesToPrune, int nManualPruneHeight, const Chainstate &chain, ChainstateManager &chainman)
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.
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
ChainstateRole
This enum describes the various roles a specific Chainstate instance can take.
#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)
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)
std::optional< fs::path > FindSnapshotChainstateDir(const fs::path &data_dir)
Return a path to the snapshot-based chainstate dir, if one exists.
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
#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 fs::path GetSnapshotCoinsDBPath(Chainstate &cs) EXCLUSIVE_LOCKS_REQUIRED(
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...
@ BASE_BLOCKHASH_MISMATCH
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.