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