Bitcoin Core 31.99.0
P2P Digital Currency
sigopcount_tests.cpp
Go to the documentation of this file.
1// Copyright (c) 2012-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 <coins.h>
9#include <key.h>
10#include <pubkey.h>
11#include <script/interpreter.h>
12#include <script/script.h>
13#include <script/solver.h>
15#include <uint256.h>
16
17#include <vector>
18
19#include <boost/test/unit_test.hpp>
20
21// Helpers:
22static std::vector<unsigned char>
24{
25 std::vector<unsigned char> sSerialized(s.begin(), s.end());
26 return sSerialized;
27}
28
30
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 {
54 keys.push_back(k.GetPubKey());
55 }
56 CScript s2 = GetScriptForMultisig(1, keys);
57 BOOST_CHECK_EQUAL(s2.GetSigOpCount(true), 3U);
58 BOOST_CHECK_EQUAL(s2.GetSigOpCount(false), 20U);
59
61 BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(true), 0U);
62 BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(false), 0U);
63 CScript scriptSig2;
64 scriptSig2 << OP_1 << ToByteVector(dummy) << ToByteVector(dummy) << Serialize(s2);
65 BOOST_CHECK_EQUAL(p2sh.GetSigOpCount(scriptSig2), 3U);
66}
67
73{
74 ScriptError error;
75 CTransaction inputi(input);
76 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);
77 BOOST_CHECK((ret == true) == (error == SCRIPT_ERR_OK));
78
79 return error;
80}
81
87static void BuildTxs(CMutableTransaction& spendingTx, CCoinsViewCache& coins, CMutableTransaction& creationTx, const CScript& scriptPubKey, const CScript& scriptSig, const CScriptWitness& witness)
88{
89 creationTx.version = 1;
90 creationTx.vin.resize(1);
91 creationTx.vin[0].prevout.SetNull();
92 creationTx.vin[0].scriptSig = CScript();
93 creationTx.vout.resize(1);
94 creationTx.vout[0].nValue = 1;
95 creationTx.vout[0].scriptPubKey = scriptPubKey;
96
97 spendingTx.version = 1;
98 spendingTx.vin.resize(1);
99 spendingTx.vin[0].prevout.hash = creationTx.GetHash();
100 spendingTx.vin[0].prevout.n = 0;
101 spendingTx.vin[0].scriptSig = scriptSig;
102 spendingTx.vin[0].scriptWitness = witness;
103 spendingTx.vout.resize(1);
104 spendingTx.vout[0].nValue = 1;
105 spendingTx.vout[0].scriptPubKey = CScript();
106
107 AddCoins(coins, CTransaction(creationTx), 0);
108}
109
110BOOST_AUTO_TEST_CASE(GetTxSigOpCost)
111{
112 // Transaction creates outputs
113 CMutableTransaction creationTx;
114 // Transaction that spends outputs and whose
115 // sig op cost is going to be tested
116 CMutableTransaction spendingTx;
117
118 // Create utxo set
120 // Create key
121 CKey key = GenerateRandomKey();
122 CPubKey pubkey = key.GetPubKey();
123 // Default flags
125
126 // Multisig script (legacy counting)
127 {
128 CScript scriptPubKey = CScript() << 1 << ToByteVector(pubkey) << ToByteVector(pubkey) << 2 << OP_CHECKMULTISIGVERIFY;
129 // Do not use a valid signature to avoid using wallet operations.
130 CScript scriptSig = CScript() << OP_0 << OP_0;
131
132 BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, CScriptWitness());
133 // Legacy counting only includes signature operations in scriptSigs and scriptPubKeys
134 // of a transaction and does not take the actual executed sig operations into account.
135 // spendingTx in itself does not contain a signature operation.
136 assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, flags) == 0);
137 // creationTx contains two signature operations in its scriptPubKey, but legacy counting
138 // is not accurate.
140 // Sanity check: script verification fails because of an invalid signature.
142 }
143
144 // Multisig nested in P2SH
145 {
146 CScript redeemScript = CScript() << 1 << ToByteVector(pubkey) << ToByteVector(pubkey) << 2 << OP_CHECKMULTISIGVERIFY;
147 CScript scriptPubKey = GetScriptForDestination(ScriptHash(redeemScript));
148 CScript scriptSig = CScript() << OP_0 << OP_0 << ToByteVector(redeemScript);
149
150 BuildTxs(spendingTx, coins, creationTx, scriptPubKey, scriptSig, CScriptWitness());
153
154 // P2SH sigops are not counted if we don't set the SCRIPT_VERIFY_P2SH flag
155 assert(GetTransactionSigOpCost(CTransaction(spendingTx), coins, /*flags=*/0) == 0);
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 {
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
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
int ret
int flags
Definition: bitcoin-tx.cpp:530
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Definition: coins.h:394
An encapsulated private key.
Definition: key.h:36
CPubKey GetPubKey() const
Compute the public key from a private key.
Definition: key.cpp:183
An encapsulated public key.
Definition: pubkey.h:34
Serialized script, used inside transaction inputs and outputs.
Definition: script.h:405
unsigned int GetSigOpCount(bool fAccurate) const
Pre-version-0.6, Bitcoin always counted CHECKMULTISIGs as 20 sigops.
Definition: script.cpp:158
The basic transaction that is broadcasted on the network and contained in blocks.
Definition: transaction.h:281
const std::vector< CTxOut > vout
Definition: transaction.h:292
const std::vector< CTxIn > vin
Definition: transaction.h:291
static CoinsViewEmpty & Get()
Definition: coins.cpp:17
iterator begin()
Definition: prevector.h:255
iterator end()
Definition: prevector.h:257
160-bit opaque blob.
Definition: uint256.h:184
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.
Definition: coins.cpp:121
static const int WITNESS_SCALE_FACTOR
Definition: consensus.h:21
BOOST_FIXTURE_TEST_SUITE(cuckoocache_tests, BasicTestingSetup)
Test Suite for CuckooCache.
BOOST_AUTO_TEST_SUITE_END()
bool VerifyScript(const CScript &scriptSig, const CScript &scriptPubKey, const CScriptWitness *witness, script_verify_flags flags, const BaseSignatureChecker &checker, ScriptError *serror)
@ ASSERT_FAIL
Abort execution through assertion failure (for consensus code)
CKey GenerateRandomKey(bool compressed) noexcept
Definition: key.cpp:475
#define BOOST_CHECK_EQUAL(v1, v2)
Definition: object.cpp:17
#define BOOST_CHECK(expr)
Definition: object.cpp:16
@ OP_2
Definition: script.h:85
@ OP_CHECKMULTISIG
Definition: script.h:192
@ OP_IF
Definition: script.h:104
@ OP_CHECKSIG
Definition: script.h:190
@ OP_ENDIF
Definition: script.h:109
@ OP_1
Definition: script.h:83
@ OP_CHECKMULTISIGVERIFY
Definition: script.h:193
@ OP_0
Definition: script.h:76
static const int MAX_PUBKEYS_PER_MULTISIG
Definition: script.h:34
std::vector< unsigned char > ToByteVector(const T &in)
Definition: script.h:67
enum ScriptError_t ScriptError
@ SCRIPT_ERR_EQUALVERIFY
Definition: script_error.h:29
@ SCRIPT_ERR_OK
Definition: script_error.h:13
@ SCRIPT_ERR_CHECKMULTISIGVERIFY
Definition: script_error.h:30
static ScriptError VerifyWithFlag(const CTransaction &output, const CMutableTransaction &input, script_verify_flags flags)
Verifies script execution of the zeroth scriptPubKey of tx output and zeroth scriptSig and witness of...
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...
static std::vector< unsigned char > Serialize(const CScript &s)
BOOST_AUTO_TEST_CASE(GetSigOpCount)
CScript GetScriptForMultisig(int nRequired, const std::vector< CPubKey > &keys)
Generate a multisig script.
Definition: solver.cpp:218
Basic testing setup.
Definition: setup_common.h:61
A mutable version of CTransaction.
Definition: transaction.h:358
std::vector< CTxOut > vout
Definition: transaction.h:360
Txid GetHash() const
Compute the hash of this CMutableTransaction.
Definition: transaction.cpp:69
std::vector< CTxIn > vin
Definition: transaction.h:359
std::vector< std::vector< unsigned char > > stack
Definition: script.h:580
int64_t GetTransactionSigOpCost(const CTransaction &tx, const CCoinsViewCache &inputs, script_verify_flags flags)
Compute total signature operation cost of a transaction.
Definition: tx_verify.cpp:143
assert(!tx.IsCoinBase())