Branch data Line data Source code
1 : : // Copyright (c) 2019-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 <chainparams.h>
6 : : #include <compressor.h>
7 : : #include <core_io.h>
8 : : #include <core_memusage.h>
9 : : #include <key_io.h>
10 : : #include <policy/policy.h>
11 : : #include <pubkey.h>
12 : : #include <rpc/util.h>
13 : : #include <script/descriptor.h>
14 : : #include <script/interpreter.h>
15 : : #include <script/script.h>
16 : : #include <script/script_error.h>
17 : : #include <script/sign.h>
18 : : #include <script/signingprovider.h>
19 : : #include <script/solver.h>
20 : : #include <streams.h>
21 : : #include <test/fuzz/FuzzedDataProvider.h>
22 : : #include <test/fuzz/fuzz.h>
23 : : #include <test/fuzz/util.h>
24 : : #include <univalue.h>
25 : : #include <util/chaintype.h>
26 : :
27 : : #include <algorithm>
28 : : #include <cassert>
29 : : #include <cstdint>
30 : : #include <optional>
31 : : #include <string>
32 : : #include <vector>
33 : :
34 : 1 : void initialize_script()
35 : : {
36 : 1 : SelectParams(ChainType::REGTEST);
37 : 1 : }
38 : :
39 [ + - ]: 2956 : FUZZ_TARGET(script, .init = initialize_script)
40 : : {
41 : 2542 : FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size());
42 : 2542 : const CScript script{ConsumeScript(fuzzed_data_provider)};
43 : :
44 : 2542 : CompressedScript compressed;
45 [ + - + + ]: 2542 : if (CompressScript(script, compressed)) {
46 [ - + ]: 27 : const unsigned int size = compressed[0];
47 : 27 : compressed.erase(compressed.begin());
48 [ - + ]: 27 : assert(size <= 5);
49 : 27 : CScript decompressed_script;
50 [ + - ]: 27 : const bool ok = DecompressScript(decompressed_script, size, compressed);
51 [ - + ]: 27 : assert(ok);
52 [ - + ]: 27 : assert(script == decompressed_script);
53 : 27 : }
54 : :
55 : 2542 : TxoutType which_type;
56 [ + - ]: 2542 : bool is_standard_ret = IsStandard(script, std::nullopt, which_type);
57 [ + + ]: 2542 : if (!is_standard_ret) {
58 [ + + + + : 2360 : assert(which_type == TxoutType::NONSTANDARD ||
- + ]
59 : : which_type == TxoutType::NULL_DATA ||
60 : : which_type == TxoutType::MULTISIG);
61 : : }
62 [ + + ]: 2542 : if (which_type == TxoutType::NONSTANDARD) {
63 [ - + ]: 907 : assert(!is_standard_ret);
64 : : }
65 [ + + ]: 2542 : if (which_type == TxoutType::NULL_DATA) {
66 [ - + ]: 14 : assert(script.IsUnspendable());
67 : : }
68 [ + + ]: 2542 : if (script.IsUnspendable()) {
69 [ + + - + ]: 114 : assert(which_type == TxoutType::NULL_DATA ||
70 : : which_type == TxoutType::NONSTANDARD);
71 : : }
72 : :
73 : 2542 : CTxDestination address;
74 [ + - ]: 2542 : bool extract_destination_ret = ExtractDestination(script, address);
75 [ + + ]: 2542 : if (!extract_destination_ret) {
76 [ - + ]: 2410 : assert(which_type == TxoutType::PUBKEY ||
77 : : which_type == TxoutType::NONSTANDARD ||
78 : : which_type == TxoutType::NULL_DATA ||
79 : : which_type == TxoutType::MULTISIG);
80 : : }
81 [ + + + + ]: 2542 : if (which_type == TxoutType::NONSTANDARD ||
82 [ + + ]: 1621 : which_type == TxoutType::NULL_DATA ||
83 : : which_type == TxoutType::MULTISIG) {
84 [ - + ]: 2369 : assert(!extract_destination_ret);
85 : : }
86 : :
87 : 2542 : const FlatSigningProvider signing_provider;
88 [ + - ]: 2542 : (void)InferDescriptor(script, signing_provider);
89 [ + - ]: 2542 : (void)IsSegWitOutput(signing_provider, script);
90 : :
91 [ + + ]: 2542 : (void)RecursiveDynamicUsage(script);
92 : :
93 : 2542 : std::vector<std::vector<unsigned char>> solutions;
94 [ + - ]: 2542 : (void)Solver(script, solutions);
95 : :
96 [ + - ]: 2542 : (void)script.HasValidOps();
97 [ + - ]: 2542 : (void)script.IsPayToAnchor();
98 [ + - ]: 2542 : (void)script.IsPayToScriptHash();
99 [ + - ]: 2542 : (void)script.IsPayToWitnessScriptHash();
100 [ + - ]: 2542 : (void)script.IsPushOnly();
101 [ + - ]: 2542 : (void)script.GetSigOpCount(/* fAccurate= */ false);
102 : :
103 : 2542 : {
104 : 2542 : const std::vector<uint8_t> bytes = ConsumeRandomLengthByteVector(fuzzed_data_provider);
105 : 2542 : CompressedScript compressed_script;
106 : 2542 : compressed_script.assign(bytes.begin(), bytes.end());
107 : : // DecompressScript(..., ..., bytes) is not guaranteed to be defined if the bytes vector is too short
108 [ + + + + ]: 3139 : if (compressed_script.size() >= 32) {
109 : 631 : CScript decompressed_script;
110 [ + - ]: 631 : DecompressScript(decompressed_script, fuzzed_data_provider.ConsumeIntegral<unsigned int>(), compressed_script);
111 : 631 : }
112 : 2542 : }
113 : :
114 : 2542 : const std::optional<CScript> other_script = ConsumeDeserializable<CScript>(fuzzed_data_provider);
115 [ + + ]: 2542 : if (other_script) {
116 : 476 : {
117 : 476 : CScript script_mut{script};
118 [ + - ]: 476 : (void)FindAndDelete(script_mut, *other_script);
119 : 0 : }
120 : 476 : const std::vector<std::string> random_string_vector = ConsumeRandomLengthStringVector(fuzzed_data_provider);
121 : 476 : const uint32_t u32{fuzzed_data_provider.ConsumeIntegral<uint32_t>()};
122 : 476 : const uint32_t flags{u32 | SCRIPT_VERIFY_P2SH};
123 : 476 : {
124 : 476 : CScriptWitness wit;
125 [ + + ]: 2033 : for (const auto& s : random_string_vector) {
126 [ + - ]: 1557 : wit.stack.emplace_back(s.begin(), s.end());
127 : : }
128 [ + - ]: 476 : (void)CountWitnessSigOps(script, *other_script, &wit, flags);
129 : 476 : wit.SetNull();
130 : 0 : }
131 : 476 : }
132 : :
133 [ + - ]: 2542 : (void)GetOpName(ConsumeOpcodeType(fuzzed_data_provider));
134 [ + - ]: 2542 : (void)ScriptErrorString(static_cast<ScriptError>(fuzzed_data_provider.ConsumeIntegralInRange<int>(0, SCRIPT_ERR_ERROR_COUNT)));
135 : :
136 : 2542 : {
137 : 2542 : const std::vector<uint8_t> bytes = ConsumeRandomLengthByteVector(fuzzed_data_provider);
138 : 2542 : CScript append_script{bytes.begin(), bytes.end()};
139 [ + - ]: 2542 : append_script << fuzzed_data_provider.ConsumeIntegral<int64_t>();
140 [ + - ]: 2542 : append_script << ConsumeOpcodeType(fuzzed_data_provider);
141 [ + - ]: 2542 : append_script << CScriptNum{fuzzed_data_provider.ConsumeIntegral<int64_t>()};
142 : 2542 : append_script << ConsumeRandomLengthByteVector(fuzzed_data_provider);
143 : 2542 : }
144 : :
145 : 2542 : {
146 : 2542 : const CTxDestination tx_destination_1{
147 [ + + ]: 2542 : fuzzed_data_provider.ConsumeBool() ?
148 : 460 : DecodeDestination(fuzzed_data_provider.ConsumeRandomLengthString()) :
149 [ + - ]: 2542 : ConsumeTxDestination(fuzzed_data_provider)};
150 : 2542 : const CTxDestination tx_destination_2{ConsumeTxDestination(fuzzed_data_provider)};
151 [ + - ]: 2542 : const std::string encoded_dest{EncodeDestination(tx_destination_1)};
152 [ + - ]: 2542 : const UniValue json_dest{DescribeAddress(tx_destination_1)};
153 [ + - ]: 2542 : (void)GetKeyForDestination(/*store=*/{}, tx_destination_1);
154 [ + - ]: 2542 : const CScript dest{GetScriptForDestination(tx_destination_1)};
155 [ + - ]: 2542 : const bool valid{IsValidDestination(tx_destination_1)};
156 : :
157 [ + + ]: 2542 : if (!std::get_if<PubKeyDestination>(&tx_destination_1)) {
158 : : // Only try to round trip non-pubkey destinations since PubKeyDestination has no encoding
159 [ + + + - ]: 2537 : Assert(dest.empty() != valid);
160 [ + - + - ]: 4936 : Assert(tx_destination_1 == DecodeDestination(encoded_dest));
161 [ + - + - ]: 2468 : Assert(valid == IsValidDestinationString(encoded_dest));
162 : : }
163 : :
164 : 2542 : (void)(tx_destination_1 < tx_destination_2);
165 [ + + ]: 2542 : if (tx_destination_1 == tx_destination_2) {
166 [ + - + - ]: 2043 : Assert(encoded_dest == EncodeDestination(tx_destination_2));
167 [ + - + - : 4086 : Assert(json_dest.write() == DescribeAddress(tx_destination_2).write());
+ - + - ]
168 [ + - + - ]: 4086 : Assert(dest == GetScriptForDestination(tx_destination_2));
169 : : }
170 : 2542 : }
171 : 2542 : }
|