diff --git a/src/index/coinstatsindex.cpp b/src/index/coinstatsindex.cpp index a9c2282b5b..016f7d770f 100644 --- a/src/index/coinstatsindex.cpp +++ b/src/index/coinstatsindex.cpp @@ -37,7 +37,6 @@ struct DBVal { CAmount total_new_outputs_ex_coinbase_amount; CAmount total_coinbase_amount; CAmount total_unspendables_genesis_block; - CAmount total_unspendables_bip30; CAmount total_unspendables_scripts; CAmount total_unspendables_unclaimed_rewards; @@ -53,7 +52,6 @@ struct DBVal { READWRITE(obj.total_new_outputs_ex_coinbase_amount); READWRITE(obj.total_coinbase_amount); READWRITE(obj.total_unspendables_genesis_block); - READWRITE(obj.total_unspendables_bip30); READWRITE(obj.total_unspendables_scripts); READWRITE(obj.total_unspendables_unclaimed_rewards); } @@ -148,13 +146,6 @@ bool CoinStatsIndex::CustomAppend(const interfaces::BlockInfo& block) for (size_t i = 0; i < block.data->vtx.size(); ++i) { const auto& tx{block.data->vtx.at(i)}; - // Skip duplicate txid coinbase transactions (BIP30). - if (IsBIP30Unspendable(*pindex) && tx->IsCoinBase()) { - m_total_unspendable_amount += block_subsidy; - m_total_unspendables_bip30 += block_subsidy; - continue; - } - for (uint32_t j = 0; j < tx->vout.size(); ++j) { const CTxOut& out{tx->vout[j]}; Coin coin{out, block.height, tx->IsCoinBase(), tx->IsCoinStake(), (int)tx->nTime}; @@ -223,7 +214,6 @@ bool CoinStatsIndex::CustomAppend(const interfaces::BlockInfo& block) value.second.total_new_outputs_ex_coinbase_amount = m_total_new_outputs_ex_coinbase_amount; value.second.total_coinbase_amount = m_total_coinbase_amount; value.second.total_unspendables_genesis_block = m_total_unspendables_genesis_block; - value.second.total_unspendables_bip30 = m_total_unspendables_bip30; value.second.total_unspendables_scripts = m_total_unspendables_scripts; value.second.total_unspendables_unclaimed_rewards = m_total_unspendables_unclaimed_rewards; @@ -339,7 +329,6 @@ std::optional CoinStatsIndex::LookUpStats(const CBlockIndex& block_ stats.total_new_outputs_ex_coinbase_amount = entry.total_new_outputs_ex_coinbase_amount; stats.total_coinbase_amount = entry.total_coinbase_amount; stats.total_unspendables_genesis_block = entry.total_unspendables_genesis_block; - stats.total_unspendables_bip30 = entry.total_unspendables_bip30; stats.total_unspendables_scripts = entry.total_unspendables_scripts; stats.total_unspendables_unclaimed_rewards = entry.total_unspendables_unclaimed_rewards; @@ -381,7 +370,6 @@ bool CoinStatsIndex::CustomInit(const std::optional& block m_total_new_outputs_ex_coinbase_amount = entry.total_new_outputs_ex_coinbase_amount; m_total_coinbase_amount = entry.total_coinbase_amount; m_total_unspendables_genesis_block = entry.total_unspendables_genesis_block; - m_total_unspendables_bip30 = entry.total_unspendables_bip30; m_total_unspendables_scripts = entry.total_unspendables_scripts; m_total_unspendables_unclaimed_rewards = entry.total_unspendables_unclaimed_rewards; } @@ -494,7 +482,6 @@ bool CoinStatsIndex::ReverseBlock(const CBlock& block, const CBlockIndex* pindex Assert(m_total_new_outputs_ex_coinbase_amount == read_out.second.total_new_outputs_ex_coinbase_amount); Assert(m_total_coinbase_amount == read_out.second.total_coinbase_amount); Assert(m_total_unspendables_genesis_block == read_out.second.total_unspendables_genesis_block); - Assert(m_total_unspendables_bip30 == read_out.second.total_unspendables_bip30); Assert(m_total_unspendables_scripts == read_out.second.total_unspendables_scripts); Assert(m_total_unspendables_unclaimed_rewards == read_out.second.total_unspendables_unclaimed_rewards); diff --git a/src/index/coinstatsindex.h b/src/index/coinstatsindex.h index 813369eb26..e84a225607 100644 --- a/src/index/coinstatsindex.h +++ b/src/index/coinstatsindex.h @@ -34,7 +34,6 @@ class CoinStatsIndex final : public BaseIndex CAmount m_total_new_outputs_ex_coinbase_amount{0}; CAmount m_total_coinbase_amount{0}; CAmount m_total_unspendables_genesis_block{0}; - CAmount m_total_unspendables_bip30{0}; CAmount m_total_unspendables_scripts{0}; CAmount m_total_unspendables_unclaimed_rewards{0}; diff --git a/src/kernel/coinstats.h b/src/kernel/coinstats.h index c0c363a842..11a1b2b7cd 100644 --- a/src/kernel/coinstats.h +++ b/src/kernel/coinstats.h @@ -60,8 +60,6 @@ struct CCoinsStats { CAmount total_coinbase_amount{0}; //! The unspendable coinbase amount from the genesis block CAmount total_unspendables_genesis_block{0}; - //! The two unspendable coinbase outputs total amount caused by BIP30 - CAmount total_unspendables_bip30{0}; //! Total cumulative amount of outputs sent to unspendable scripts (OP_RETURN for example) up to and including this block CAmount total_unspendables_scripts{0}; //! Total cumulative amount of coins lost due to unclaimed miner rewards up to and including this block diff --git a/src/rpc/blockchain.cpp b/src/rpc/blockchain.cpp index 6280939ac7..f9bff99597 100644 --- a/src/rpc/blockchain.cpp +++ b/src/rpc/blockchain.cpp @@ -871,7 +871,6 @@ static RPCHelpMan gettxoutsetinfo() {RPCResult::Type::OBJ, "unspendables", "Detailed view of the unspendable categories", { {RPCResult::Type::STR_AMOUNT, "genesis_block", "The unspendable amount of the Genesis block subsidy"}, - {RPCResult::Type::STR_AMOUNT, "bip30", "Transactions overridden by duplicates (no longer possible with BIP30)"}, {RPCResult::Type::STR_AMOUNT, "scripts", "Amounts sent to scripts that are unspendable (for example OP_RETURN outputs)"}, {RPCResult::Type::STR_AMOUNT, "unclaimed_rewards", "Fee rewards that miners did not claim in their coinbase transaction"}, }} @@ -975,7 +974,6 @@ static RPCHelpMan gettxoutsetinfo() UniValue unspendables(UniValue::VOBJ); unspendables.pushKV("genesis_block", ValueFromAmount(stats.total_unspendables_genesis_block - prev_stats.total_unspendables_genesis_block)); - unspendables.pushKV("bip30", ValueFromAmount(stats.total_unspendables_bip30 - prev_stats.total_unspendables_bip30)); unspendables.pushKV("scripts", ValueFromAmount(stats.total_unspendables_scripts - prev_stats.total_unspendables_scripts)); unspendables.pushKV("unclaimed_rewards", ValueFromAmount(stats.total_unspendables_unclaimed_rewards - prev_stats.total_unspendables_unclaimed_rewards)); block_info.pushKV("unspendables", unspendables); @@ -1812,9 +1810,9 @@ static RPCHelpMan getblockstats() size_t out_size = GetSerializeSize(out, PROTOCOL_VERSION) + PER_UTXO_OVERHEAD; utxo_size_inc += out_size; - // The Genesis block and the repeated BIP30 block coinbases don't change the UTXO + // The Genesis block doesn't change the UTXO // set counts, so they have to be excluded from the statistics - if (pindex.nHeight == 0 || (IsBIP30Repeat(pindex) && tx->IsCoinBase())) continue; + if (pindex.nHeight == 0) continue; // Skip unspendable outputs since they are not included in the UTXO set if (out.scriptPubKey.IsUnspendable()) continue; diff --git a/src/validation.cpp b/src/validation.cpp index 3bb19770a1..44bde6a2ab 100644 --- a/src/validation.cpp +++ b/src/validation.cpp @@ -2267,94 +2267,6 @@ bool Chainstate::ConnectBlock(const CBlock& block, BlockValidationState& state, Ticks(time_check), Ticks(time_check) / num_blocks_total); - /* - // Blackcoin - // Do not allow blocks that contain transactions which 'overwrite' older transactions, - // unless those are already completely spent. - // If such overwrites are allowed, coinbases and transactions depending upon those - // can be duplicated to remove the ability to spend the first instance -- even after - // being sent to another address. - // See BIP30, CVE-2012-1909, and http://r6.ca/blog/20120206T005236Z.html for more information. - // This rule was originally applied to all blocks with a timestamp after March 15, 2012, 0:00 UTC. - // Now that the whole chain is irreversibly beyond that time it is applied to all blocks except the - // two in the chain that violate it. This prevents exploiting the issue against nodes during their - // initial block download. - bool fEnforceBIP30 = !IsBIP30Repeat(*pindex); - - // Once BIP34 activated it was not possible to create new duplicate coinbases and thus other than starting - // with the 2 existing duplicate coinbase pairs, not possible to create overwriting txs. But by the - // time BIP34 activated, in each of the existing pairs the duplicate coinbase had overwritten the first - // before the first had been spent. Since those coinbases are sufficiently buried it's no longer possible to create further - // duplicate transactions descending from the known pairs either. - // If we're on the known chain at height greater than where BIP34 activated, we can save the db accesses needed for the BIP30 check. - - // BIP34 requires that a block at height X (block X) has its coinbase - // scriptSig start with a CScriptNum of X (indicated height X). The above - // logic of no longer requiring BIP30 once BIP34 activates is flawed in the - // case that there is a block X before the BIP34 height of 227,931 which has - // an indicated height Y where Y is greater than X. The coinbase for block - // X would also be a valid coinbase for block Y, which could be a BIP30 - // violation. An exhaustive search of all mainnet coinbases before the - // BIP34 height which have an indicated height greater than the block height - // reveals many occurrences. The 3 lowest indicated heights found are - // 209,921, 490,897, and 1,983,702 and thus coinbases for blocks at these 3 - // heights would be the first opportunity for BIP30 to be violated. - - // The search reveals a great many blocks which have an indicated height - // greater than 1,983,702, so we simply remove the optimization to skip - // BIP30 checking for blocks at height 1,983,702 or higher. Before we reach - // that block in another 25 years or so, we should take advantage of a - // future consensus change to do a new and improved version of BIP34 that - // will actually prevent ever creating any duplicate coinbases in the - // future. - static constexpr int BIP34_IMPLIES_BIP30_LIMIT = 1983702; - - // There is no potential to create a duplicate coinbase at block 209,921 - // because this is still before the BIP34 height and so explicit BIP30 - // checking is still active. - - // The final case is block 176,684 which has an indicated height of - // 490,897. Unfortunately, this issue was not discovered until about 2 weeks - // before block 490,897 so there was not much opportunity to address this - // case other than to carefully analyze it and determine it would not be a - // problem. Block 490,897 was, in fact, mined with a different coinbase than - // block 176,684, but it is important to note that even if it hadn't been or - // is remined on an alternate fork with a duplicate coinbase, we would still - // not run into a BIP30 violation. This is because the coinbase for 176,684 - // is spent in block 185,956 in transaction - // d4f7fbbf92f4a3014a230b2dc70b8058d02eb36ac06b4a0736d9d60eaa9e8781. This - // spending transaction can't be duplicated because it also spends coinbase - // 0328dd85c331237f18e781d692c92de57649529bd5edf1d01036daea32ffde29. This - // coinbase has an indicated height of over 4.2 billion, and wouldn't be - // duplicatable until that height, and it's currently impossible to create a - // chain that long. Nevertheless we may wish to consider a future soft fork - // which retroactively prevents block 490,897 from creating a duplicate - // coinbase. The two historical BIP30 violations often provide a confusing - // edge case when manipulating the UTXO and it would be simpler not to have - // another edge case to deal with. - - // testnet3 has no blocks before the BIP34 height with indicated heights - // post BIP34 before approximately height 486,000,000. After block - // 1,983,702 testnet3 starts doing unnecessary BIP30 checking again. - assert(pindex->pprev); - CBlockIndex* pindexBIP34height = pindex->pprev->GetAncestor(params.GetConsensus().BIP34Height); - //Only continue to enforce if we're below BIP34 activation height or the block hash at that height doesn't correspond. - fEnforceBIP30 = fEnforceBIP30 && (!pindexBIP34height || !(pindexBIP34height->GetBlockHash() == params.GetConsensus().BIP34Hash)); - - // TODO: Remove BIP30 checking from block height 1,983,702 on, once we have a - // consensus change that ensures coinbases at those heights cannot - // duplicate earlier coinbases. - if (fEnforceBIP30 || pindex->nHeight >= BIP34_IMPLIES_BIP30_LIMIT) { - for (const auto& tx : block.vtx) { - for (size_t o = 0; o < tx->vout.size(); o++) { - if (view.HaveCoin(COutPoint(tx->GetHash(), o))) { - LogPrintf("ERROR: ConnectBlock(): tried to overwrite transaction\n"); - return state.Invalid(BlockValidationResult::BLOCK_CONSENSUS, "bad-txns-BIP30"); - } - } - } - */ - // BIP30 and BIP34 are always active for (const auto& tx : block.vtx) { for (size_t o = 0; o < tx->vout.size(); o++) { @@ -2365,7 +2277,7 @@ bool Chainstate::ConnectBlock(const CBlock& block, BlockValidationState& state, } } - // Enforce BIP68 (sequence locks) + // Enforce BIP68 (sequence locks) and BIP112 (CHECKSEQUENCEVERIFY) int nLockTimeFlags = 0; // if (DeploymentActiveAt(*pindex, m_chainman, Consensus::DEPLOYMENT_CSV)) { if (params.GetConsensus().IsProtocolV3_1(pindex->GetBlockTime())) { @@ -5985,18 +5897,6 @@ Chainstate& ChainstateManager::ActivateExistingSnapshot(uint256 base_blockhash) return *m_snapshot_chainstate; } -bool IsBIP30Repeat(const CBlockIndex& block_index) -{ - return (block_index.nHeight==91842 && block_index.GetBlockHash() == uint256S("0x00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec")) || - (block_index.nHeight==91880 && block_index.GetBlockHash() == uint256S("0x00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721")); -} - -bool IsBIP30Unspendable(const CBlockIndex& block_index) -{ - return (block_index.nHeight==91722 && block_index.GetBlockHash() == uint256S("0x00000000000271a2dc26e7667f8419f2e15416dc6955e5a6c6cdf3f2574dd08e")) || - (block_index.nHeight==91812 && block_index.GetBlockHash() == uint256S("0x00000000000af0aed4792b1acee3d966af36cf5def14935db8de83d6f9306f2f")); -} - static fs::path GetSnapshotCoinsDBPath(Chainstate& cs) EXCLUSIVE_LOCKS_REQUIRED(::cs_main) { AssertLockHeld(::cs_main); diff --git a/src/validation.h b/src/validation.h index 824e4faa0f..93dbde8898 100644 --- a/src/validation.h +++ b/src/validation.h @@ -1276,10 +1276,4 @@ bool DeploymentEnabled(const ChainstateManager& chainman, DEP dep) return DeploymentEnabled(chainman.GetConsensus(), dep); } -/** Identifies blocks that overwrote an existing coinbase output in the UTXO set (see BIP30) */ -bool IsBIP30Repeat(const CBlockIndex& block_index); - -/** Identifies blocks which coinbase output was subsequently overwritten in the UTXO set (see BIP30) */ -bool IsBIP30Unspendable(const CBlockIndex& block_index); - #endif // BITCOIN_VALIDATION_H diff --git a/src/wallet/rpc/wallet.cpp b/src/wallet/rpc/wallet.cpp index 09976964ff..002ada781b 100644 --- a/src/wallet/rpc/wallet.cpp +++ b/src/wallet/rpc/wallet.cpp @@ -61,6 +61,7 @@ static RPCHelpMan getwalletinfo() {RPCResult::Type::STR_AMOUNT, "paytxfee", "the transaction fee configuration, set in " + CURRENCY_UNIT + "/kvB"}, {RPCResult::Type::STR_HEX, "hdseedid", /*optional=*/true, "the Hash160 of the HD seed (only present when HD is enabled)"}, {RPCResult::Type::BOOL, "private_keys_enabled", "false if privatekeys are disabled for this wallet (enforced watch-only wallet)"}, + {RPCResult::Type::BOOL, "unlocked_staking_only", /*optional=*/true, "true if wallet is unlocked for staking only"}, {RPCResult::Type::BOOL, "avoid_reuse", "whether this wallet tracks clean/dirty coins in terms of reuse"}, {RPCResult::Type::OBJ, "scanning", "current scanning details, or false if no scan is in progress", { @@ -120,6 +121,8 @@ static RPCHelpMan getwalletinfo() } if (pwallet->IsCrypted()) { obj.pushKV("unlocked_until", pwallet->nRelockTime); + bool stakingOnly = pwallet->m_wallet_unlock_staking_only; + obj.pushKV("unlocked_staking_only", stakingOnly); } obj.pushKV("paytxfee", ValueFromAmount(pwallet->m_pay_tx_fee.GetFeePerK())); obj.pushKV("private_keys_enabled", !pwallet->IsWalletFlagSet(WALLET_FLAG_DISABLE_PRIVATE_KEYS));