Bitcoin Core  26.1.0
P2P Digital Currency
sigopcount_tests.cpp
Go to the documentation of this file.
1 // Copyright (c) 2012-2022 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 <coins.h>
7 #include <consensus/consensus.h>
8 #include <consensus/tx_verify.h>
9 #include <key.h>
10 #include <pubkey.h>
11 #include <script/interpreter.h>
12 #include <script/script.h>
13 #include <script/solver.h>
14 #include <test/util/setup_common.h>
15 #include <uint256.h>
16 
17 #include <vector>
18 
19 #include <boost/test/unit_test.hpp>
20 
21 // Helpers:
22 static std::vector<unsigned char>
23 Serialize(const CScript& s)
24 {
25  std::vector<unsigned char> sSerialized(s.begin(), s.end());
26  return sSerialized;
27 }
28 
29 BOOST_FIXTURE_TEST_SUITE(sigopcount_tests, BasicTestingSetup)
30 
31 BOOST_AUTO_TEST_CASE(GetSigOpCount)
32 {
33  // Test CScript::GetSigOpCount()
34  CScript s1;
35  BOOST_CHECK_EQUAL(s1.GetSigOpCount(false), 0U);
36  BOOST_CHECK_EQUAL(s1.GetSigOpCount(true), 0U);
37 
38  uint160 dummy;
39  s1 << OP_1 << ToByteVector(dummy) << ToByteVector(dummy) << OP_2 << OP_CHECKMULTISIG;
40  BOOST_CHECK_EQUAL(s1.GetSigOpCount(true), 2U);
41  s1 << OP_IF << OP_CHECKSIG << OP_ENDIF;
42  BOOST_CHECK_EQUAL(s1.GetSigOpCount(true), 3U);
43  BOOST_CHECK_EQUAL(s1.GetSigOpCount(false), 21U);
44 
46  CScript scriptSig;
47  scriptSig << OP_0 << Serialize(s1);
48  BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(scriptSig), 3U);
49 
50  std::vector<CPubKey> keys;
51  for (int i = 0; i < 3; i++)
52  {
53  CKey k;
54  k.MakeNewKey(true);
55  keys.push_back(k.GetPubKey());
56  }
57  CScript s2 = GetScriptForMultisig(1, keys);
58  BOOST_CHECK_EQUAL(s2.GetSigOpCount(true), 3U);
59  BOOST_CHECK_EQUAL(s2.GetSigOpCount(false), 20U);
60 
62  BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(true), 0U);
63  BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(false), 0U);
64  CScript scriptSig2;
65  scriptSig2 << OP_1 << ToByteVector(dummy) << ToByteVector(dummy) << Serialize(s2);
66  BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(scriptSig2), 3U);
67 }
68 
73 static ScriptError VerifyWithFlag(const CTransaction& output, const CMutableTransaction& input, uint32_t flags)
74 {
76  CTransaction inputi(input);
77  bool ret = VerifyScript(inputi.vin[0].scriptSig, output.vout[0].scriptPubKey, &inputi.vin[0].scriptWitness, flags, TransactionSignatureChecker(&inputi, 0, output.vout[0].nValue, MissingDataBehavior::ASSERT_FAIL), &error);
78  BOOST_CHECK((ret == true) == (error == SCRIPT_ERR_OK));
79 
80  return error;
81 }
82 
88 static void BuildTxs(CMutableTransaction& spendingTx, CCoinsViewCache& coins, CMutableTransaction& creationTx, const CScript& scriptPubKey, const CScript& scriptSig, const CScriptWitness& witness)
89 {
90  creationTx.nVersion = 1;
91  creationTx.vin.resize(1);
92  creationTx.vin[0].prevout.SetNull();
93  creationTx.vin[0].scriptSig = CScript();
94  creationTx.vout.resize(1);
95  creationTx.vout[0].nValue = 1;
96  creationTx.vout[0].scriptPubKey = scriptPubKey;
97 
98  spendingTx.nVersion = 1;
99  spendingTx.vin.resize(1);
100  spendingTx.vin[0].prevout.hash = creationTx.GetHash();
101  spendingTx.vin[0].prevout.n = 0;
102  spendingTx.vin[0].scriptSig = scriptSig;
103  spendingTx.vin[0].scriptWitness = witness;
104  spendingTx.vout.resize(1);
105  spendingTx.vout[0].nValue = 1;
106  spendingTx.vout[0].scriptPubKey = CScript();
107 
108  AddCoins(coins, CTransaction(creationTx), 0);
109 }
110 
111 BOOST_AUTO_TEST_CASE(GetTxSigOpCost)
112 {
113  // Transaction creates outputs
114  CMutableTransaction creationTx;
115  // Transaction that spends outputs and whose
116  // sig op cost is going to be tested
117  CMutableTransaction spendingTx;
118 
119  // Create utxo set
120  CCoinsView coinsDummy;
121  CCoinsViewCache coins(&coinsDummy);
122  // Create key
123  CKey key;
124  key.MakeNewKey(true);
125  CPubKey pubkey = key.GetPubKey();
126  // Default flags
128 
129  // Multisig script (legacy counting)
130  {
131  CScript scriptPubKey = CScript() << 1 << ToByteVector(pubkey) << ToByteVector(pubkey) << 2 << OP_CHECKMULTISIGVERIFY;
132  // Do not use a valid signature to avoid using wallet operations.
133  CScript scriptSig = CScript() << OP_0 << OP_0;
134 
135  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, CScriptWitness());
136  // Legacy counting only includes signature operations in scriptSigs and scriptPubKeys
137  // of a transaction and does not take the actual executed sig operations into account.
138  // spendingTx in itself does not contain a signature operation.
139  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 0);
140  // creationTx contains two signature operations in its scriptPubKey, but legacy counting
141  // is not accurate.
143  // Sanity check: script verification fails because of an invalid signature.
145  }
146 
147  // Multisig nested in P2SH
148  {
149  CScript redeemScript = CScript() << 1 << ToByteVector(pubkey) << ToByteVector(pubkey) << 2 << OP_CHECKMULTISIGVERIFY;
150  CScript scriptPubKey = GetScriptForDestination(ScriptHash(redeemScript));
151  CScript scriptSig = CScript() << OP_0 << OP_0 << ToByteVector(redeemScript);
152 
153  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, CScriptWitness());
156  }
157 
158  // P2WPKH witness program
159  {
160  CScript scriptPubKey = GetScriptForDestination(WitnessV0KeyHash(pubkey));
161  CScript scriptSig = CScript();
162  CScriptWitness scriptWitness;
163  scriptWitness.stack.emplace_back(0);
164  scriptWitness.stack.emplace_back(0);
165 
166 
167  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, scriptWitness);
168  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 1);
169  // No signature operations if we don't verify the witness.
171  assert(VerifyWithFlag(CTransaction(creationTx), spendingTx, flags) == SCRIPT_ERR_EQUALVERIFY);
172 
173  // The sig op cost for witness version != 0 is zero.
174  assert(scriptPubKey[0] == 0x00);
175  scriptPubKey[0] = 0x51;
176  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, scriptWitness);
177  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 0);
178  scriptPubKey[0] = 0x00;
179  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, scriptWitness);
180 
181  // The witness of a coinbase transaction is not taken into account.
182  spendingTx.vin[0].prevout.SetNull();
183  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 0);
184  }
185 
186  // P2WPKH nested in P2SH
187  {
188  CScript scriptSig = GetScriptForDestination(WitnessV0KeyHash(pubkey));
189  CScript scriptPubKey = GetScriptForDestination(ScriptHash(scriptSig));
190  scriptSig = CScript() << ToByteVector(scriptSig);
191  CScriptWitness scriptWitness;
192  scriptWitness.stack.emplace_back(0);
193  scriptWitness.stack.emplace_back(0);
194 
195  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, scriptWitness);
196  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 1);
197  assert(VerifyWithFlag(CTransaction(creationTx), spendingTx, flags) == SCRIPT_ERR_EQUALVERIFY);
198  }
199 
200  // P2WSH witness program
201  {
202  CScript witnessScript = CScript() << 1 << ToByteVector(pubkey) << ToByteVector(pubkey) << 2 << OP_CHECKMULTISIGVERIFY;
203  CScript scriptPubKey = GetScriptForDestination(WitnessV0ScriptHash(witnessScript));
204  CScript scriptSig = CScript();
205  CScriptWitness scriptWitness;
206  scriptWitness.stack.emplace_back(0);
207  scriptWitness.stack.emplace_back(0);
208  scriptWitness.stack.emplace_back(witnessScript.begin(), witnessScript.end());
209 
210  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, scriptWitness);
211  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 2);
214  }
215 
216  // P2WSH nested in P2SH
217  {
218  CScript witnessScript = CScript() << 1 << ToByteVector(pubkey) << ToByteVector(pubkey) << 2 << OP_CHECKMULTISIGVERIFY;
219  CScript redeemScript = GetScriptForDestination(WitnessV0ScriptHash(witnessScript));
220  CScript scriptPubKey = GetScriptForDestination(ScriptHash(redeemScript));
221  CScript scriptSig = CScript() << ToByteVector(redeemScript);
222  CScriptWitness scriptWitness;
223  scriptWitness.stack.emplace_back(0);
224  scriptWitness.stack.emplace_back(0);
225  scriptWitness.stack.emplace_back(witnessScript.begin(), witnessScript.end());
226 
227  BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, scriptWitness);
228  assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 2);
230  }
231 }
232 
unsigned int GetSigOpCount(bool fAccurate) const
Pre-version-0.6, Bitcoin always counted CHECKMULTISIGs as 20 sigops.
Definition: script.cpp:156
CScript GetScriptForMultisig(int nRequired, const std::vector< CPubKey > &keys)
Generate a multisig script.
Definition: solver.cpp:214
int ret
BOOST_AUTO_TEST_CASE(GetSigOpCount)
static const int WITNESS_SCALE_FACTOR
Definition: consensus.h:21
assert(!tx.IsCoinBase())
enum ScriptError_t ScriptError
CPubKey GetPubKey() const
Compute the public key from a private key.
Definition: key.cpp:188
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, const CScriptWitness *witness, unsigned int flags, const BaseSignatureChecker &checker, ScriptError *serror)
std::vector< CTxIn > vin
Definition: transaction.h:381
static std::vector< unsigned char > Serialize(const CScript &s)
std::vector< std::vector< unsigned char > > stack
Definition: script.h:568
Definition: script.h:74
static const int MAX_PUBKEYS_PER_MULTISIG
Definition: script.h:32
const std::vector< CTxIn > vin
Definition: transaction.h:305
Basic testing setup.
Definition: setup_common.h:49
Definition: script.h:81
Definition: script.h:102
void AddCoins(CCoinsViewCache &cache, const CTransaction &tx, int nHeight, bool check_for_overwrite)
Utility function to add all of a transaction&#39;s outputs to a cache.
Definition: coins.cpp:118
iterator end()
Definition: prevector.h:301
Abort execution through assertion failure (for consensus code)
int64_t GetTransactionSigOpCost(const CTransaction &tx, const CCoinsViewCache &inputs, uint32_t flags)
Compute total signature operation cost of a transaction.
Definition: tx_verify.cpp:147
Abstract view on the open txout dataset.
Definition: coins.h:172
static void BuildTxs(CMutableTransaction &spendingTx, CCoinsViewCache &coins, CMutableTransaction &creationTx, const CScript &scriptPubKey, const CScript &scriptSig, const CScriptWitness &witness)
Builds a creationTx from scriptPubKey and a spendingTx from scriptSig and witness such that spendingT...
BOOST_AUTO_TEST_SUITE_END()
An encapsulated public key.
Definition: pubkey.h:33
void MakeNewKey(bool fCompressed)
Generate a new private key using a cryptographic PRNG.
Definition: key.cpp:161
const std::vector< CTxOut > vout
Definition: transaction.h:306
std::vector< CTxOut > vout
Definition: transaction.h:382
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
int flags
Definition: bitcoin-tx.cpp:528
std::vector< unsigned char > ToByteVector(const T &in)
Definition: script.h:65
uint256 GetHash() const
Compute the hash of this CMutableTransaction.
Definition: transaction.cpp:68
#define BOOST_CHECK_EQUAL(v1, v2)
Definition: object.cpp:18
bool error(const char *fmt, const Args &... args)
Definition: logging.h:262
Serialized script, used inside transaction inputs and outputs.
Definition: script.h:412
static ScriptError VerifyWithFlag(const CTransaction &output, const CMutableTransaction &input, uint32_t flags)
Verifies script execution of the zeroth scriptPubKey of tx output and zeroth scriptSig and witness of...
160-bit opaque blob.
Definition: uint256.h:95
iterator begin()
Definition: prevector.h:299
A mutable version of CTransaction.
Definition: transaction.h:379
Definition: script.h:83
An encapsulated private key.
Definition: key.h:32
The basic transaction that is broadcasted on the network and contained in blocks. ...
Definition: transaction.h:294
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Definition: coins.h:228
#define BOOST_CHECK(expr)
Definition: object.cpp:17