Branch data Line data Source code
1 : : // Copyright (c) 2009-2010 Satoshi Nakamoto
2 : : // Copyright (c) 2009-2022 The Bitcoin Core developers
3 : : // Distributed under the MIT software license, see the accompanying
4 : : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 : :
6 : : #include <pubkey.h>
7 : : #include <script/interpreter.h>
8 : : #include <script/script.h>
9 : : #include <script/solver.h>
10 : : #include <span.h>
11 : :
12 : : #include <algorithm>
13 : : #include <cassert>
14 : : #include <string>
15 : :
16 : : typedef std::vector<unsigned char> valtype;
17 : :
18 : 1401081 : std::string GetTxnOutputType(TxoutType t)
19 : : {
20 [ + + + + : 1401081 : switch (t) {
+ + + + +
+ + - ]
21 : 960724 : case TxoutType::NONSTANDARD: return "nonstandard";
22 : 32275 : case TxoutType::PUBKEY: return "pubkey";
23 : 28060 : case TxoutType::PUBKEYHASH: return "pubkeyhash";
24 : 151938 : case TxoutType::SCRIPTHASH: return "scripthash";
25 : 28698 : case TxoutType::MULTISIG: return "multisig";
26 : 5139 : case TxoutType::NULL_DATA: return "nulldata";
27 : 2080 : case TxoutType::ANCHOR: return "anchor";
28 : 67418 : case TxoutType::WITNESS_V0_KEYHASH: return "witness_v0_keyhash";
29 : 80782 : case TxoutType::WITNESS_V0_SCRIPTHASH: return "witness_v0_scripthash";
30 : 16056 : case TxoutType::WITNESS_V1_TAPROOT: return "witness_v1_taproot";
31 : 27911 : case TxoutType::WITNESS_UNKNOWN: return "witness_unknown";
32 : : } // no default case, so the compiler can warn about missing cases
33 : 0 : assert(false);
34 : : }
35 : :
36 : 4787207 : static bool MatchPayToPubkey(const CScript& script, valtype& pubkey)
37 : : {
38 [ + + + + : 4787207 : if (script.size() == CPubKey::SIZE + 2 && script[0] == CPubKey::SIZE && script.back() == OP_CHECKSIG) {
+ + + + ]
39 : 49070 : pubkey = valtype(script.begin() + 1, script.begin() + CPubKey::SIZE + 1);
40 : 24535 : return CPubKey::ValidSize(pubkey);
41 : : }
42 [ + + + + : 4762672 : if (script.size() == CPubKey::COMPRESSED_SIZE + 2 && script[0] == CPubKey::COMPRESSED_SIZE && script.back() == OP_CHECKSIG) {
+ + + + ]
43 : 221450 : pubkey = valtype(script.begin() + 1, script.begin() + CPubKey::COMPRESSED_SIZE + 1);
44 : 110725 : return CPubKey::ValidSize(pubkey);
45 : : }
46 : : return false;
47 : : }
48 : :
49 : 4665883 : static bool MatchPayToPubkeyHash(const CScript& script, valtype& pubkeyhash)
50 : : {
51 [ + + + + : 8178268 : if (script.size() == 25 && script[0] == OP_DUP && script[1] == OP_HASH160 && script[2] == 20 && script[23] == OP_EQUALVERIFY && script[24] == OP_CHECKSIG) {
+ + + + +
+ + + + +
+ + + + +
+ + + +
+ ]
52 : 954398 : pubkeyhash = valtype(script.begin () + 3, script.begin() + 23);
53 : 477199 : return true;
54 : : }
55 : : return false;
56 : : }
57 : :
58 : : /** Test for "small positive integer" script opcodes - OP_1 through OP_16. */
59 : 326620 : static constexpr bool IsSmallInteger(opcodetype opcode)
60 : : {
61 : 326620 : return opcode >= OP_1 && opcode <= OP_16;
62 : : }
63 : :
64 : : /** Retrieve a minimally-encoded number in range [min,max] from an (opcode, data) pair,
65 : : * whether it's OP_n or through a push. */
66 : 326620 : static std::optional<int> GetScriptNumber(opcodetype opcode, valtype data, int min, int max)
67 : : {
68 : 326620 : int count;
69 [ + + ]: 326620 : if (IsSmallInteger(opcode)) {
70 : 217735 : count = CScript::DecodeOP_N(opcode);
71 [ + + ]: 108885 : } else if (IsPushdataOp(opcode)) {
72 [ + + ]: 76636 : if (!CheckMinimalPush(data, opcode)) return {};
73 : 69987 : try {
74 [ + + ]: 69987 : count = CScriptNum(data, /* fRequireMinimal = */ true).getint();
75 [ - + ]: 23283 : } catch (const scriptnum_error&) {
76 : 23283 : return {};
77 : 23283 : }
78 : : } else {
79 : 32249 : return {};
80 : : }
81 [ + + + + ]: 264439 : if (count < min || count > max) return {};
82 : 239976 : return count;
83 : : }
84 : :
85 : 4188684 : static bool MatchMultisig(const CScript& script, int& required_sigs, std::vector<valtype>& pubkeys)
86 : : {
87 : 4188684 : opcodetype opcode;
88 : 4188684 : valtype data;
89 : :
90 [ + + ]: 4188684 : CScript::const_iterator it = script.begin();
91 [ + + + + : 8637033 : if (script.size() < 1 || script.back() != OP_CHECKMULTISIG) return false;
+ + + + ]
92 : :
93 [ + - + + ]: 196890 : if (!script.GetOp(it, opcode, data)) return false;
94 [ + - + - ]: 188544 : auto req_sigs = GetScriptNumber(opcode, data, 1, MAX_PUBKEYS_PER_MULTISIG);
95 [ + + ]: 188544 : if (!req_sigs) return false;
96 : 135936 : required_sigs = *req_sigs;
97 [ + - + + : 655318 : while (script.GetOp(it, opcode, data) && CPubKey::ValidSize(data)) {
+ + ]
98 [ + - ]: 519382 : pubkeys.emplace_back(std::move(data));
99 : : }
100 [ + - + - ]: 135936 : auto num_keys = GetScriptNumber(opcode, data, required_sigs, MAX_PUBKEYS_PER_MULTISIG);
101 [ + + ]: 135936 : if (!num_keys) return false;
102 [ + + ]: 101900 : if (pubkeys.size() != static_cast<unsigned long>(*num_keys)) return false;
103 : :
104 : 98572 : return (it + 1 == script.end());
105 : 4188684 : }
106 : :
107 : 9007 : std::optional<std::pair<int, std::vector<Span<const unsigned char>>>> MatchMultiA(const CScript& script)
108 : : {
109 : 9007 : std::vector<Span<const unsigned char>> keyspans;
110 : :
111 : : // Redundant, but very fast and selective test.
112 [ + + + - : 30685 : if (script.size() == 0 || script[0] != 32 || script.back() != OP_NUMEQUAL) return {};
+ + + + -
+ + + ]
113 : :
114 : : // Parse keys
115 : 2140 : auto it = script.begin();
116 [ + + ]: 323344 : while (script.end() - it >= 34) {
117 [ - + ]: 321204 : if (*it != 32) return {};
118 [ + - ]: 321204 : ++it;
119 [ + - ]: 321204 : keyspans.emplace_back(&*it, 32);
120 [ + + ]: 321204 : it += 32;
121 [ - + ]: 321204 : if (*it != (keyspans.size() == 1 ? OP_CHECKSIG : OP_CHECKSIGADD)) return {};
122 : 321204 : ++it;
123 : : }
124 [ + - + - ]: 2140 : if (keyspans.size() == 0 || keyspans.size() > MAX_PUBKEYS_PER_MULTI_A) return {};
125 : :
126 : : // Parse threshold.
127 : 2140 : opcodetype opcode;
128 : 2140 : std::vector<unsigned char> data;
129 [ + - - + ]: 2140 : if (!script.GetOp(it, opcode, data)) return {};
130 [ - + ]: 2140 : if (it == script.end()) return {};
131 [ - + ]: 2140 : if (*it != OP_NUMEQUAL) return {};
132 [ - + ]: 2140 : ++it;
133 [ - + ]: 2140 : if (it != script.end()) return {};
134 [ + - + - ]: 2140 : auto threshold = GetScriptNumber(opcode, data, 1, (int)keyspans.size());
135 [ - + ]: 2140 : if (!threshold) return {};
136 : :
137 : : // Construct result.
138 : 2140 : return std::pair{*threshold, std::move(keyspans)};
139 : 11147 : }
140 : :
141 : 17790254 : TxoutType Solver(const CScript& scriptPubKey, std::vector<std::vector<unsigned char>>& vSolutionsRet)
142 : : {
143 : 17790254 : vSolutionsRet.clear();
144 : :
145 : : // Shortcut for pay-to-script-hash, which are more constrained than the other types:
146 : : // it is always OP_HASH160 20 [20 byte hash] OP_EQUAL
147 [ + + ]: 17790254 : if (scriptPubKey.IsPayToScriptHash())
148 : : {
149 [ + + ]: 2214498 : std::vector<unsigned char> hashBytes(scriptPubKey.begin()+2, scriptPubKey.begin()+22);
150 [ + - ]: 1107249 : vSolutionsRet.push_back(hashBytes);
151 : 1107249 : return TxoutType::SCRIPTHASH;
152 : 1107249 : }
153 : :
154 : 16683005 : int witnessversion;
155 : 16683005 : std::vector<unsigned char> witnessprogram;
156 [ + - + + ]: 16683005 : if (scriptPubKey.IsWitnessProgram(witnessversion, witnessprogram)) {
157 [ + + + + ]: 11871264 : if (witnessversion == 0 && witnessprogram.size() == WITNESS_V0_KEYHASH_SIZE) {
158 [ + - ]: 659966 : vSolutionsRet.push_back(std::move(witnessprogram));
159 : : return TxoutType::WITNESS_V0_KEYHASH;
160 : : }
161 [ + + + + ]: 11211298 : if (witnessversion == 0 && witnessprogram.size() == WITNESS_V0_SCRIPTHASH_SIZE) {
162 [ + - ]: 10824246 : vSolutionsRet.push_back(std::move(witnessprogram));
163 : : return TxoutType::WITNESS_V0_SCRIPTHASH;
164 : : }
165 [ + + + + ]: 387052 : if (witnessversion == 1 && witnessprogram.size() == WITNESS_V1_TAPROOT_SIZE) {
166 [ + - ]: 257894 : vSolutionsRet.push_back(std::move(witnessprogram));
167 : : return TxoutType::WITNESS_V1_TAPROOT;
168 : : }
169 [ + - + + ]: 129158 : if (scriptPubKey.IsPayToAnchor()) {
170 : : return TxoutType::ANCHOR;
171 : : }
172 [ + + ]: 127813 : if (witnessversion != 0) {
173 [ + - + - ]: 242608 : vSolutionsRet.push_back(std::vector<unsigned char>{(unsigned char)witnessversion});
174 [ + - ]: 16683005 : vSolutionsRet.push_back(std::move(witnessprogram));
175 : : return TxoutType::WITNESS_UNKNOWN;
176 : : }
177 : : return TxoutType::NONSTANDARD;
178 : : }
179 : :
180 : : // Provably prunable, data-carrying output
181 : : //
182 : : // So long as script passes the IsUnspendable() test and all but the first
183 : : // byte passes the IsPushOnly() test we don't care what exactly is in the
184 : : // script.
185 [ + + + + : 10022509 : if (scriptPubKey.size() >= 1 && scriptPubKey[0] == OP_RETURN && scriptPubKey.IsPushOnly(scriptPubKey.begin()+1)) {
+ + + + +
- + + ]
186 : : return TxoutType::NULL_DATA;
187 : : }
188 : :
189 : 4787207 : std::vector<unsigned char> data;
190 [ + - + + ]: 4787207 : if (MatchPayToPubkey(scriptPubKey, data)) {
191 [ + - ]: 121324 : vSolutionsRet.push_back(std::move(data));
192 : : return TxoutType::PUBKEY;
193 : : }
194 : :
195 [ + - + + ]: 4665883 : if (MatchPayToPubkeyHash(scriptPubKey, data)) {
196 [ + - ]: 4787207 : vSolutionsRet.push_back(std::move(data));
197 : : return TxoutType::PUBKEYHASH;
198 : : }
199 : :
200 : 4188684 : int required;
201 : 4188684 : std::vector<std::vector<unsigned char>> keys;
202 [ + - + + ]: 4188684 : if (MatchMultisig(scriptPubKey, required, keys)) {
203 [ + - + - ]: 196794 : vSolutionsRet.push_back({static_cast<unsigned char>(required)}); // safe as required is in range 1..20
204 [ + - ]: 98397 : vSolutionsRet.insert(vSolutionsRet.end(), keys.begin(), keys.end());
205 [ + - ]: 196794 : vSolutionsRet.push_back({static_cast<unsigned char>(keys.size())}); // safe as size is in range 1..20
206 : 98397 : return TxoutType::MULTISIG;
207 : : }
208 : :
209 : 4090287 : vSolutionsRet.clear();
210 : 4090287 : return TxoutType::NONSTANDARD;
211 : 25658896 : }
212 : :
213 : 6151 : CScript GetScriptForRawPubKey(const CPubKey& pubKey)
214 : : {
215 [ + - + - ]: 6151 : return CScript() << std::vector<unsigned char>(pubKey.begin(), pubKey.end()) << OP_CHECKSIG;
216 : : }
217 : :
218 : 6970 : CScript GetScriptForMultisig(int nRequired, const std::vector<CPubKey>& keys)
219 : : {
220 : 6970 : CScript script;
221 : :
222 [ + - ]: 6970 : script << nRequired;
223 [ + + ]: 65010 : for (const CPubKey& key : keys)
224 [ + - ]: 58040 : script << ToByteVector(key);
225 [ + - + - ]: 6970 : script << keys.size() << OP_CHECKMULTISIG;
226 : :
227 : 6970 : return script;
228 : 0 : }
|