5#ifndef BITCOIN_TEST_FUZZ_UTIL_H
6#define BITCOIN_TEST_FUZZ_UTIL_H
39template <
typename... Callables>
42 constexpr size_t call_size{
sizeof...(callables)};
43 static_assert(call_size >= 1);
47 ((i++ == call_index ? callables() : void()), ...);
51template <
typename Collection>
54 const auto sz = col.size();
56 auto it = col.begin();
63 const std::string s = max_length ?
64 fuzzed_data_provider.ConsumeRandomLengthString(*max_length) :
65 fuzzed_data_provider.ConsumeRandomLengthString();
66 return {s.begin(), s.end()};
82 std::vector<std::string> r;
83 for (
size_t i = 0; i < n_elements; ++i) {
94 for (
size_t i = 0; i < n_elements; ++i) {
108 }
catch (
const std::ios_base::failure&) {
114template <
typename WeakEnumType,
size_t size>
119 WeakEnumType(fuzzed_data_provider.
ConsumeIntegral<
typename std::underlying_type<WeakEnumType>::type>());
129[[nodiscard]] int64_t
ConsumeTime(
FuzzedDataProvider& fuzzed_data_provider,
const std::optional<int64_t>& min = std::nullopt,
const std::optional<int64_t>& max = std::nullopt)
noexcept;
146 const std::vector<uint8_t> v160 = fuzzed_data_provider.
ConsumeBytes<uint8_t>(160 / 8);
147 if (v160.size() != 160 / 8) {
155 const std::vector<uint8_t> v256 = fuzzed_data_provider.
ConsumeBytes<uint8_t>(256 / 8);
156 if (v256.size() != 256 / 8) {
174 static_assert(std::is_integral<T>::value,
"Integral required.");
175 if (std::numeric_limits<T>::is_signed) {
178 return i > (std::numeric_limits<T>::max() / j);
180 return j < (std::numeric_limits<T>::min() / i);
184 return i < (std::numeric_limits<T>::min() / j);
186 return i != 0 && (j < (std::numeric_limits<T>::max() / i));
190 return j != 0 && i > std::numeric_limits<T>::max() / j;
197 static_assert(std::is_integral<T>::value,
"Integral required.");
198 if (std::numeric_limits<T>::is_signed) {
199 return (i > 0 && j > std::numeric_limits<T>::max() - i) ||
200 (i < 0 && j < std::numeric_limits<T>::min() - i);
202 return std::numeric_limits<T>::max() - i < j;
210template <
typename T,
size_t size>
232 std::vector<uint8_t> result(length);
233 const std::vector<uint8_t> random_bytes = fuzzed_data_provider.
ConsumeBytes<uint8_t>(length);
234 if (!random_bytes.empty()) {
235 std::memcpy(result.data(), random_bytes.data(), random_bytes.size());
257template <
bool ReturnUniquePtr = false>
264 const uint64_t keyed_net_group = fuzzed_data_provider.
ConsumeIntegral<uint64_t>();
265 const uint64_t local_host_nonce = fuzzed_data_provider.
ConsumeIntegral<uint64_t>();
270 if constexpr (ReturnUniquePtr) {
271 return std::make_unique<CNode>(node_id, local_services, socket, address, keyed_net_group, local_host_nonce, addr_bind, addr_name, conn_type, inbound_onion);
273 return CNode{node_id, local_services, socket, address, keyed_net_group, local_host_nonce, addr_bind, addr_name, conn_type, inbound_onion};
292 static ssize_t
read(
void* cookie,
char* buf,
size_t size);
294 static ssize_t
write(
void* cookie,
const char* buf,
size_t size);
296 static int seek(
void* cookie, int64_t* offset,
int whence);
298 static int close(
void* cookie);
303 return {fuzzed_data_provider};
324 return {fuzzed_data_provider};
327#define WRITE_TO_STREAM_CASE(type, consume) \
332template <
typename Stream>
338 fuzzed_data_provider,
350 WRITE_TO_STREAM_CASE(std::vector<char>, ConsumeRandomLengthIntegralVector<char>(fuzzed_data_provider)));
351 }
catch (
const std::ios_base::failure&) {
357#define READ_FROM_STREAM_CASE(type) \
362template <
typename Stream>
368 fuzzed_data_provider,
381 }
catch (
const std::ios_base::failure&) {
405 void Reset()
override;
407 ssize_t
Send(
const void* data,
size_t len,
int flags)
const override;
409 ssize_t
Recv(
void* buf,
size_t len,
int flags)
const override;
411 int Connect(
const sockaddr*, socklen_t)
const override;
413 int GetSockOpt(
int level,
int opt_name,
void* opt_val, socklen_t* opt_len)
const override;
415 bool Wait(std::chrono::milliseconds timeout,
Event requested,
Event* occurred =
nullptr)
const override;
417 bool IsConnected(std::string& errmsg)
const override;
int64_t CAmount
Amount in satoshis (Can be negative)
arith_uint256 UintToArith256(const uint256 &a)
A CService with information about it as peer.
Non-refcounted RAII wrapper for FILE*.
CCoinsView that adds a memory cache for transactions to another CCoinsView.
Double ended buffer combining vector and stream-like interfaces.
Information about a peer.
Serialized script, used inside transaction inputs and outputs.
A combination of a network address (CNetAddr) and a (TCP) port.
The basic transaction that is broadcasted on the network and contained in blocks.
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
FuzzedFileProvider m_fuzzed_file_provider
FuzzedDataProvider & m_fuzzed_data_provider
FuzzedAutoFileProvider(FuzzedDataProvider &fuzzed_data_provider)
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)
FuzzedDataProvider & m_fuzzed_data_provider
static int seek(void *cookie, int64_t *offset, int whence)
static int close(void *cookie)
FuzzedFileProvider(FuzzedDataProvider &fuzzed_data_provider)
static ssize_t read(void *cookie, char *buf, size_t size)
int GetSockOpt(int level, int opt_name, void *opt_val, socklen_t *opt_len) const override
getsockopt(2) wrapper.
bool Wait(std::chrono::milliseconds timeout, Event requested, Event *occurred=nullptr) const override
Wait for readiness for input (recv) or output (send).
bool IsConnected(std::string &errmsg) const override
Check if still connected.
int Connect(const sockaddr *, socklen_t) const override
connect(2) wrapper.
ssize_t Send(const void *data, size_t len, int flags) const override
send(2) wrapper.
FuzzedSock & operator=(Sock &&other) override
Move assignment operator, grab the socket from another object and close ours (if set).
std::optional< uint8_t > m_peek_data
Data to return when MSG_PEEK is used as a Recv() flag.
FuzzedDataProvider & m_fuzzed_data_provider
void Reset() override
Close if non-empty.
ssize_t Recv(void *buf, size_t len, int flags) const override
recv(2) wrapper.
FuzzedSock(FuzzedDataProvider &fuzzed_data_provider)
RAII helper class that manages a socket.
256-bit unsigned big integer.
#define T(expected, seed, data)
std::vector< bool > BytesToBits(const std::vector< unsigned char > &bytes)
ConnectionType
Different types of connections to a peer.
@ INBOUND
Inbound connections are those initiated by a peer.
ServiceFlags
nServices flags
static const unsigned int MAX_OPCODE
opcodetype
Script opcodes.
std::variant< CNoDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, WitnessUnknown > CTxDestination
A txout script template with a specific destination.
A mutable version of CTransaction.
CScriptNum ConsumeScriptNum(FuzzedDataProvider &fuzzed_data_provider) noexcept
CService ConsumeService(FuzzedDataProvider &fuzzed_data_provider) noexcept
WeakEnumType ConsumeWeakEnum(FuzzedDataProvider &fuzzed_data_provider, const WeakEnumType(&all_types)[size]) noexcept
CSubNet ConsumeSubNet(FuzzedDataProvider &fuzzed_data_provider) noexcept
uint32_t ConsumeSequence(FuzzedDataProvider &fuzzed_data_provider) noexcept
auto & PickValue(FuzzedDataProvider &fuzzed_data_provider, Collection &col)
std::unique_ptr< CNode > ConsumeNodeAsUniquePtr(FuzzedDataProvider &fdp, const std::optional< NodeId > &node_id_in=std::nullopt)
void ReadFromStream(FuzzedDataProvider &fuzzed_data_provider, Stream &stream) noexcept
std::vector< bool > ConsumeRandomLengthBitVector(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
CScriptWitness ConsumeScriptWitness(FuzzedDataProvider &fuzzed_data_provider, const size_t max_stack_elem_size=32) noexcept
CDataStream ConsumeDataStream(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
FuzzedAutoFileProvider ConsumeAutoFile(FuzzedDataProvider &fuzzed_data_provider) noexcept
bool ContainsSpentInput(const CTransaction &tx, const CCoinsViewCache &inputs) noexcept
uint256 ConsumeUInt256(FuzzedDataProvider &fuzzed_data_provider) noexcept
#define READ_FROM_STREAM_CASE(type)
arith_uint256 ConsumeArithUInt256(FuzzedDataProvider &fuzzed_data_provider) noexcept
size_t CallOneOf(FuzzedDataProvider &fuzzed_data_provider, Callables... callables)
std::vector< T > ConsumeRandomLengthIntegralVector(FuzzedDataProvider &fuzzed_data_provider, const size_t max_vector_size=16) noexcept
void WriteToStream(FuzzedDataProvider &fuzzed_data_provider, Stream &stream) noexcept
CScript ConsumeScript(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt, const bool maybe_p2wsh=false) noexcept
int64_t ConsumeTime(FuzzedDataProvider &fuzzed_data_provider, const std::optional< int64_t > &min=std::nullopt, const std::optional< int64_t > &max=std::nullopt) noexcept
CMutableTransaction ConsumeTransaction(FuzzedDataProvider &fuzzed_data_provider, const std::optional< std::vector< uint256 > > &prevout_txids, const int max_num_in=10, const int max_num_out=10) noexcept
bool AdditionOverflow(const T i, const T j) noexcept
auto ConsumeNode(FuzzedDataProvider &fuzzed_data_provider, const std::optional< NodeId > &node_id_in=std::nullopt) noexcept
CNetAddr ConsumeNetAddr(FuzzedDataProvider &fuzzed_data_provider) noexcept
std::vector< std::string > ConsumeRandomLengthStringVector(FuzzedDataProvider &fuzzed_data_provider, const size_t max_vector_size=16, const size_t max_string_length=16) noexcept
std::optional< T > ConsumeDeserializable(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
bool MultiplicationOverflow(const T i, const T j) noexcept
opcodetype ConsumeOpcodeType(FuzzedDataProvider &fuzzed_data_provider) noexcept
CAddress ConsumeAddress(FuzzedDataProvider &fuzzed_data_provider) noexcept
void SetFuzzedErrNo(FuzzedDataProvider &fuzzed_data_provider, const std::array< T, size > &errnos)
Sets errno to a value selected from the given std::array errnos.
void FillNode(FuzzedDataProvider &fuzzed_data_provider, CNode &node, bool init_version) noexcept
uint160 ConsumeUInt160(FuzzedDataProvider &fuzzed_data_provider) noexcept
std::vector< uint8_t > 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...
#define WRITE_TO_STREAM_CASE(type, consume)
CTxDestination ConsumeTxDestination(FuzzedDataProvider &fuzzed_data_provider) noexcept
CAmount ConsumeMoney(FuzzedDataProvider &fuzzed_data_provider, const std::optional< CAmount > &max=std::nullopt) noexcept
CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider &fuzzed_data_provider, const CTransaction &tx) noexcept
FuzzedFileProvider ConsumeFile(FuzzedDataProvider &fuzzed_data_provider) noexcept
FuzzedSock ConsumeSock(FuzzedDataProvider &fuzzed_data_provider)
std::vector< uint8_t > ConsumeRandomLengthByteVector(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
constexpr ServiceFlags ALL_SERVICE_FLAGS[]
constexpr ConnectionType ALL_CONNECTION_TYPES[]
static const int INIT_PROTO_VERSION
initial proto version, to be increased after version/verack negotiation