Bitcoin Core  26.1.0
P2P Digital Currency
util.cpp
Go to the documentation of this file.
1 // Copyright (c) 2021-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 <consensus/amount.h>
6 #include <pubkey.h>
7 #include <test/fuzz/util.h>
8 #include <test/util/script.h>
9 #include <util/check.h>
10 #include <util/overflow.h>
11 #include <util/rbf.h>
12 #include <util/time.h>
13 #include <version.h>
14 
15 #include <memory>
16 
17 std::vector<uint8_t> ConstructPubKeyBytes(FuzzedDataProvider& fuzzed_data_provider, Span<const uint8_t> byte_data, const bool compressed) noexcept
18 {
19  uint8_t pk_type;
20  if (compressed) {
21  pk_type = fuzzed_data_provider.PickValueInArray({0x02, 0x03});
22  } else {
23  pk_type = fuzzed_data_provider.PickValueInArray({0x04, 0x06, 0x07});
24  }
25  std::vector<uint8_t> pk_data{byte_data.begin(), byte_data.begin() + (compressed ? CPubKey::COMPRESSED_SIZE : CPubKey::SIZE)};
26  pk_data[0] = pk_type;
27  return pk_data;
28 }
29 
30 CAmount ConsumeMoney(FuzzedDataProvider& fuzzed_data_provider, const std::optional<CAmount>& max) noexcept
31 {
32  return fuzzed_data_provider.ConsumeIntegralInRange<CAmount>(0, max.value_or(MAX_MONEY));
33 }
34 
35 int64_t ConsumeTime(FuzzedDataProvider& fuzzed_data_provider, const std::optional<int64_t>& min, const std::optional<int64_t>& max) noexcept
36 {
37  // Avoid t=0 (1970-01-01T00:00:00Z) since SetMockTime(0) disables mocktime.
38  static const int64_t time_min{946684801}; // 2000-01-01T00:00:01Z
39  static const int64_t time_max{4133980799}; // 2100-12-31T23:59:59Z
40  return fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(min.value_or(time_min), max.value_or(time_max));
41 }
42 
43 CMutableTransaction ConsumeTransaction(FuzzedDataProvider& fuzzed_data_provider, const std::optional<std::vector<uint256>>& prevout_txids, const int max_num_in, const int max_num_out) noexcept
44 {
45  CMutableTransaction tx_mut;
46  const auto p2wsh_op_true = fuzzed_data_provider.ConsumeBool();
47  tx_mut.nVersion = fuzzed_data_provider.ConsumeBool() ?
49  fuzzed_data_provider.ConsumeIntegral<int32_t>();
50  tx_mut.nLockTime = fuzzed_data_provider.ConsumeIntegral<uint32_t>();
51  const auto num_in = fuzzed_data_provider.ConsumeIntegralInRange<int>(0, max_num_in);
52  const auto num_out = fuzzed_data_provider.ConsumeIntegralInRange<int>(0, max_num_out);
53  for (int i = 0; i < num_in; ++i) {
54  const auto& txid_prev = prevout_txids ?
55  PickValue(fuzzed_data_provider, *prevout_txids) :
56  ConsumeUInt256(fuzzed_data_provider);
57  const auto index_out = fuzzed_data_provider.ConsumeIntegralInRange<uint32_t>(0, max_num_out);
58  const auto sequence = ConsumeSequence(fuzzed_data_provider);
59  const auto script_sig = p2wsh_op_true ? CScript{} : ConsumeScript(fuzzed_data_provider);
60  CScriptWitness script_wit;
61  if (p2wsh_op_true) {
62  script_wit.stack = std::vector<std::vector<uint8_t>>{WITNESS_STACK_ELEM_OP_TRUE};
63  } else {
64  script_wit = ConsumeScriptWitness(fuzzed_data_provider);
65  }
66  CTxIn in;
67  in.prevout = COutPoint{txid_prev, index_out};
68  in.nSequence = sequence;
69  in.scriptSig = script_sig;
70  in.scriptWitness = script_wit;
71 
72  tx_mut.vin.push_back(in);
73  }
74  for (int i = 0; i < num_out; ++i) {
75  const auto amount = fuzzed_data_provider.ConsumeIntegralInRange<CAmount>(-10, 50 * COIN + 10);
76  const auto script_pk = p2wsh_op_true ?
78  ConsumeScript(fuzzed_data_provider, /*maybe_p2wsh=*/true);
79  tx_mut.vout.emplace_back(amount, script_pk);
80  }
81  return tx_mut;
82 }
83 
84 CScriptWitness ConsumeScriptWitness(FuzzedDataProvider& fuzzed_data_provider, const size_t max_stack_elem_size) noexcept
85 {
87  const auto n_elements = fuzzed_data_provider.ConsumeIntegralInRange<size_t>(0, max_stack_elem_size);
88  for (size_t i = 0; i < n_elements; ++i) {
89  ret.stack.push_back(ConsumeRandomLengthByteVector(fuzzed_data_provider));
90  }
91  return ret;
92 }
93 
94 CScript ConsumeScript(FuzzedDataProvider& fuzzed_data_provider, const bool maybe_p2wsh) noexcept
95 {
96  CScript r_script{};
97  {
98  // Keep a buffer of bytes to allow the fuzz engine to produce smaller
99  // inputs to generate CScripts with repeated data.
100  static constexpr unsigned MAX_BUFFER_SZ{128};
101  std::vector<uint8_t> buffer(MAX_BUFFER_SZ, uint8_t{'a'});
102  while (fuzzed_data_provider.ConsumeBool()) {
103  CallOneOf(
104  fuzzed_data_provider,
105  [&] {
106  // Insert byte vector directly to allow malformed or unparsable scripts
107  r_script.insert(r_script.end(), buffer.begin(), buffer.begin() + fuzzed_data_provider.ConsumeIntegralInRange(0U, MAX_BUFFER_SZ));
108  },
109  [&] {
110  // Push a byte vector from the buffer
111  r_script << std::vector<uint8_t>{buffer.begin(), buffer.begin() + fuzzed_data_provider.ConsumeIntegralInRange(0U, MAX_BUFFER_SZ)};
112  },
113  [&] {
114  // Push multisig
115  // There is a special case for this to aid the fuzz engine
116  // navigate the highly structured multisig format.
117  r_script << fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(0, 22);
118  int num_data{fuzzed_data_provider.ConsumeIntegralInRange(1, 22)};
119  while (num_data--) {
120  auto pubkey_bytes{ConstructPubKeyBytes(fuzzed_data_provider, buffer, fuzzed_data_provider.ConsumeBool())};
121  if (fuzzed_data_provider.ConsumeBool()) {
122  pubkey_bytes.back() = num_data; // Make each pubkey different
123  }
124  r_script << pubkey_bytes;
125  }
126  r_script << fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(0, 22);
127  },
128  [&] {
129  // Mutate the buffer
130  const auto vec{ConsumeRandomLengthByteVector(fuzzed_data_provider, /*max_length=*/MAX_BUFFER_SZ)};
131  std::copy(vec.begin(), vec.end(), buffer.begin());
132  },
133  [&] {
134  // Push an integral
135  r_script << fuzzed_data_provider.ConsumeIntegral<int64_t>();
136  },
137  [&] {
138  // Push an opcode
139  r_script << ConsumeOpcodeType(fuzzed_data_provider);
140  },
141  [&] {
142  // Push a scriptnum
143  r_script << ConsumeScriptNum(fuzzed_data_provider);
144  });
145  }
146  }
147  if (maybe_p2wsh && fuzzed_data_provider.ConsumeBool()) {
148  uint256 script_hash;
149  CSHA256().Write(r_script.data(), r_script.size()).Finalize(script_hash.begin());
150  r_script.clear();
151  r_script << OP_0 << ToByteVector(script_hash);
152  }
153  return r_script;
154 }
155 
156 uint32_t ConsumeSequence(FuzzedDataProvider& fuzzed_data_provider) noexcept
157 {
158  return fuzzed_data_provider.ConsumeBool() ?
159  fuzzed_data_provider.PickValueInArray({
163  }) :
164  fuzzed_data_provider.ConsumeIntegral<uint32_t>();
165 }
166 
168 {
169  CTxDestination tx_destination;
170  const size_t call_size{CallOneOf(
171  fuzzed_data_provider,
172  [&] {
173  tx_destination = CNoDestination{};
174  },
175  [&] {
176  bool compressed = fuzzed_data_provider.ConsumeBool();
178  fuzzed_data_provider,
179  ConsumeFixedLengthByteVector(fuzzed_data_provider, (compressed ? CPubKey::COMPRESSED_SIZE : CPubKey::SIZE)),
180  compressed
181  )};
182  tx_destination = PubKeyDestination{pk};
183  },
184  [&] {
185  tx_destination = PKHash{ConsumeUInt160(fuzzed_data_provider)};
186  },
187  [&] {
188  tx_destination = ScriptHash{ConsumeUInt160(fuzzed_data_provider)};
189  },
190  [&] {
191  tx_destination = WitnessV0ScriptHash{ConsumeUInt256(fuzzed_data_provider)};
192  },
193  [&] {
194  tx_destination = WitnessV0KeyHash{ConsumeUInt160(fuzzed_data_provider)};
195  },
196  [&] {
197  tx_destination = WitnessV1Taproot{XOnlyPubKey{ConsumeUInt256(fuzzed_data_provider)}};
198  },
199  [&] {
200  std::vector<unsigned char> program{ConsumeRandomLengthByteVector(fuzzed_data_provider, /*max_length=*/40)};
201  if (program.size() < 2) {
202  program = {0, 0};
203  }
204  tx_destination = WitnessUnknown{fuzzed_data_provider.ConsumeIntegralInRange<unsigned int>(2, 16), program};
205  })};
206  Assert(call_size == std::variant_size_v<CTxDestination>);
207  return tx_destination;
208 }
209 
210 CKey ConsumePrivateKey(FuzzedDataProvider& fuzzed_data_provider, std::optional<bool> compressed) noexcept
211 {
212  auto key_data = fuzzed_data_provider.ConsumeBytes<uint8_t>(32);
213  key_data.resize(32);
214  CKey key;
215  bool compressed_value = compressed ? *compressed : fuzzed_data_provider.ConsumeBool();
216  key.Set(key_data.begin(), key_data.end(), compressed_value);
217  return key;
218 }
219 
220 bool ContainsSpentInput(const CTransaction& tx, const CCoinsViewCache& inputs) noexcept
221 {
222  for (const CTxIn& tx_in : tx.vin) {
223  const Coin& coin = inputs.AccessCoin(tx_in.prevout);
224  if (coin.IsSpent()) {
225  return true;
226  }
227  }
228  return false;
229 }
230 
232 {
235  return nullptr;
236  }
237  std::string mode;
238  CallOneOf(
240  [&] {
241  mode = "r";
242  },
243  [&] {
244  mode = "r+";
245  },
246  [&] {
247  mode = "w";
248  },
249  [&] {
250  mode = "w+";
251  },
252  [&] {
253  mode = "a";
254  },
255  [&] {
256  mode = "a+";
257  });
258 #if defined _GNU_SOURCE && (defined(__linux__) || defined(__FreeBSD__))
259  const cookie_io_functions_t io_hooks = {
264  };
265  return fopencookie(this, mode.c_str(), io_hooks);
266 #else
267  (void)mode;
268  return nullptr;
269 #endif
270 }
271 
272 ssize_t FuzzedFileProvider::read(void* cookie, char* buf, size_t size)
273 {
274  FuzzedFileProvider* fuzzed_file = (FuzzedFileProvider*)cookie;
276  if (buf == nullptr || size == 0 || fuzzed_file->m_fuzzed_data_provider.ConsumeBool()) {
277  return fuzzed_file->m_fuzzed_data_provider.ConsumeBool() ? 0 : -1;
278  }
279  const std::vector<uint8_t> random_bytes = fuzzed_file->m_fuzzed_data_provider.ConsumeBytes<uint8_t>(size);
280  if (random_bytes.empty()) {
281  return 0;
282  }
283  std::memcpy(buf, random_bytes.data(), random_bytes.size());
284  if (AdditionOverflow(fuzzed_file->m_offset, (int64_t)random_bytes.size())) {
285  return fuzzed_file->m_fuzzed_data_provider.ConsumeBool() ? 0 : -1;
286  }
287  fuzzed_file->m_offset += random_bytes.size();
288  return random_bytes.size();
289 }
290 
291 ssize_t FuzzedFileProvider::write(void* cookie, const char* buf, size_t size)
292 {
293  FuzzedFileProvider* fuzzed_file = (FuzzedFileProvider*)cookie;
295  const ssize_t n = fuzzed_file->m_fuzzed_data_provider.ConsumeIntegralInRange<ssize_t>(0, size);
296  if (AdditionOverflow(fuzzed_file->m_offset, (int64_t)n)) {
297  return 0;
298  }
299  fuzzed_file->m_offset += n;
300  return n;
301 }
302 
303 int FuzzedFileProvider::seek(void* cookie, int64_t* offset, int whence)
304 {
305  assert(whence == SEEK_SET || whence == SEEK_CUR || whence == SEEK_END);
306  FuzzedFileProvider* fuzzed_file = (FuzzedFileProvider*)cookie;
308  int64_t new_offset = 0;
309  if (whence == SEEK_SET) {
310  new_offset = *offset;
311  } else if (whence == SEEK_CUR) {
312  if (AdditionOverflow(fuzzed_file->m_offset, *offset)) {
313  return -1;
314  }
315  new_offset = fuzzed_file->m_offset + *offset;
316  } else if (whence == SEEK_END) {
317  const int64_t n = fuzzed_file->m_fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(0, 4096);
318  if (AdditionOverflow(n, *offset)) {
319  return -1;
320  }
321  new_offset = n + *offset;
322  }
323  if (new_offset < 0) {
324  return -1;
325  }
326  fuzzed_file->m_offset = new_offset;
327  *offset = new_offset;
328  return fuzzed_file->m_fuzzed_data_provider.ConsumeIntegralInRange<int>(-1, 0);
329 }
330 
331 int FuzzedFileProvider::close(void* cookie)
332 {
333  FuzzedFileProvider* fuzzed_file = (FuzzedFileProvider*)cookie;
335  return fuzzed_file->m_fuzzed_data_provider.ConsumeIntegralInRange<int>(-1, 0);
336 }
CSHA256 & Write(const unsigned char *data, size_t len)
Definition: sha256.cpp:694
bool IsSpent() const
Either this coin never existed (see e.g.
Definition: coins.h:80
int ret
std::vector< B > ConsumeFixedLengthByteVector(FuzzedDataProvider &fuzzed_data_provider, const size_t length) noexcept
Returns a byte vector of specified size regardless of the number of remaining bytes available from th...
Definition: util.h:237
static constexpr unsigned int SIZE
secp256k1:
Definition: pubkey.h:39
bool AdditionOverflow(const T i, const T j) noexcept
Definition: overflow.h:13
assert(!tx.IsCoinBase())
static int seek(void *cookie, int64_t *offset, int whence)
Definition: util.cpp:303
A UTXO entry.
Definition: coins.h:31
int64_t m_offset
Definition: util.h:248
std::vector< CTxIn > vin
Definition: transaction.h:381
static const CScript P2WSH_OP_TRUE
Definition: script.h:12
CScriptWitness scriptWitness
Only serialized through CTransaction.
Definition: transaction.h:80
static const uint32_t SEQUENCE_FINAL
Setting nSequence to this value for every input in a transaction disables nLockTime/IsFinalTx().
Definition: transaction.h:89
opcodetype ConsumeOpcodeType(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:139
CScriptWitness ConsumeScriptWitness(FuzzedDataProvider &fuzzed_data_provider, const size_t max_stack_elem_size) noexcept
Definition: util.cpp:84
CScript ConsumeScript(FuzzedDataProvider &fuzzed_data_provider, const bool maybe_p2wsh) noexcept
Definition: util.cpp:94
FuzzedDataProvider & m_fuzzed_data_provider
Definition: util.h:247
std::vector< std::vector< unsigned char > > stack
Definition: script.h:568
Definition: script.h:74
std::vector< B > ConsumeRandomLengthByteVector(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
Definition: util.h:58
constexpr unsigned char * begin()
Definition: uint256.h:68
static int close(void *cookie)
Definition: util.cpp:331
int64_t CAmount
Amount in satoshis (Can be negative)
Definition: amount.h:12
static constexpr unsigned int COMPRESSED_SIZE
Definition: pubkey.h:40
static constexpr uint32_t MAX_BIP125_RBF_SEQUENCE
Definition: rbf.h:12
An input of a transaction.
Definition: transaction.h:74
CTxDestination subtype to encode any future Witness version.
Definition: addresstype.h:92
static const std::vector< uint8_t > WITNESS_STACK_ELEM_OP_TRUE
Definition: script.h:11
An encapsulated public key.
Definition: pubkey.h:33
bool ContainsSpentInput(const CTransaction &tx, const CCoinsViewCache &inputs) noexcept
Definition: util.cpp:220
std::vector< uint8_t > ConstructPubKeyBytes(FuzzedDataProvider &fuzzed_data_provider, Span< const uint8_t > byte_data, const bool compressed) noexcept
Definition: util.cpp:17
An outpoint - a combination of a transaction hash and an index n into its vout.
Definition: transaction.h:35
std::vector< CTxOut > vout
Definition: transaction.h:382
void Set(const T pbegin, const T pend, bool fCompressedIn)
Initialize using begin and end iterators to byte data.
Definition: key.h:99
FILE * open()
Definition: util.cpp:231
CScript scriptSig
Definition: transaction.h:78
std::vector< T > ConsumeBytes(size_t num_bytes)
std::variant< CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, WitnessUnknown > CTxDestination
A txout script categorized into standard templates.
Definition: addresstype.h:129
std::vector< unsigned char > ToByteVector(const T &in)
Definition: script.h:65
256-bit opaque blob.
Definition: uint256.h:106
static ssize_t write(void *cookie, const char *buf, size_t size)
Definition: util.cpp:291
int64_t ConsumeTime(FuzzedDataProvider &fuzzed_data_provider, const std::optional< int64_t > &min, const std::optional< int64_t > &max) noexcept
Definition: util.cpp:35
static ssize_t read(void *cookie, char *buf, size_t size)
Definition: util.cpp:272
Serialized script, used inside transaction inputs and outputs.
Definition: script.h:412
uint32_t nSequence
Definition: transaction.h:79
CScriptNum ConsumeScriptNum(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:156
CAmount ConsumeMoney(FuzzedDataProvider &fuzzed_data_provider, const std::optional< CAmount > &max) noexcept
Definition: util.cpp:30
void SetFuzzedErrNo(FuzzedDataProvider &fuzzed_data_provider, const std::array< T, size > &errnos)
Sets errno to a value selected from the given std::array errnos.
Definition: util.h:217
static constexpr CAmount MAX_MONEY
No amount larger than this (in satoshi) is valid.
Definition: amount.h:26
A mutable version of CTransaction.
Definition: transaction.h:379
static const uint32_t MAX_SEQUENCE_NONFINAL
This is the maximum sequence number that enables both nLockTime and OP_CHECKLOCKTIMEVERIFY (BIP 65)...
Definition: transaction.h:95
size_t CallOneOf(FuzzedDataProvider &fuzzed_data_provider, Callables... callables)
Definition: util.h:36
uint64_t sequence
CMutableTransaction ConsumeTransaction(FuzzedDataProvider &fuzzed_data_provider, const std::optional< std::vector< uint256 >> &prevout_txids, const int max_num_in, const int max_num_out) noexcept
Definition: util.cpp:43
auto & PickValue(FuzzedDataProvider &fuzzed_data_provider, Collection &col)
Definition: util.h:48
An encapsulated private key.
Definition: key.h:32
A Span is an object that can refer to a contiguous sequence of objects.
Definition: solver.h:20
The basic transaction that is broadcasted on the network and contained in blocks. ...
Definition: transaction.h:294
static const int32_t CURRENT_VERSION
Definition: transaction.h:298
uint160 ConsumeUInt160(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:161
T ConsumeIntegralInRange(T min, T max)
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Definition: coins.h:228
uint256 ConsumeUInt256(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:170
A hasher class for SHA-256.
Definition: sha256.h:13
COutPoint prevout
Definition: transaction.h:77
CTxDestination ConsumeTxDestination(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.cpp:167
CKey ConsumePrivateKey(FuzzedDataProvider &fuzzed_data_provider, std::optional< bool > compressed) noexcept
Definition: util.cpp:210
#define Assert(val)
Identity function.
Definition: check.h:73
uint32_t ConsumeSequence(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.cpp:156
static constexpr CAmount COIN
The amount of satoshis in one BTC.
Definition: amount.h:15