LCOV - code coverage report
Current view: top level - src/test/fuzz - partially_downloaded_block.cpp (source / functions) Hit Total Coverage
Test: fuzz_coverage.info Lines: 4 71 5.6 %
Date: 2023-09-26 12:08:55 Functions: 5 9 55.6 %

          Line data    Source code
       1             : #include <blockencodings.h>
       2             : #include <consensus/merkle.h>
       3             : #include <consensus/validation.h>
       4             : #include <primitives/block.h>
       5             : #include <primitives/transaction.h>
       6             : #include <test/fuzz/FuzzedDataProvider.h>
       7             : #include <test/fuzz/fuzz.h>
       8             : #include <test/fuzz/util.h>
       9             : #include <test/fuzz/util/mempool.h>
      10             : #include <test/util/setup_common.h>
      11             : #include <test/util/txmempool.h>
      12             : #include <txmempool.h>
      13             : 
      14             : #include <cstddef>
      15             : #include <cstdint>
      16             : #include <limits>
      17           2 : #include <memory>
      18           2 : #include <optional>
      19             : #include <set>
      20             : #include <vector>
      21             : 
      22             : namespace {
      23             : const TestingSetup* g_setup;
      24             : } // namespace
      25             : 
      26           0 : void initialize_pdb()
      27             : {
      28           0 :     static const auto testing_setup = MakeNoLogFileContext<const TestingSetup>();
      29           0 :     g_setup = testing_setup.get();
      30           0 : }
      31             : 
      32           0 : PartiallyDownloadedBlock::CheckBlockFn FuzzedCheckBlock(std::optional<BlockValidationResult> result)
      33             : {
      34           0 :     return [result](const CBlock&, BlockValidationState& state, const Consensus::Params&, bool, bool) {
      35           0 :         if (result) {
      36           0 :             return state.Invalid(*result);
      37             :         }
      38             : 
      39           0 :         return true;
      40           0 :     };
      41             : }
      42             : 
      43           4 : FUZZ_TARGET(partially_downloaded_block, .init = initialize_pdb)
      44             : {
      45           0 :     FuzzedDataProvider fuzzed_data_provider{buffer.data(), buffer.size()};
      46             : 
      47           0 :     auto block{ConsumeDeserializable<CBlock>(fuzzed_data_provider)};
      48           0 :     if (!block || block->vtx.size() == 0 ||
      49           0 :         block->vtx.size() >= std::numeric_limits<uint16_t>::max()) {
      50           0 :         return;
      51             :     }
      52             : 
      53           0 :     CBlockHeaderAndShortTxIDs cmpctblock{*block};
      54             : 
      55           0 :     CTxMemPool pool{MemPoolOptionsForTest(g_setup->m_node)};
      56           0 :     PartiallyDownloadedBlock pdb{&pool};
      57             : 
      58             :     // Set of available transactions (mempool or extra_txn)
      59           0 :     std::set<uint16_t> available;
      60             :     // The coinbase is always available
      61           0 :     available.insert(0);
      62             : 
      63           0 :     std::vector<std::pair<uint256, CTransactionRef>> extra_txn;
      64           0 :     for (size_t i = 1; i < block->vtx.size(); ++i) {
      65           0 :         auto tx{block->vtx[i]};
      66             : 
      67           0 :         bool add_to_extra_txn{fuzzed_data_provider.ConsumeBool()};
      68           0 :         bool add_to_mempool{fuzzed_data_provider.ConsumeBool()};
      69             : 
      70           0 :         if (add_to_extra_txn) {
      71           0 :             extra_txn.emplace_back(tx->GetWitnessHash(), tx);
      72           0 :             available.insert(i);
      73           0 :         }
      74           2 : 
      75           0 :         if (add_to_mempool) {
      76           0 :             LOCK2(cs_main, pool.cs);
      77           0 :             pool.addUnchecked(ConsumeTxMemPoolEntry(fuzzed_data_provider, *tx));
      78           0 :             available.insert(i);
      79           0 :         }
      80           0 :     }
      81             : 
      82           0 :     auto init_status{pdb.InitData(cmpctblock, extra_txn)};
      83             : 
      84           0 :     std::vector<CTransactionRef> missing;
      85             :     // Whether we skipped a transaction that should be included in `missing`.
      86             :     // FillBlock should never return READ_STATUS_OK if that is the case.
      87           0 :     bool skipped_missing{false};
      88           0 :     for (size_t i = 0; i < cmpctblock.BlockTxCount(); i++) {
      89             :         // If init_status == READ_STATUS_OK then a available transaction in the
      90             :         // compact block (i.e. IsTxAvailable(i) == true) implies that we marked
      91             :         // that transaction as available above (i.e. available.count(i) > 0).
      92             :         // The reverse is not true, due to possible compact block short id
      93             :         // collisions (i.e. available.count(i) > 0 does not imply
      94             :         // IsTxAvailable(i) == true).
      95           0 :         if (init_status == READ_STATUS_OK) {
      96           0 :             assert(!pdb.IsTxAvailable(i) || available.count(i) > 0);
      97           0 :         }
      98             : 
      99           0 :         bool skip{fuzzed_data_provider.ConsumeBool()};
     100           0 :         if (!pdb.IsTxAvailable(i) && !skip) {
     101           0 :             missing.push_back(block->vtx[i]);
     102           0 :         }
     103             : 
     104           0 :         skipped_missing |= (!pdb.IsTxAvailable(i) && skip);
     105           0 :     }
     106             : 
     107             :     // Mock CheckBlock
     108           0 :     bool fail_check_block{fuzzed_data_provider.ConsumeBool()};
     109           0 :     auto validation_result =
     110           0 :         fuzzed_data_provider.PickValueInArray(
     111           0 :             {BlockValidationResult::BLOCK_RESULT_UNSET,
     112             :              BlockValidationResult::BLOCK_CONSENSUS,
     113             :              BlockValidationResult::BLOCK_RECENT_CONSENSUS_CHANGE,
     114             :              BlockValidationResult::BLOCK_CACHED_INVALID,
     115             :              BlockValidationResult::BLOCK_INVALID_HEADER,
     116             :              BlockValidationResult::BLOCK_MUTATED,
     117             :              BlockValidationResult::BLOCK_MISSING_PREV,
     118             :              BlockValidationResult::BLOCK_INVALID_PREV,
     119             :              BlockValidationResult::BLOCK_TIME_FUTURE,
     120             :              BlockValidationResult::BLOCK_CHECKPOINT,
     121             :              BlockValidationResult::BLOCK_HEADER_LOW_WORK});
     122           0 :     pdb.m_check_block_mock = FuzzedCheckBlock(
     123           0 :         fail_check_block ?
     124           0 :             std::optional<BlockValidationResult>{validation_result} :
     125           0 :             std::nullopt);
     126             : 
     127           0 :     CBlock reconstructed_block;
     128           0 :     auto fill_status{pdb.FillBlock(reconstructed_block, missing)};
     129           0 :     switch (fill_status) {
     130             :     case READ_STATUS_OK:
     131           0 :         assert(!skipped_missing);
     132           0 :         assert(!fail_check_block);
     133           0 :         assert(block->GetHash() == reconstructed_block.GetHash());
     134           0 :         break;
     135             :     case READ_STATUS_CHECKBLOCK_FAILED: [[fallthrough]];
     136             :     case READ_STATUS_FAILED:
     137           0 :         assert(fail_check_block);
     138           0 :         break;
     139             :     case READ_STATUS_INVALID:
     140           0 :         break;
     141             :     }
     142           0 : }

Generated by: LCOV version 1.14