Branch data Line data Source code
1 : : #include <arith_uint256.h>
2 : : #include <chain.h>
3 : : #include <chainparams.h>
4 : : #include <headerssync.h>
5 : : #include <test/fuzz/fuzz.h>
6 : : #include <test/fuzz/util.h>
7 : : #include <test/util/setup_common.h>
8 : : #include <uint256.h>
9 : : #include <util/chaintype.h>
10 : : #include <util/time.h>
11 : : #include <validation.h>
12 : :
13 : : #include <iterator>
14 : : #include <vector>
15 : :
16 : 1 : static void initialize_headers_sync_state_fuzz()
17 : : {
18 : 1 : static const auto testing_setup = MakeNoLogFileContext<>(
19 [ + - + - ]: 2 : /*chain_type=*/ChainType::MAIN);
20 [ + - ]: 2 : }
21 : :
22 : 202616 : void MakeHeadersContinuous(
23 : : const CBlockHeader& genesis_header,
24 : : const std::vector<CBlockHeader>& all_headers,
25 : : std::vector<CBlockHeader>& new_headers)
26 : : {
27 : 202616 : Assume(!new_headers.empty());
28 : :
29 : 202616 : const CBlockHeader* prev_header{
30 [ + + ]: 202616 : all_headers.empty() ? &genesis_header : &all_headers.back()};
31 : :
32 [ + + ]: 417548 : for (auto& header : new_headers) {
33 : 214932 : header.hashPrevBlock = prev_header->GetHash();
34 : :
35 : 214932 : prev_header = &header;
36 : : }
37 : 202616 : }
38 : :
39 : 0 : class FuzzedHeadersSyncState : public HeadersSyncState
40 : : {
41 : : public:
42 : 261 : FuzzedHeadersSyncState(const unsigned commit_offset, const CBlockIndex* chain_start, const arith_uint256& minimum_required_work)
43 : 261 : : HeadersSyncState(/*id=*/0, Params().GetConsensus(), chain_start, minimum_required_work)
44 : : {
45 : 261 : const_cast<unsigned&>(m_commit_offset) = commit_offset;
46 : 261 : }
47 : : };
48 : :
49 [ + - ]: 678 : FUZZ_TARGET(headers_sync_state, .init = initialize_headers_sync_state_fuzz)
50 : : {
51 : 266 : FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size());
52 : 266 : auto mock_time{ConsumeTime(fuzzed_data_provider)};
53 : :
54 : 266 : CBlockHeader genesis_header{Params().GenesisBlock()};
55 : 266 : CBlockIndex start_index(genesis_header);
56 : :
57 [ + + ]: 266 : if (mock_time < start_index.GetMedianTimePast()) return;
58 : 261 : SetMockTime(mock_time);
59 : :
60 : 261 : const uint256 genesis_hash = genesis_header.GetHash();
61 : 261 : start_index.phashBlock = &genesis_hash;
62 : :
63 : 261 : arith_uint256 min_work{UintToArith256(ConsumeUInt256(fuzzed_data_provider))};
64 : 261 : FuzzedHeadersSyncState headers_sync(
65 : : /*commit_offset=*/fuzzed_data_provider.ConsumeIntegralInRange<unsigned>(1, 1024),
66 : : /*chain_start=*/&start_index,
67 : 261 : /*minimum_required_work=*/min_work);
68 : :
69 : : // Store headers for potential redownload phase.
70 : 261 : std::vector<CBlockHeader> all_headers;
71 : 261 : std::vector<CBlockHeader>::const_iterator redownloaded_it;
72 : 261 : bool presync{true};
73 : 261 : bool requested_more{true};
74 : :
75 [ + + ]: 203121 : while (requested_more) {
76 : 202964 : std::vector<CBlockHeader> headers;
77 : :
78 : : // Consume headers from fuzzer or maybe replay headers if we got to the
79 : : // redownload phase.
80 [ + + + + ]: 203205 : if (presync || fuzzed_data_provider.ConsumeBool()) {
81 : 202759 : auto deser_headers = ConsumeDeserializable<std::vector<CBlockHeader>>(fuzzed_data_provider);
82 [ + + + + ]: 202759 : if (!deser_headers || deser_headers->empty()) return;
83 : :
84 [ + + ]: 202655 : if (fuzzed_data_provider.ConsumeBool()) {
85 [ + - ]: 202616 : MakeHeadersContinuous(genesis_header, all_headers, *deser_headers);
86 : : }
87 : :
88 : 202655 : headers.swap(*deser_headers);
89 [ + - ]: 202964 : } else if (auto num_headers_left{std::distance(redownloaded_it, all_headers.cend())}; num_headers_left > 0) {
90 : : // Consume some headers from the redownload buffer (At least one
91 : : // header is consumed).
92 : 205 : auto begin_it{redownloaded_it};
93 : 205 : std::advance(redownloaded_it, fuzzed_data_provider.ConsumeIntegralInRange<int>(1, num_headers_left));
94 [ + - ]: 205 : headers.insert(headers.cend(), begin_it, redownloaded_it);
95 : : }
96 : :
97 [ + - ]: 202860 : if (headers.empty()) return;
98 [ + - ]: 202860 : auto result = headers_sync.ProcessNextHeaders(headers, fuzzed_data_provider.ConsumeBool());
99 : 202860 : requested_more = result.request_more;
100 : :
101 [ + + ]: 202860 : if (result.request_more) {
102 [ + + ]: 202703 : if (presync) {
103 [ + - ]: 202549 : all_headers.insert(all_headers.cend(), headers.cbegin(), headers.cend());
104 : :
105 [ + + ]: 202549 : if (headers_sync.GetState() == HeadersSyncState::State::REDOWNLOAD) {
106 : 87 : presync = false;
107 [ + - ]: 87 : redownloaded_it = all_headers.cbegin();
108 : :
109 : : // If we get to redownloading, the presynced headers need
110 : : // to have the min amount of work on them.
111 [ + - - + ]: 174 : assert(CalculateClaimedHeadersWork(all_headers) >= min_work);
112 : : }
113 : : }
114 : :
115 [ + - ]: 405406 : (void)headers_sync.NextHeadersRequestLocator();
116 : : }
117 : 202964 : }
118 : 522 : }
|