Branch data Line data Source code
1 : : // Copyright (c) 2022 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 : : #include <random.h>
6 : : #include <span.h>
7 : : #include <support/allocators/pool.h>
8 : : #include <test/fuzz/FuzzedDataProvider.h>
9 : : #include <test/fuzz/fuzz.h>
10 : : #include <test/fuzz/util.h>
11 : : #include <test/util/poolresourcetester.h>
12 : :
13 : : #include <cstdint>
14 : : #include <tuple>
15 : : #include <vector>
16 : :
17 : : namespace {
18 : :
19 : : template <std::size_t MAX_BLOCK_SIZE_BYTES, std::size_t ALIGN_BYTES>
20 : 753 : class PoolResourceFuzzer
21 : : {
22 : : FuzzedDataProvider& m_provider;
23 : : PoolResource<MAX_BLOCK_SIZE_BYTES, ALIGN_BYTES> m_test_resource;
24 : : uint64_t m_sequence{0};
25 : : size_t m_total_allocated{};
26 : :
27 : : struct Entry {
28 : : Span<std::byte> span;
29 : : size_t alignment;
30 : : uint64_t seed;
31 : :
32 : 498290 : Entry(Span<std::byte> s, size_t a, uint64_t se) : span(s), alignment(a), seed(se) {}
33 : : };
34 : :
35 : : std::vector<Entry> m_entries;
36 : :
37 : : public:
38 : 753 : PoolResourceFuzzer(FuzzedDataProvider& provider)
39 : 753 : : m_provider{provider},
40 : 753 : m_test_resource{provider.ConsumeIntegralInRange<size_t>(MAX_BLOCK_SIZE_BYTES, 262144)}
41 : : {
42 : 753 : }
43 : :
44 : 498290 : void Allocate(size_t size, size_t alignment)
45 : : {
46 [ - + ]: 498290 : assert(size > 0); // Must allocate at least 1 byte.
47 [ - + ]: 498290 : assert(alignment > 0); // Alignment must be at least 1.
48 [ - + ]: 498290 : assert((alignment & (alignment - 1)) == 0); // Alignment must be power of 2.
49 [ - + ]: 498290 : assert((size & (alignment - 1)) == 0); // Size must be a multiple of alignment.
50 : :
51 [ - + ]: 498290 : auto span = Span(static_cast<std::byte*>(m_test_resource.Allocate(size, alignment)), size);
52 : 498290 : m_total_allocated += size;
53 : :
54 : 498290 : auto ptr_val = reinterpret_cast<std::uintptr_t>(span.data());
55 [ - + ]: 498290 : assert((ptr_val & (alignment - 1)) == 0);
56 : :
57 : 498290 : uint64_t seed = m_sequence++;
58 : 498290 : RandomContentFill(m_entries.emplace_back(span, alignment, seed));
59 : 498290 : }
60 : :
61 : : void
62 : 500537 : Allocate()
63 : : {
64 [ + + ]: 500537 : if (m_total_allocated > 0x1000000) return;
65 : 498290 : size_t alignment_bits = m_provider.ConsumeIntegralInRange<size_t>(0, 7);
66 : 498290 : size_t alignment = size_t{1} << alignment_bits;
67 : 498290 : size_t size_bits = m_provider.ConsumeIntegralInRange<size_t>(0, 16 - alignment_bits);
68 : 498290 : size_t size = m_provider.ConsumeIntegralInRange<size_t>(size_t{1} << size_bits, (size_t{1} << (size_bits + 1)) - 1U) << alignment_bits;
69 : 498290 : Allocate(size, alignment);
70 : : }
71 : :
72 : 498290 : void RandomContentFill(Entry& entry)
73 : : {
74 : 498290 : InsecureRandomContext(entry.seed).fillrand(entry.span);
75 : 498290 : }
76 : :
77 : 498290 : void RandomContentCheck(const Entry& entry)
78 : : {
79 : 498290 : std::vector<std::byte> expect(entry.span.size());
80 : 498290 : InsecureRandomContext(entry.seed).fillrand(expect);
81 [ - + ]: 498290 : assert(std::ranges::equal(entry.span, expect));
82 : 498290 : }
83 : :
84 : 498290 : void Deallocate(const Entry& entry)
85 : : {
86 [ - + ]: 498290 : auto ptr_val = reinterpret_cast<std::uintptr_t>(entry.span.data());
87 [ - + ]: 498290 : assert((ptr_val & (entry.alignment - 1)) == 0);
88 : 498290 : RandomContentCheck(entry);
89 : 498290 : m_total_allocated -= entry.span.size();
90 : 498290 : m_test_resource.Deallocate(entry.span.data(), entry.span.size(), entry.alignment);
91 : 498290 : }
92 : :
93 : 516767 : void Deallocate()
94 : : {
95 [ + + ]: 516767 : if (m_entries.empty()) {
96 : : return;
97 : : }
98 : :
99 : 498290 : size_t idx = m_provider.ConsumeIntegralInRange<size_t>(0, m_entries.size() - 1);
100 : 498290 : Deallocate(m_entries[idx]);
101 [ + + ]: 498290 : if (idx != m_entries.size() - 1) {
102 : 494362 : m_entries[idx] = std::move(m_entries.back());
103 : : }
104 : 498290 : m_entries.pop_back();
105 : : }
106 : :
107 : 753 : void Clear()
108 : : {
109 [ + + ]: 488514 : while (!m_entries.empty()) {
110 : 487761 : Deallocate();
111 : : }
112 : :
113 : 753 : PoolResourceTester::CheckAllDataAccountedFor(m_test_resource);
114 : 753 : }
115 : :
116 : 753 : void Fuzz()
117 : : {
118 [ + + + + ]: 530296 : LIMITED_WHILE(m_provider.ConsumeBool(), 10000)
119 : : {
120 : 529543 : CallOneOf(
121 : : m_provider,
122 : 500537 : [&] { Allocate(); },
123 : 29006 : [&] { Deallocate(); });
124 : : }
125 : 753 : Clear();
126 : 753 : }
127 : : };
128 : :
129 : :
130 : : } // namespace
131 : :
132 [ + - ]: 1167 : FUZZ_TARGET(pool_resource)
133 : : {
134 : 753 : FuzzedDataProvider provider(buffer.data(), buffer.size());
135 : 753 : CallOneOf(
136 : : provider,
137 [ + - ]: 194 : [&] { PoolResourceFuzzer<128, 1>{provider}.Fuzz(); },
138 [ + - ]: 162 : [&] { PoolResourceFuzzer<128, 2>{provider}.Fuzz(); },
139 [ + - ]: 176 : [&] { PoolResourceFuzzer<128, 4>{provider}.Fuzz(); },
140 [ + - ]: 192 : [&] { PoolResourceFuzzer<128, 8>{provider}.Fuzz(); },
141 : :
142 [ + - ]: 172 : [&] { PoolResourceFuzzer<8, 8>{provider}.Fuzz(); },
143 [ + - ]: 232 : [&] { PoolResourceFuzzer<16, 16>{provider}.Fuzz(); },
144 : :
145 [ + - ]: 190 : [&] { PoolResourceFuzzer<256, alignof(max_align_t)>{provider}.Fuzz(); },
146 [ + - ]: 188 : [&] { PoolResourceFuzzer<256, 64>{provider}.Fuzz(); });
147 : 753 : }
|