Bitcoin Core 31.0.0
P2P Digital Currency
Loading...
Searching...
No Matches
connectblock.cpp
Go to the documentation of this file.
1// Copyright (c) 2025-present The Bitcoin Core developers
2// Distributed under the MIT software license, see the accompanying
3// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5#include <addresstype.h>
6#include <bench/bench.h>
7#include <interfaces/chain.h>
8#include <kernel/cs_main.h>
10#include <sync.h>
12#include <validation.h>
13
14#include <cassert>
15#include <vector>
16
17/*
18 * Creates a test block containing transactions with the following properties:
19 * - Each transaction has the same number of inputs and outputs
20 * - All Taproot inputs use simple key path spends (no script path spends)
21 * - All signatures use SIGHASH_ALL (default sighash)
22 * - Each transaction spends all outputs from the previous transaction
23 */
26 const std::vector<CKey>& keys,
27 const std::vector<CTxOut>& outputs,
28 int num_txs = 1000)
29{
30 Chainstate& chainstate{test_setup.m_node.chainman->ActiveChainstate()};
31
32 const WitnessV1Taproot coinbase_taproot{XOnlyPubKey(test_setup.coinbaseKey.GetPubKey())};
33
34 // Create the outputs that will be spent in the first transaction of the test block
35 // Doing this in a separate block excludes the validation of its inputs from the benchmark
36 auto& coinbase_to_spend{test_setup.m_coinbase_txns[0]};
37 const auto [first_tx, _]{test_setup.CreateValidTransaction(
39 {COutPoint(coinbase_to_spend->GetHash(), 0)},
40 chainstate.m_chain.Height() + 1, keys, outputs, {}, {})};
42 test_setup.CreateAndProcessBlock({first_tx}, coinbase_spk, &chainstate);
43
44 std::vector<CMutableTransaction> txs;
47 for (int i{0}; i < num_txs; i++) {
48 std::vector<COutPoint> inputs;
49 inputs.reserve(outputs.size());
50
51 for (size_t j{0}; j < outputs.size(); j++) {
52 inputs.emplace_back(tx_to_spend->GetHash(), j);
53 }
54
55 const auto [tx, _]{test_setup.CreateValidTransaction(
56 {tx_to_spend}, inputs, chainstate.m_chain.Height() + 1, keys, outputs, {}, {})};
57 txs.emplace_back(tx);
59 }
60
61 // Coinbase output can use any output type as it is not spent and will not change the benchmark
62 return test_setup.CreateBlock(txs, coinbase_spk, chainstate);
63}
64
65/*
66 * Creates key pairs and corresponding outputs for the benchmark transactions.
67 * - For Schnorr signatures: Creates simple key path spendable outputs
68 * - For Ecdsa signatures: Creates P2WPKH (native SegWit v0) outputs
69 * - All outputs have value of 1 BTC
70 */
71std::pair<std::vector<CKey>, std::vector<CTxOut>> CreateKeysAndOutputs(const CKey& coinbaseKey, size_t num_schnorr, size_t num_ecdsa)
72{
73 std::vector<CKey> keys{coinbaseKey};
74 keys.reserve(num_schnorr + num_ecdsa + 1);
75
76 std::vector<CTxOut> outputs;
77 outputs.reserve(num_schnorr + num_ecdsa);
78
79 for (size_t i{0}; i < num_ecdsa; ++i) {
80 keys.emplace_back(GenerateRandomKey());
81 outputs.emplace_back(COIN, GetScriptForDestination(WitnessV0KeyHash{keys.back().GetPubKey()}));
82 }
83
84 for (size_t i{0}; i < num_schnorr; ++i) {
85 keys.emplace_back(GenerateRandomKey());
86 outputs.emplace_back(COIN, GetScriptForDestination(WitnessV1Taproot{XOnlyPubKey(keys.back().GetPubKey())}));
87 }
88
89 return {keys, outputs};
90}
91
92void BenchmarkConnectBlock(benchmark::Bench& bench, std::vector<CKey>& keys, std::vector<CTxOut>& outputs, TestChain100Setup& test_setup)
93{
94 const auto& test_block{CreateTestBlock(test_setup, keys, outputs)};
95 bench.unit("block").run([&] {
97 auto& chainman{test_setup.m_node.chainman};
98 auto& chainstate{chainman->ActiveChainstate()};
100 auto* pindex{chainman->m_blockman.AddToBlockIndex(test_block, chainman->m_best_header)}; // Doing this here doesn't impact the benchmark
102
103 assert(chainstate.ConnectBlock(test_block, test_block_state, pindex, viewNew));
104 });
105}
106
108{
110 auto [keys, outputs]{CreateKeysAndOutputs(test_setup->coinbaseKey, /*num_schnorr=*/5, /*num_ecdsa=*/0)};
111 BenchmarkConnectBlock(bench, keys, outputs, *test_setup);
112}
113
115{
117 // Blocks in range 848000 to 868000 have a roughly 20 to 80 ratio of schnorr to ecdsa inputs
118 auto [keys, outputs]{CreateKeysAndOutputs(test_setup->coinbaseKey, /*num_schnorr=*/1, /*num_ecdsa=*/4)};
119 BenchmarkConnectBlock(bench, keys, outputs, *test_setup);
120}
121
123{
125 auto [keys, outputs]{CreateKeysAndOutputs(test_setup->coinbaseKey, /*num_schnorr=*/0, /*num_ecdsa=*/5)};
126 BenchmarkConnectBlock(bench, keys, outputs, *test_setup);
127}
128
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
static constexpr CAmount COIN
The amount of satoshis in one BTC.
Definition amount.h:15
#define BENCHMARK(n)
Definition bench.h:68
Definition block.h:74
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Definition coins.h:368
An encapsulated private key.
Definition key.h:36
An outpoint - a combination of a transaction hash and an index n into its vout.
Definition transaction.h:29
Serialized script, used inside transaction inputs and outputs.
Definition script.h:405
Chainstate stores and provides an API to update our local knowledge of the current best chain.
Definition validation.h:551
Main entry point to nanobench's benchmarking facility.
Definition nanobench.h:627
void reserve(size_type new_capacity)
Definition prevector.h:293
void BenchmarkConnectBlock(benchmark::Bench &bench, std::vector< CKey > &keys, std::vector< CTxOut > &outputs, TestChain100Setup &test_setup)
static void ConnectBlockMixedEcdsaSchnorr(benchmark::Bench &bench)
static void ConnectBlockAllEcdsa(benchmark::Bench &bench)
static void ConnectBlockAllSchnorr(benchmark::Bench &bench)
std::pair< std::vector< CKey >, std::vector< CTxOut > > CreateKeysAndOutputs(const CKey &coinbaseKey, size_t num_schnorr, size_t num_ecdsa)
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
Definition cs_main.cpp:8
CKey GenerateRandomKey(bool compressed) noexcept
Definition key.cpp:475
static CTransactionRef MakeTransactionRef(Tx &&txIn)
std::shared_ptr< const CTransaction > CTransactionRef
static CBlock CreateTestBlock()
static bool GetPubKey(const SigningProvider &provider, const SignatureData &sigdata, const CKeyID &address, CPubKey &pubkey)
Definition sign.cpp:221
Testing fixture that pre-creates a 100-block REGTEST-mode block chain.
#define LOCK(cs)
Definition sync.h:258
consteval auto _(util::TranslatedLiteral str)
Definition translation.h:79
constexpr auto Ticks(Dur2 d)
Helper to count the seconds of a duration/time_point.
Definition time.h:73
assert(!tx.IsCoinBase())