Bitcoin Core 31.99.0
P2P Digital Currency
util.h
Go to the documentation of this file.
1// Copyright (c) 2009-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#ifndef BITCOIN_TEST_FUZZ_UTIL_H
6#define BITCOIN_TEST_FUZZ_UTIL_H
7
8#include <addresstype.h>
9#include <arith_uint256.h>
10#include <coins.h>
11#include <compat/compat.h>
12#include <consensus/amount.h>
13#include <consensus/consensus.h>
14#include <key.h>
15#include <merkleblock.h>
16#include <pow.h>
18#include <script/script.h>
19#include <serialize.h>
20#include <streams.h>
22#include <test/fuzz/fuzz.h>
23#include <uint256.h>
24#include <validation.h>
25
26#include <algorithm>
27#include <array>
28#include <cstdint>
29#include <cstdio>
30#include <optional>
31#include <string>
32#include <vector>
33
34class PeerManager;
35
36template <typename... Callables>
37size_t CallOneOf(FuzzedDataProvider& fuzzed_data_provider, Callables... callables)
38{
39 constexpr size_t call_size{sizeof...(callables)};
40 static_assert(call_size >= 1);
41 const size_t call_index{fuzzed_data_provider.ConsumeIntegralInRange<size_t>(0, call_size - 1)};
42
43 size_t i{0};
44 ((i++ == call_index ? callables() : void()), ...);
45 return call_size;
46}
47
48template <typename Collection>
50{
51 auto sz{col.size()};
52 assert(sz >= 1);
53 auto it = col.begin();
54 std::advance(it, fuzzed_data_provider.ConsumeIntegralInRange<decltype(sz)>(0, sz - 1));
55 return *it;
56}
57
58template<typename B = uint8_t>
59[[nodiscard]] inline std::vector<B> ConsumeRandomLengthByteVector(FuzzedDataProvider& fuzzed_data_provider, const std::optional<size_t>& max_length = std::nullopt) noexcept
60{
61 static_assert(sizeof(B) == 1);
62 const std::string s = max_length ?
65 std::vector<B> ret(s.size());
66 std::copy(s.begin(), s.end(), reinterpret_cast<char*>(ret.data()));
67 return ret;
68}
69
70[[nodiscard]] inline DataStream ConsumeDataStream(FuzzedDataProvider& fuzzed_data_provider, const std::optional<size_t>& max_length = std::nullopt) noexcept
71{
73}
74
75[[nodiscard]] inline std::vector<std::string> ConsumeRandomLengthStringVector(FuzzedDataProvider& fuzzed_data_provider, const size_t max_vector_size = 16, const size_t max_string_length = 16) noexcept
76{
77 const size_t n_elements = fuzzed_data_provider.ConsumeIntegralInRange<size_t>(0, max_vector_size);
78 std::vector<std::string> r;
79 r.reserve(n_elements);
80 for (size_t i = 0; i < n_elements; ++i) {
81 r.push_back(fuzzed_data_provider.ConsumeRandomLengthString(max_string_length));
82 }
83 return r;
84}
85
86template <typename T>
87[[nodiscard]] inline std::vector<T> ConsumeRandomLengthIntegralVector(FuzzedDataProvider& fuzzed_data_provider, const size_t max_vector_size = 16) noexcept
88{
89 const size_t n_elements = fuzzed_data_provider.ConsumeIntegralInRange<size_t>(0, max_vector_size);
90 std::vector<T> r;
91 r.reserve(n_elements);
92 for (size_t i = 0; i < n_elements; ++i) {
94 }
95 return r;
96}
97
98template <typename P>
100
101template <typename T, typename P>
102[[nodiscard]] std::optional<T> ConsumeDeserializable(FuzzedDataProvider& fuzzed_data_provider, const P& params, const std::optional<size_t>& max_length = std::nullopt) noexcept
103{
104 const std::vector<uint8_t> buffer{ConsumeRandomLengthByteVector(fuzzed_data_provider, max_length)};
105 SpanReader ds{buffer};
106 T obj;
107 try {
108 ds >> params(obj);
109 } catch (const std::ios_base::failure&) {
110 return std::nullopt;
111 }
112 return obj;
113}
114
115template <typename T>
116[[nodiscard]] inline std::optional<T> ConsumeDeserializable(FuzzedDataProvider& fuzzed_data_provider, const std::optional<size_t>& max_length = std::nullopt) noexcept
117{
118 const std::vector<uint8_t> buffer = ConsumeRandomLengthByteVector(fuzzed_data_provider, max_length);
119 SpanReader ds{buffer};
120 T obj;
121 try {
122 ds >> obj;
123 } catch (const std::ios_base::failure&) {
124 return std::nullopt;
125 }
126 return obj;
127}
128
129template <typename T>
130[[nodiscard]] inline std::optional<T> ConsumeDeserializableConstructor(FuzzedDataProvider& fuzzed_data_provider, const std::optional<size_t>& max_length = std::nullopt) noexcept
131{
132 const std::vector<uint8_t> buffer = ConsumeRandomLengthByteVector(fuzzed_data_provider, max_length);
133 SpanReader ds{buffer};
134 try {
135 T obj(deserialize, ds);
136 return obj;
137 } catch (const std::ios_base::failure&) {
138 return std::nullopt;
139 }
140}
141
142template <typename WeakEnumType, size_t size>
143[[nodiscard]] WeakEnumType ConsumeWeakEnum(FuzzedDataProvider& fuzzed_data_provider, const WeakEnumType (&all_types)[size]) noexcept
144{
146 fuzzed_data_provider.PickValueInArray<WeakEnumType>(all_types) :
147 WeakEnumType(fuzzed_data_provider.ConsumeIntegral<std::underlying_type_t<WeakEnumType>>());
148}
149
151{
152 return static_cast<opcodetype>(fuzzed_data_provider.ConsumeIntegralInRange<uint32_t>(0, MAX_OPCODE));
153}
154
155[[nodiscard]] CAmount ConsumeMoney(FuzzedDataProvider& fuzzed_data_provider, const std::optional<CAmount>& max = std::nullopt) noexcept;
156
157[[nodiscard]] NodeSeconds ConsumeTime(FuzzedDataProvider& fuzzed_data_provider, const std::optional<int64_t>& min = std::nullopt, const std::optional<int64_t>& max = std::nullopt) noexcept;
158
159template <class Dur>
160// Having the compiler infer the template argument from the function argument
161// is dangerous, because the desired return value generally has a different
162// type than the function argument. So std::common_type is used to force the
163// call site to specify the type of the return value.
164[[nodiscard]] Dur ConsumeDuration(FuzzedDataProvider& fuzzed_data_provider, std::common_type_t<Dur> min, std::common_type_t<Dur> max) noexcept
165{
166 return Dur{fuzzed_data_provider.ConsumeIntegralInRange(min.count(), max.count())};
167}
168
169[[nodiscard]] CMutableTransaction ConsumeTransaction(FuzzedDataProvider& fuzzed_data_provider, const std::optional<std::vector<Txid>>& prevout_txids, int max_num_in = 10, int max_num_out = 10) noexcept;
170
171[[nodiscard]] CScriptWitness ConsumeScriptWitness(FuzzedDataProvider& fuzzed_data_provider, size_t max_stack_elem_size = 32) noexcept;
172
173[[nodiscard]] CScript ConsumeScript(FuzzedDataProvider& fuzzed_data_provider, bool maybe_p2wsh = false) noexcept;
174
175[[nodiscard]] uint32_t ConsumeSequence(FuzzedDataProvider& fuzzed_data_provider) noexcept;
176
178{
180}
181
183{
184 const std::vector<uint8_t> v160 = fuzzed_data_provider.ConsumeBytes<uint8_t>(160 / 8);
185 if (v160.size() != 160 / 8) {
186 return {};
187 }
188 return uint160{v160};
189}
190
192{
193 const std::vector<uint8_t> v256 = fuzzed_data_provider.ConsumeBytes<uint8_t>(256 / 8);
194 if (v256.size() != 256 / 8) {
195 return {};
196 }
197 return uint256{v256};
198}
199
201{
203}
204
206{
207 assert(min <= max);
208 const arith_uint256 range = max - min;
210 arith_uint256 result = value;
211 // Avoid division by 0, in case range + 1 results in overflow.
212 if (range != ~arith_uint256(0)) {
213 const arith_uint256 quotient = value / (range + 1);
214 result = value - (quotient * (range + 1));
215 }
216 result += min;
217 assert(result >= min && result <= max);
218 return result;
219}
220
221[[nodiscard]] std::map<COutPoint, Coin> ConsumeCoins(FuzzedDataProvider& fuzzed_data_provider) noexcept;
222
224
225[[nodiscard]] CKey ConsumePrivateKey(FuzzedDataProvider& fuzzed_data_provider, std::optional<bool> compressed = std::nullopt) noexcept;
226
227template <typename T>
228[[nodiscard]] bool MultiplicationOverflow(const T i, const T j) noexcept
229{
230 static_assert(std::is_integral_v<T>, "Integral required.");
231 if (std::numeric_limits<T>::is_signed) {
232 if (i > 0) {
233 if (j > 0) {
234 return i > (std::numeric_limits<T>::max() / j);
235 } else {
236 return j < (std::numeric_limits<T>::min() / i);
237 }
238 } else {
239 if (j > 0) {
240 return i < (std::numeric_limits<T>::min() / j);
241 } else {
242 return i != 0 && (j < (std::numeric_limits<T>::max() / i));
243 }
244 }
245 } else {
246 return j != 0 && i > std::numeric_limits<T>::max() / j;
247 }
248}
249
250[[nodiscard]] bool ContainsSpentInput(const CTransaction& tx, const CCoinsViewCache& inputs) noexcept;
251
255template <typename T, size_t size>
256void SetFuzzedErrNo(FuzzedDataProvider& fuzzed_data_provider, const std::array<T, size>& errnos)
257{
259}
260
261/*
262 * Sets a fuzzed errno in the range [0, 133 (EHWPOISON)]. Can be used from functions emulating
263 * standard library functions that set errno, or in other contexts where the value of errno
264 * might be relevant for the execution path that will be taken.
265 */
267{
268 errno = fuzzed_data_provider.ConsumeIntegralInRange<int>(0, 133);
269}
270
275template<typename B = uint8_t>
276[[nodiscard]] inline std::vector<B> ConsumeFixedLengthByteVector(FuzzedDataProvider& fuzzed_data_provider, const size_t length) noexcept
277{
278 static_assert(sizeof(B) == 1);
279 auto random_bytes = fuzzed_data_provider.ConsumeBytes<B>(length);
280 random_bytes.resize(length);
281 return random_bytes;
282}
283
285{
287 int64_t m_offset = 0;
288
289public:
291 {
292 }
293
294 FILE* open();
295
296 static ssize_t read(void* cookie, char* buf, size_t size);
297
298 static ssize_t write(void* cookie, const char* buf, size_t size);
299
300 static int seek(void* cookie, int64_t* offset, int whence);
301
302 static int close(void* cookie);
303};
304
305#define WRITE_TO_STREAM_CASE(type, consume) \
306 [&] { \
307 type o = consume; \
308 stream << o; \
309 }
310template <typename Stream>
312{
314 try {
315 CallOneOf(
327 WRITE_TO_STREAM_CASE(std::vector<uint8_t>, ConsumeRandomLengthIntegralVector<uint8_t>(fuzzed_data_provider)));
328 } catch (const std::ios_base::failure&) {
329 break;
330 }
331 }
332}
333
334#define READ_FROM_STREAM_CASE(type) \
335 [&] { \
336 type o; \
337 stream >> o; \
338 }
339template <typename Stream>
341{
343 try {
344 CallOneOf(
347 READ_FROM_STREAM_CASE(int8_t),
348 READ_FROM_STREAM_CASE(uint8_t),
349 READ_FROM_STREAM_CASE(int16_t),
350 READ_FROM_STREAM_CASE(uint16_t),
351 READ_FROM_STREAM_CASE(int32_t),
352 READ_FROM_STREAM_CASE(uint32_t),
353 READ_FROM_STREAM_CASE(int64_t),
354 READ_FROM_STREAM_CASE(uint64_t),
355 READ_FROM_STREAM_CASE(std::string),
356 READ_FROM_STREAM_CASE(std::vector<uint8_t>));
357 } catch (const std::ios_base::failure&) {
358 break;
359 }
360 }
361}
362
363inline void FinalizeHeader(CBlockHeader& header, const ChainstateManager& chainman)
364{
365 while (!CheckProofOfWork(header.GetHash(), header.nBits, chainman.GetParams().GetConsensus())) {
366 ++(header.nNonce);
367 }
368}
369
370#endif // BITCOIN_TEST_FUZZ_UTIL_H
std::variant< CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown > CTxDestination
A txout script categorized into standard templates.
Definition: addresstype.h:143
int64_t CAmount
Amount in satoshis (Can be negative)
Definition: amount.h:12
arith_uint256 UintToArith256(const uint256 &a)
int ret
Nodes collect new transactions into a block, hash them into a hash tree, and scan through nonce value...
Definition: block.h:27
uint32_t nNonce
Definition: block.h:35
uint32_t nBits
Definition: block.h:34
uint256 GetHash() const
Definition: block.cpp:14
const Consensus::Params & GetConsensus() const
Definition: chainparams.h:89
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Definition: coins.h:394
An encapsulated private key.
Definition: key.h:36
Serialized script, used inside transaction inputs and outputs.
Definition: script.h:405
The basic transaction that is broadcasted on the network and contained in blocks.
Definition: transaction.h:281
Interface for managing multiple Chainstate objects, where each chainstate is associated with chainsta...
Definition: validation.h:940
const CChainParams & GetParams() const
Definition: validation.h:1007
Double ended buffer combining vector and stream-like interfaces.
Definition: streams.h:165
std::string ConsumeRandomLengthString(size_t max_length)
std::vector< T > ConsumeBytes(size_t num_bytes)
T ConsumeIntegralInRange(T min, T max)
T PickValueInArray(const T(&array)[size])
static ssize_t write(void *cookie, const char *buf, size_t size)
Definition: util.cpp:311
FuzzedDataProvider & m_fuzzed_data_provider
Definition: util.h:286
int64_t m_offset
Definition: util.h:287
static int seek(void *cookie, int64_t *offset, int whence)
Definition: util.cpp:323
static int close(void *cookie)
Definition: util.cpp:351
FuzzedFileProvider(FuzzedDataProvider &fuzzed_data_provider)
Definition: util.h:290
static ssize_t read(void *cookie, char *buf, size_t size)
Definition: util.cpp:292
FILE * open()
Definition: util.cpp:251
Minimal stream for reading from an existing byte array by std::span.
Definition: streams.h:83
256-bit unsigned big integer.
160-bit opaque blob.
Definition: uint256.h:184
256-bit opaque blob.
Definition: uint256.h:196
bool CheckProofOfWork(uint256 hash, unsigned int nBits, const Consensus::Params &params)
Check whether a block hash satisfies the proof-of-work requirement specified by nBits.
Definition: pow.cpp:140
static const unsigned int MAX_OPCODE
Definition: script.h:216
opcodetype
Script opcodes.
Definition: script.h:74
constexpr deserialize_type deserialize
Definition: serialize.h:51
A mutable version of CTransaction.
Definition: transaction.h:358
CScriptNum ConsumeScriptNum(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:177
WeakEnumType ConsumeWeakEnum(FuzzedDataProvider &fuzzed_data_provider, const WeakEnumType(&all_types)[size]) noexcept
Definition: util.h:143
uint32_t ConsumeSequence(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.cpp:155
DataStream ConsumeDataStream(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
Definition: util.h:70
NodeSeconds ConsumeTime(FuzzedDataProvider &fuzzed_data_provider, const std::optional< int64_t > &min=std::nullopt, const std::optional< int64_t > &max=std::nullopt) noexcept
Definition: util.cpp:34
auto & PickValue(FuzzedDataProvider &fuzzed_data_provider, Collection &col)
Definition: util.h:49
void ReadFromStream(FuzzedDataProvider &fuzzed_data_provider, Stream &stream) noexcept
Definition: util.h:340
std::optional< T > ConsumeDeserializableConstructor(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
Definition: util.h:130
P ConsumeDeserializationParams(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: net.cpp:94
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:276
bool ContainsSpentInput(const CTransaction &tx, const CCoinsViewCache &inputs) noexcept
Definition: util.cpp:240
uint256 ConsumeUInt256(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:191
#define READ_FROM_STREAM_CASE(type)
Definition: util.h:334
arith_uint256 ConsumeArithUInt256(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:200
size_t CallOneOf(FuzzedDataProvider &fuzzed_data_provider, Callables... callables)
Definition: util.h:37
std::vector< B > ConsumeRandomLengthByteVector(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
Definition: util.h:59
std::vector< T > ConsumeRandomLengthIntegralVector(FuzzedDataProvider &fuzzed_data_provider, const size_t max_vector_size=16) noexcept
Definition: util.h:87
CScriptWitness ConsumeScriptWitness(FuzzedDataProvider &fuzzed_data_provider, size_t max_stack_elem_size=32) noexcept
Definition: util.cpp:83
void WriteToStream(FuzzedDataProvider &fuzzed_data_provider, Stream &stream) noexcept
Definition: util.h:311
CScript ConsumeScript(FuzzedDataProvider &fuzzed_data_provider, bool maybe_p2wsh=false) noexcept
Definition: util.cpp:93
std::optional< T > ConsumeDeserializable(FuzzedDataProvider &fuzzed_data_provider, const P &params, const std::optional< size_t > &max_length=std::nullopt) noexcept
Definition: util.h:102
std::vector< std::string > ConsumeRandomLengthStringVector(FuzzedDataProvider &fuzzed_data_provider, const size_t max_vector_size=16, const size_t max_string_length=16) noexcept
Definition: util.h:75
CMutableTransaction ConsumeTransaction(FuzzedDataProvider &fuzzed_data_provider, const std::optional< std::vector< Txid > > &prevout_txids, int max_num_in=10, int max_num_out=10) noexcept
Definition: util.cpp:42
CKey ConsumePrivateKey(FuzzedDataProvider &fuzzed_data_provider, std::optional< bool > compressed=std::nullopt) noexcept
Definition: util.cpp:230
bool MultiplicationOverflow(const T i, const T j) noexcept
Definition: util.h:228
Dur ConsumeDuration(FuzzedDataProvider &fuzzed_data_provider, std::common_type_t< Dur > min, std::common_type_t< Dur > max) noexcept
Definition: util.h:164
opcodetype ConsumeOpcodeType(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:150
void FinalizeHeader(CBlockHeader &header, const ChainstateManager &chainman)
Definition: util.h:363
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:256
uint160 ConsumeUInt160(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.h:182
#define WRITE_TO_STREAM_CASE(type, consume)
Definition: util.h:305
CTxDestination ConsumeTxDestination(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.cpp:184
arith_uint256 ConsumeArithUInt256InRange(FuzzedDataProvider &fuzzed_data_provider, const arith_uint256 &min, const arith_uint256 &max) noexcept
Definition: util.h:205
std::map< COutPoint, Coin > ConsumeCoins(FuzzedDataProvider &fuzzed_data_provider) noexcept
Definition: util.cpp:166
CAmount ConsumeMoney(FuzzedDataProvider &fuzzed_data_provider, const std::optional< CAmount > &max=std::nullopt) noexcept
Definition: util.cpp:29
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds
Definition: time.h:35
#define B
Definition: util_tests.cpp:563
assert(!tx.IsCoinBase())
FuzzedDataProvider & fuzzed_data_provider
Definition: fees.cpp:39