19 : m_fuzzed_data_provider{fuzzed_data_provider}
35 assert(
false &&
"Move of Sock into FuzzedSock not allowed.");
41 constexpr std::array send_errnos{
76 constexpr std::array recv_errnos{
88 assert(buf !=
nullptr || len == 0);
96 std::vector<uint8_t> random_bytes;
101 if ((
flags & MSG_PEEK) == 0) {
104 pad_to_len_bytes =
false;
105 }
else if ((
flags & MSG_PEEK) != 0) {
108 if (!random_bytes.empty()) {
110 pad_to_len_bytes =
false;
116 if (random_bytes.empty()) {
123 std::memcpy(buf, random_bytes.data(), random_bytes.size());
124 if (pad_to_len_bytes) {
125 if (len > random_bytes.size()) {
126 std::memset((
char*)buf + random_bytes.size(), 0, len - random_bytes.size());
131 std::this_thread::sleep_for(std::chrono::milliseconds{2});
133 return random_bytes.size();
141 constexpr std::array connect_errnos{
165 constexpr std::array bind_errnos{
185 constexpr std::array listen_errnos{
199 constexpr std::array accept_errnos{
206 return std::unique_ptr<FuzzedSock>();
213 constexpr std::array getsockopt_errnos{
221 if (opt_val ==
nullptr) {
232 constexpr std::array setsockopt_errnos{
245 constexpr std::array getsockname_errnos{
259 constexpr std::array wait_errnos{
268 if (occurred !=
nullptr) {
276 for (
auto& [sock, events] : events_per_sock) {
288 errmsg =
"disconnected at random by the fuzzer";
294 connman.Handshake(
node,
295 fuzzed_data_provider.ConsumeBool(),
298 fuzzed_data_provider.ConsumeIntegralInRange<int32_t>(
MIN_PEER_PROTO_VERSION, std::numeric_limits<int32_t>::max()),
299 fuzzed_data_provider.ConsumeBool());
304 return fuzzed_data_provider.ConsumeIntegralInRange<
CAmount>(0, max.value_or(
MAX_MONEY));
312 return fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(min.value_or(time_min), max.value_or(time_max));
318 const auto p2wsh_op_true = fuzzed_data_provider.ConsumeBool();
319 tx_mut.
nVersion = fuzzed_data_provider.ConsumeBool() ?
321 fuzzed_data_provider.ConsumeIntegral<int32_t>();
322 tx_mut.
nLockTime = fuzzed_data_provider.ConsumeIntegral<uint32_t>();
323 const auto num_in = fuzzed_data_provider.ConsumeIntegralInRange<
int>(0, max_num_in);
324 const auto num_out = fuzzed_data_provider.ConsumeIntegralInRange<
int>(0, max_num_out);
325 for (
int i = 0; i < num_in; ++i) {
326 const auto& txid_prev = prevout_txids ?
327 PickValue(fuzzed_data_provider, *prevout_txids) :
329 const auto index_out = fuzzed_data_provider.ConsumeIntegralInRange<uint32_t>(0, max_num_out);
344 tx_mut.
vin.push_back(in);
346 for (
int i = 0; i < num_out; ++i) {
347 const auto amount = fuzzed_data_provider.ConsumeIntegralInRange<
CAmount>(-10, 50 *
COIN + 10);
348 const auto script_pk = p2wsh_op_true ?
351 tx_mut.
vout.emplace_back(amount, script_pk);
359 const auto n_elements = fuzzed_data_provider.ConsumeIntegralInRange<
size_t>(0, max_stack_elem_size);
360 for (
size_t i = 0; i < n_elements; ++i) {
372 static constexpr
unsigned MAX_BUFFER_SZ{128};
373 std::vector<uint8_t> buffer(MAX_BUFFER_SZ, uint8_t{
'a'});
374 while (fuzzed_data_provider.ConsumeBool()) {
376 fuzzed_data_provider,
379 r_script.insert(r_script.end(), buffer.begin(), buffer.begin() + fuzzed_data_provider.ConsumeIntegralInRange(0U, MAX_BUFFER_SZ));
383 r_script << std::vector<uint8_t>{buffer.begin(), buffer.begin() + fuzzed_data_provider.ConsumeIntegralInRange(0U, MAX_BUFFER_SZ)};
389 r_script << fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(0, 22);
390 int num_data{fuzzed_data_provider.ConsumeIntegralInRange(1, 22)};
392 pubkey_comp.front() = fuzzed_data_provider.ConsumeIntegralInRange(2, 3);
393 std::vector<uint8_t> pubkey_uncomp{buffer.begin(), buffer.begin() +
CPubKey::SIZE};
394 pubkey_uncomp.front() = fuzzed_data_provider.ConsumeIntegralInRange(4, 7);
396 auto& pubkey{fuzzed_data_provider.ConsumeBool() ? pubkey_uncomp : pubkey_comp};
397 if (fuzzed_data_provider.ConsumeBool()) {
398 pubkey.back() = num_data;
402 r_script << fuzzed_data_provider.ConsumeIntegralInRange<int64_t>(0, 22);
407 std::copy(vec.begin(), vec.end(), buffer.begin());
411 r_script << fuzzed_data_provider.ConsumeIntegral<int64_t>();
423 if (maybe_p2wsh && fuzzed_data_provider.ConsumeBool()) {
434 return fuzzed_data_provider.ConsumeBool() ?
435 fuzzed_data_provider.PickValueInArray({
440 fuzzed_data_provider.ConsumeIntegral<uint32_t>();
447 fuzzed_data_provider,
468 witness_unknown.
version = fuzzed_data_provider.ConsumeIntegralInRange(2, 16);
469 std::vector<uint8_t> witness_unknown_program_1{fuzzed_data_provider.ConsumeBytes<uint8_t>(40)};
470 if (witness_unknown_program_1.size() < 2) {
471 witness_unknown_program_1 = {0, 0};
473 witness_unknown.length = witness_unknown_program_1.size();
474 std::copy(witness_unknown_program_1.begin(), witness_unknown_program_1.end(), witness_unknown.program);
475 tx_destination = witness_unknown;
477 Assert(call_size == std::variant_size_v<CTxDestination>);
478 return tx_destination;
487 const CAmount fee = std::min<CAmount>(
ConsumeMoney(fuzzed_data_provider), std::numeric_limits<CAmount>::max() /
static_cast<CAmount>(100000));
489 const int64_t time = fuzzed_data_provider.ConsumeIntegral<int64_t>();
490 const unsigned int entry_height = fuzzed_data_provider.ConsumeIntegral<
unsigned int>();
491 const bool spends_coinbase = fuzzed_data_provider.ConsumeBool();
492 const unsigned int sig_op_cost = fuzzed_data_provider.ConsumeIntegralInRange<
unsigned int>(0,
MAX_BLOCK_SIGOPS_COST);
498 for (
const CTxIn& tx_in : tx.vin) {
499 const Coin& coin = inputs.AccessCoin(tx_in.
prevout);
512 in_addr v4_addr = {};
513 v4_addr.s_addr = fuzzed_data_provider.ConsumeIntegral<uint32_t>();
516 if (fuzzed_data_provider.remaining_bytes() >= 16) {
517 in6_addr v6_addr = {};
518 memcpy(v6_addr.s6_addr, fuzzed_data_provider.ConsumeBytes<uint8_t>(16).data(), 16);
519 net_addr =
CNetAddr{v6_addr, fuzzed_data_provider.ConsumeIntegral<uint32_t>()};
522 net_addr.
SetInternal(fuzzed_data_provider.ConsumeBytesAsString(32));
524 net_addr.
SetSpecial(fuzzed_data_provider.ConsumeBytesAsString(32));
561 #if defined _GNU_SOURCE && !defined __ANDROID__ 562 const cookie_io_functions_t io_hooks = {
568 return fopencookie(
this, mode.c_str(), io_hooks);
583 if (random_bytes.empty()) {
586 std::memcpy(buf, random_bytes.data(), random_bytes.size());
590 fuzzed_file->
m_offset += random_bytes.size();
591 return random_bytes.size();
608 assert(whence == SEEK_SET || whence == SEEK_CUR || whence == SEEK_END);
611 int64_t new_offset = 0;
612 if (whence == SEEK_SET) {
613 new_offset = *offset;
614 }
else if (whence == SEEK_CUR) {
618 new_offset = fuzzed_file->
m_offset + *offset;
619 }
else if (whence == SEEK_END) {
624 new_offset = n + *offset;
626 if (new_offset < 0) {
630 *offset = new_offset;
CSHA256 & Write(const unsigned char *data, size_t len)
bool IsSpent() const
Either this coin never existed (see e.g.
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...
static constexpr unsigned int SIZE
secp256k1:
bool AdditionOverflow(const T i, const T j) noexcept
FuzzedSock & operator=(Sock &&other) override
Move assignment operator, grab the socket from another object and close ours (if set).
A set of addresses that represent the hash of a string or FQDN.
static int seek(void *cookie, int64_t *offset, int whence)
WeakEnumType ConsumeWeakEnum(FuzzedDataProvider &fuzzed_data_provider, const WeakEnumType(&all_types)[size]) noexcept
CTxMemPoolEntry ConsumeTxMemPoolEntry(FuzzedDataProvider &fuzzed_data_provider, const CTransaction &tx) noexcept
CService ConsumeService(FuzzedDataProvider &fuzzed_data_provider) noexcept
static const CScript P2WSH_OP_TRUE
CScriptWitness scriptWitness
Only serialized through CTransaction.
static const uint32_t SEQUENCE_FINAL
Setting nSequence to this value for every input in a transaction disables nLockTime/IsFinalTx().
opcodetype ConsumeOpcodeType(FuzzedDataProvider &fuzzed_data_provider) noexcept
bool MoneyRange(const CAmount &nValue)
CScriptWitness ConsumeScriptWitness(FuzzedDataProvider &fuzzed_data_provider, const size_t max_stack_elem_size) noexcept
CScript ConsumeScript(FuzzedDataProvider &fuzzed_data_provider, const bool maybe_p2wsh) noexcept
SOCKET m_socket
Contained socket.
int SetSockOpt(int level, int opt_name, const void *opt_val, socklen_t opt_len) const override
setsockopt(2) wrapper.
FuzzedDataProvider & m_fuzzed_data_provider
std::vector< std::vector< unsigned char > > stack
bool SetSpecial(const std::string &addr)
Parse a Tor or I2P address and set this object to it.
bool IsConnected(std::string &errmsg) const override
Check if still connected.
ssize_t Recv(void *buf, size_t len, int flags) const override
recv(2) wrapper.
std::optional< uint8_t > m_peek_data
Data to return when MSG_PEEK is used as a Recv() flag.
static const int64_t MAX_BLOCK_SIGOPS_COST
The maximum allowed number of signature check operations in a block (network rule) ...
int64_t ParseISO8601DateTime(const std::string &str)
CTxMemPoolEntry stores data about the corresponding transaction, as well as data about all in-mempool...
ssize_t Send(const void *data, size_t len, int flags) const override
send(2) wrapper.
static int close(void *cookie)
int64_t CAmount
Amount in satoshis (Can be negative)
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds
constexpr ServiceFlags ALL_SERVICE_FLAGS[]
CAddress ConsumeAddress(FuzzedDataProvider &fuzzed_data_provider) noexcept
static constexpr unsigned int COMPRESSED_SIZE
static constexpr uint32_t MAX_BIP125_RBF_SEQUENCE
bool Wait(std::chrono::milliseconds timeout, Event requested, Event *occurred=nullptr) const override
Wait for readiness for input (recv) or output (send).
std::unordered_map< std::shared_ptr< const Sock >, Events, HashSharedPtrSock, EqualSharedPtrSock > EventsPerSock
On which socket to wait for what events in WaitMany().
An input of a transaction.
CTxDestination subtype to encode any future Witness version.
static const std::vector< uint8_t > WITNESS_STACK_ELEM_OP_TRUE
CNetAddr ConsumeNetAddr(FuzzedDataProvider &fuzzed_data_provider) noexcept
A CService with information about it as peer.
bool ContainsSpentInput(const CTransaction &tx, const CCoinsViewCache &inputs) noexcept
std::vector< uint8_t > ConsumeRandomLengthByteVector(FuzzedDataProvider &fuzzed_data_provider, const std::optional< size_t > &max_length=std::nullopt) noexcept
std::variant< CNoDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, WitnessUnknown > CTxDestination
A txout script template with a specific destination.
int GetSockOpt(int level, int opt_name, void *opt_val, socklen_t *opt_len) const override
getsockopt(2) wrapper.
std::unique_ptr< Sock > Accept(sockaddr *addr, socklen_t *addr_len) const override
accept(2) wrapper.
An outpoint - a combination of a transaction hash and an index n into its vout.
size_t ConsumeData(void *destination, size_t num_bytes)
std::vector< CTxOut > vout
int Connect(const sockaddr *, socklen_t) const override
connect(2) wrapper.
static CTransactionRef MakeTransactionRef(Tx &&txIn)
FuzzedSock(FuzzedDataProvider &fuzzed_data_provider)
std::vector< T > ConsumeBytes(size_t num_bytes)
std::vector< unsigned char > ToByteVector(const T &in)
FuzzedDataProvider & m_fuzzed_data_provider
static ssize_t write(void *cookie, const char *buf, size_t size)
int64_t ConsumeTime(FuzzedDataProvider &fuzzed_data_provider, const std::optional< int64_t > &min, const std::optional< int64_t > &max) noexcept
static ssize_t read(void *cookie, char *buf, size_t size)
Serialized script, used inside transaction inputs and outputs.
static const int MIN_PEER_PROTO_VERSION
disconnect from peers older than this proto version
bool WaitMany(std::chrono::milliseconds timeout, EventsPerSock &events_per_sock) const override
Same as Wait(), but wait on many sockets within the same timeout.
CScriptNum ConsumeScriptNum(FuzzedDataProvider &fuzzed_data_provider) noexcept
CAmount ConsumeMoney(FuzzedDataProvider &fuzzed_data_provider, const std::optional< CAmount > &max) noexcept
void FillNode(FuzzedDataProvider &fuzzed_data_provider, ConnmanTestMsg &connman, CNode &node) 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.
static constexpr CAmount MAX_MONEY
No amount larger than this (in satoshi) is valid.
A mutable version of CTransaction.
RAII helper class that manages a socket.
static const uint32_t MAX_SEQUENCE_NONFINAL
This is the maximum sequence number that enables both nLockTime and OP_CHECKLOCKTIMEVERIFY (BIP 65)...
size_t CallOneOf(FuzzedDataProvider &fuzzed_data_provider, Callables... callables)
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
auto & PickValue(FuzzedDataProvider &fuzzed_data_provider, Collection &col)
int GetSockName(sockaddr *name, socklen_t *name_len) const override
getsockname(2) wrapper.
The basic transaction that is broadcasted on the network and contained in blocks. ...
static const int32_t CURRENT_VERSION
Information about a peer.
uint160 ConsumeUInt160(FuzzedDataProvider &fuzzed_data_provider) noexcept
T ConsumeIntegralInRange(T min, T max)
CCoinsView that adds a memory cache for transactions to another CCoinsView.
bool SetInternal(const std::string &name)
Create an "internal" address that represents a name or FQDN.
uint256 ConsumeUInt256(FuzzedDataProvider &fuzzed_data_provider) noexcept
A hasher class for SHA-256.
int Bind(const sockaddr *, socklen_t) const override
bind(2) wrapper.
CTxDestination ConsumeTxDestination(FuzzedDataProvider &fuzzed_data_provider) noexcept
int Listen(int backlog) const override
listen(2) wrapper.
#define Assert(val)
Identity function.
uint32_t ConsumeSequence(FuzzedDataProvider &fuzzed_data_provider) noexcept
static constexpr CAmount COIN
The amount of satoshis in one BTC.