Branch data Line data Source code
1 : : // Copyright (c) 2023 The Bitcoin Core developers
2 : : // Distributed under the MIT software license, see the accompanying
3 : : // file COPYING or https://www.opensource.org/licenses/mit-license.php.
4 : :
5 : : #include <addresstype.h>
6 : :
7 : : #include <crypto/sha256.h>
8 : : #include <hash.h>
9 : : #include <pubkey.h>
10 : : #include <script/script.h>
11 : : #include <script/solver.h>
12 : : #include <uint256.h>
13 : : #include <util/hash_type.h>
14 : :
15 : : #include <cassert>
16 : : #include <vector>
17 : :
18 : : typedef std::vector<unsigned char> valtype;
19 : :
20 : 136535 : ScriptHash::ScriptHash(const CScript& in) : BaseHash(Hash160(in)) {}
21 : 1 : ScriptHash::ScriptHash(const CScriptID& in) : BaseHash{in} {}
22 : :
23 : 2379 : PKHash::PKHash(const CPubKey& pubkey) : BaseHash(pubkey.GetID()) {}
24 : 153304 : PKHash::PKHash(const CKeyID& pubkey_id) : BaseHash(pubkey_id) {}
25 : :
26 : 50 : WitnessV0KeyHash::WitnessV0KeyHash(const CPubKey& pubkey) : BaseHash(pubkey.GetID()) {}
27 : 1 : WitnessV0KeyHash::WitnessV0KeyHash(const PKHash& pubkey_hash) : BaseHash{pubkey_hash} {}
28 : :
29 : 196 : CKeyID ToKeyID(const PKHash& key_hash)
30 : : {
31 : 196 : return CKeyID{uint160{key_hash}};
32 : : }
33 : :
34 : 842 : CKeyID ToKeyID(const WitnessV0KeyHash& key_hash)
35 : : {
36 : 842 : return CKeyID{uint160{key_hash}};
37 : : }
38 : :
39 : 9229 : CScriptID ToScriptID(const ScriptHash& script_hash)
40 : : {
41 : 9229 : return CScriptID{uint160{script_hash}};
42 : : }
43 : :
44 : 17744 : WitnessV0ScriptHash::WitnessV0ScriptHash(const CScript& in)
45 : : {
46 [ + + + + ]: 52940 : CSHA256().Write(in.data(), in.size()).Finalize(begin());
47 : 17744 : }
48 : :
49 : 348143 : bool ExtractDestination(const CScript& scriptPubKey, CTxDestination& addressRet)
50 : : {
51 : 348143 : std::vector<valtype> vSolutions;
52 [ + - ]: 348143 : TxoutType whichType = Solver(scriptPubKey, vSolutions);
53 : :
54 [ + + + + : 348143 : switch (whichType) {
+ + + + +
- ]
55 : 561 : case TxoutType::PUBKEY: {
56 : 561 : CPubKey pubKey(vSolutions[0]);
57 [ - + ]: 561 : if (!pubKey.IsValid()) {
58 : 0 : addressRet = CNoDestination(scriptPubKey);
59 : : } else {
60 : 561 : addressRet = PubKeyDestination(pubKey);
61 : : }
62 : : return false;
63 : : }
64 : 78312 : case TxoutType::PUBKEYHASH: {
65 : 78312 : addressRet = PKHash(uint160(vSolutions[0]));
66 : 78312 : return true;
67 : : }
68 : 16795 : case TxoutType::SCRIPTHASH: {
69 : 16795 : addressRet = ScriptHash(uint160(vSolutions[0]));
70 : 16795 : return true;
71 : : }
72 : 180517 : case TxoutType::WITNESS_V0_KEYHASH: {
73 : 180517 : WitnessV0KeyHash hash;
74 : 180517 : std::copy(vSolutions[0].begin(), vSolutions[0].end(), hash.begin());
75 : 180517 : addressRet = hash;
76 : 180517 : return true;
77 : : }
78 : 3260 : case TxoutType::WITNESS_V0_SCRIPTHASH: {
79 : 3260 : WitnessV0ScriptHash hash;
80 : 3260 : std::copy(vSolutions[0].begin(), vSolutions[0].end(), hash.begin());
81 : 3260 : addressRet = hash;
82 : 3260 : return true;
83 : : }
84 : 21605 : case TxoutType::WITNESS_V1_TAPROOT: {
85 : 21605 : WitnessV1Taproot tap;
86 : 21605 : std::copy(vSolutions[0].begin(), vSolutions[0].end(), tap.begin());
87 : 21605 : addressRet = tap;
88 : 21605 : return true;
89 : : }
90 : 8 : case TxoutType::ANCHOR: {
91 [ + - ]: 8 : addressRet = PayToAnchor();
92 : 8 : return true;
93 : : }
94 : 53 : case TxoutType::WITNESS_UNKNOWN: {
95 [ + - ]: 53 : addressRet = WitnessUnknown{vSolutions[0][0], vSolutions[1]};
96 : 53 : return true;
97 : : }
98 : 47032 : case TxoutType::MULTISIG:
99 : 47032 : case TxoutType::NULL_DATA:
100 : 47032 : case TxoutType::NONSTANDARD:
101 : 47032 : addressRet = CNoDestination(scriptPubKey);
102 : 47032 : return false;
103 : : } // no default case, so the compiler can warn about missing cases
104 : 0 : assert(false);
105 : 348143 : }
106 : :
107 : : namespace {
108 : : class CScriptVisitor
109 : : {
110 : : public:
111 : 45 : CScript operator()(const CNoDestination& dest) const
112 : : {
113 : 45 : return dest.GetScript();
114 : : }
115 : :
116 : 27 : CScript operator()(const PubKeyDestination& dest) const
117 : : {
118 [ + - + - ]: 27 : return CScript() << ToByteVector(dest.GetPubKey()) << OP_CHECKSIG;
119 : : }
120 : :
121 : 190400 : CScript operator()(const PKHash& keyID) const
122 : : {
123 [ + - + - : 190400 : return CScript() << OP_DUP << OP_HASH160 << ToByteVector(keyID) << OP_EQUALVERIFY << OP_CHECKSIG;
+ - + - +
- ]
124 : : }
125 : :
126 : 143452 : CScript operator()(const ScriptHash& scriptID) const
127 : : {
128 [ + - + - : 143452 : return CScript() << OP_HASH160 << ToByteVector(scriptID) << OP_EQUAL;
+ - ]
129 : : }
130 : :
131 : 431715 : CScript operator()(const WitnessV0KeyHash& id) const
132 : : {
133 [ + - + - ]: 431715 : return CScript() << OP_0 << ToByteVector(id);
134 : : }
135 : :
136 : 19274 : CScript operator()(const WitnessV0ScriptHash& id) const
137 : : {
138 [ + - + - ]: 19274 : return CScript() << OP_0 << ToByteVector(id);
139 : : }
140 : :
141 : 134363 : CScript operator()(const WitnessV1Taproot& tap) const
142 : : {
143 [ + - + - ]: 134363 : return CScript() << OP_1 << ToByteVector(tap);
144 : : }
145 : :
146 : 66 : CScript operator()(const WitnessUnknown& id) const
147 : : {
148 [ + - ]: 66 : return CScript() << CScript::EncodeOP_N(id.GetWitnessVersion()) << id.GetWitnessProgram();
149 : : }
150 : : };
151 : :
152 : : class ValidDestinationVisitor
153 : : {
154 : : public:
155 : : bool operator()(const CNoDestination& dest) const { return false; }
156 : : bool operator()(const PubKeyDestination& dest) const { return false; }
157 : : bool operator()(const PKHash& dest) const { return true; }
158 : : bool operator()(const ScriptHash& dest) const { return true; }
159 : : bool operator()(const WitnessV0KeyHash& dest) const { return true; }
160 : : bool operator()(const WitnessV0ScriptHash& dest) const { return true; }
161 : : bool operator()(const WitnessV1Taproot& dest) const { return true; }
162 : : bool operator()(const WitnessUnknown& dest) const { return true; }
163 : : };
164 : : } // namespace
165 : :
166 : 919342 : CScript GetScriptForDestination(const CTxDestination& dest)
167 : : {
168 : 919342 : return std::visit(CScriptVisitor(), dest);
169 : : }
170 : :
171 : 21818 : bool IsValidDestination(const CTxDestination& dest) {
172 : 21818 : return std::visit(ValidDestinationVisitor(), dest);
173 : : }
|