Branch data Line data Source code
1 : : // Copyright (c) 2015-present 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 <prevector.h>
6 : : #include <serialize.h>
7 : : #include <streams.h>
8 : : #include <test/fuzz/FuzzedDataProvider.h>
9 : : #include <test/fuzz/fuzz.h>
10 : :
11 : : #include <ranges>
12 : : #include <vector>
13 : : namespace {
14 : :
15 : : template <unsigned int N, typename T>
16 : : class prevector_tester
17 : : {
18 : : typedef std::vector<T> realtype;
19 : : realtype real_vector;
20 : : realtype real_vector_alt;
21 : :
22 : : typedef prevector<N, T> pretype;
23 : : pretype pre_vector;
24 : : pretype pre_vector_alt;
25 : :
26 : : typedef typename pretype::size_type Size;
27 : :
28 : : public:
29 : 165 : void test() const
30 : : {
31 : 165 : const pretype& const_pre_vector = pre_vector;
32 [ - + + + : 265 : assert(real_vector.size() == pre_vector.size());
- + ]
33 [ - + ]: 165 : assert(real_vector.empty() == pre_vector.empty());
34 [ + + ]: 478842 : for (Size s = 0; s < real_vector.size(); s++) {
35 [ + + - + ]: 957354 : assert(real_vector[s] == pre_vector[s]);
36 [ + + - + ]: 957354 : assert(&(pre_vector[s]) == &(pre_vector.begin()[s]));
37 [ - + ]: 478677 : assert(&(pre_vector[s]) == &*(pre_vector.begin() + s));
38 [ - + ]: 478677 : assert(&(pre_vector[s]) == &*((pre_vector.end() + s) - real_vector.size()));
39 : : }
40 : : // assert(realtype(pre_vector) == real_vector);
41 [ - + ]: 165 : assert(pretype(real_vector.begin(), real_vector.end()) == pre_vector);
42 [ + + - + ]: 330 : assert(pretype(pre_vector.begin(), pre_vector.end()) == pre_vector);
43 : 165 : size_t pos = 0;
44 [ + + + + ]: 479007 : for (const T& v : pre_vector) {
45 [ - + ]: 478677 : assert(v == real_vector[pos]);
46 : 478677 : ++pos;
47 : : }
48 [ + + ]: 478842 : for (const T& v : pre_vector | std::views::reverse) {
49 : 478677 : --pos;
50 [ - + ]: 478677 : assert(v == real_vector[pos]);
51 : : }
52 [ + + ]: 478842 : for (const T& v : const_pre_vector) {
53 [ - + ]: 478677 : assert(v == real_vector[pos]);
54 : 478677 : ++pos;
55 : : }
56 [ + + ]: 478842 : for (const T& v : const_pre_vector | std::views::reverse) {
57 : 478677 : --pos;
58 [ - + ]: 478677 : assert(v == real_vector[pos]);
59 : : }
60 : 165 : DataStream ss1{};
61 : 165 : DataStream ss2{};
62 [ + - ]: 165 : ss1 << real_vector;
63 [ + - - + ]: 330 : ss2 << pre_vector;
64 [ - + - + ]: 165 : assert(ss1.size() == ss2.size());
65 [ + + ]: 1915104 : for (Size s = 0; s < ss1.size(); s++) {
66 [ - + ]: 1914939 : assert(ss1[s] == ss2[s]);
67 : : }
68 : 165 : }
69 : :
70 : 7343 : void resize(Size s)
71 : : {
72 : 7343 : real_vector.resize(s);
73 [ - + - + ]: 7343 : assert(real_vector.size() == s);
74 : 7343 : pre_vector.resize(s);
75 [ + + - + ]: 10720 : assert(pre_vector.size() == s);
76 : 7343 : }
77 : :
78 : 3904 : void reserve(Size s)
79 : : {
80 : 3904 : real_vector.reserve(s);
81 [ - + - + ]: 3904 : assert(real_vector.capacity() >= s);
82 : 3904 : pre_vector.reserve(s);
83 [ + + - + ]: 7488 : assert(pre_vector.capacity() >= s);
84 : 3904 : }
85 : :
86 : 10177 : void insert(Size position, const T& value)
87 : : {
88 : 10177 : real_vector.insert(real_vector.begin() + position, value);
89 [ + + ]: 20354 : pre_vector.insert(pre_vector.begin() + position, value);
90 : 10177 : }
91 : :
92 : 8531 : void insert(Size position, Size count, const T& value)
93 : : {
94 : 8531 : real_vector.insert(real_vector.begin() + position, count, value);
95 [ + + ]: 17062 : pre_vector.insert(pre_vector.begin() + position, count, value);
96 : 8531 : }
97 : :
98 : : template <typename I>
99 : 5351 : void insert_range(Size position, I first, I last)
100 : : {
101 : 5351 : real_vector.insert(real_vector.begin() + position, first, last);
102 [ + + ]: 10702 : pre_vector.insert(pre_vector.begin() + position, first, last);
103 : 5351 : }
104 : :
105 : 2848 : void erase(Size position)
106 : : {
107 : 2848 : real_vector.erase(real_vector.begin() + position);
108 [ + + ]: 5696 : pre_vector.erase(pre_vector.begin() + position);
109 : 2848 : }
110 : :
111 : 4647 : void erase(Size first, Size last)
112 : : {
113 : 4647 : real_vector.erase(real_vector.begin() + first, real_vector.begin() + last);
114 [ + + ]: 9294 : pre_vector.erase(pre_vector.begin() + first, pre_vector.begin() + last);
115 : 4647 : }
116 : :
117 : 2415 : void update(Size pos, const T& value)
118 : : {
119 [ + + ]: 2415 : real_vector[pos] = value;
120 [ + + ]: 2415 : pre_vector[pos] = value;
121 : 2415 : }
122 : :
123 : 3148 : void push_back(const T& value)
124 : : {
125 : 3148 : real_vector.push_back(value);
126 : 3148 : pre_vector.push_back(value);
127 : 3148 : }
128 : :
129 : 3430 : void pop_back()
130 : : {
131 : 3430 : real_vector.pop_back();
132 : 3430 : pre_vector.pop_back();
133 : 3430 : }
134 : :
135 : 3501 : void clear()
136 : : {
137 [ + + ]: 3501 : real_vector.clear();
138 : 3501 : pre_vector.clear();
139 : 3501 : }
140 : :
141 : 8238 : void assign(Size n, const T& value)
142 : : {
143 : 8238 : real_vector.assign(n, value);
144 : 8238 : pre_vector.assign(n, value);
145 : 8238 : }
146 : :
147 : 138442 : Size size() const
148 : : {
149 : 276884 : return real_vector.size();
150 : : }
151 : :
152 : : Size capacity() const
153 : : {
154 : : return pre_vector.capacity();
155 : : }
156 : :
157 : 6280 : void shrink_to_fit()
158 : : {
159 : 6280 : pre_vector.shrink_to_fit();
160 : 6280 : }
161 : :
162 : 4038 : void swap() noexcept
163 : : {
164 : 4038 : real_vector.swap(real_vector_alt);
165 : 4038 : pre_vector.swap(pre_vector_alt);
166 : 4038 : }
167 : :
168 : 7880 : void move()
169 : : {
170 : 7880 : real_vector = std::move(real_vector_alt);
171 [ - + ]: 7880 : real_vector_alt.clear();
172 : 7880 : pre_vector = std::move(pre_vector_alt);
173 : 7880 : pre_vector_alt.clear();
174 : 7880 : }
175 : :
176 : 8394 : void copy()
177 : : {
178 : 8394 : real_vector = real_vector_alt;
179 : 8394 : pre_vector = pre_vector_alt;
180 : 8394 : }
181 : :
182 [ - + ]: 2358 : void resize_uninitialized(realtype values)
183 : : {
184 : 2358 : size_t r = values.size();
185 [ - + ]: 2358 : size_t s = real_vector.size() / 2;
186 [ - + + + ]: 2358 : if (real_vector.capacity() < s + r) {
187 : 949 : real_vector.reserve(s + r);
188 : : }
189 : 2358 : real_vector.resize(s);
190 : 2358 : pre_vector.resize_uninitialized(s);
191 [ + + ]: 17395 : for (auto v : values) {
192 : 15037 : real_vector.push_back(v);
193 : : }
194 [ + + ]: 2358 : auto p = pre_vector.size();
195 : 2358 : pre_vector.resize_uninitialized(p + r);
196 [ + + ]: 17395 : for (auto v : values) {
197 [ + + ]: 15037 : pre_vector[p] = v;
198 : 15037 : ++p;
199 : : }
200 : 2358 : }
201 : : };
202 : :
203 : : } // namespace
204 : :
205 [ + - ]: 623 : FUZZ_TARGET(prevector)
206 : : {
207 : 165 : FuzzedDataProvider prov(buffer.data(), buffer.size());
208 : 165 : prevector_tester<8, int> test;
209 : :
210 [ + + + + ]: 92648 : LIMITED_WHILE(prov.remaining_bytes(), 3000)
211 : : {
212 [ + + + + : 115648 : switch (prov.ConsumeIntegralInRange<int>(0, 13 + 3 * (test.size() > 0))) {
+ + + + +
+ + + + +
+ + + + +
- - + ]
213 : 10177 : case 0: {
214 [ - + ]: 10177 : auto position = prov.ConsumeIntegralInRange<size_t>(0, test.size());
215 : 10177 : auto value = prov.ConsumeIntegral<int>();
216 [ + - ]: 10177 : test.insert(position, value);
217 : : } break;
218 : 7343 : case 1:
219 [ - + + + : 20039 : test.resize(std::max(0, std::min(30, (int)test.size() + prov.ConsumeIntegralInRange<int>(0, 4) - 2)));
+ + + - ]
220 : : break;
221 : 8531 : case 2: {
222 [ - + ]: 8531 : auto position = prov.ConsumeIntegralInRange<size_t>(0, test.size());
223 : 8531 : auto count = 1 + prov.ConsumeBool();
224 : 8531 : auto value = prov.ConsumeIntegral<int>();
225 [ + - ]: 8531 : test.insert(position, count, value);
226 : : } break;
227 : 4647 : case 3: {
228 [ - + ]: 4647 : int del = prov.ConsumeIntegralInRange<int>(0, test.size());
229 [ - + ]: 4647 : int beg = prov.ConsumeIntegralInRange<int>(0, test.size() - del);
230 : 4647 : test.erase(beg, beg + del);
231 : : break;
232 : : }
233 : 3148 : case 4:
234 [ + - ]: 3148 : test.push_back(prov.ConsumeIntegral<int>());
235 : : break;
236 : 5351 : case 5: {
237 : 5351 : int values[4];
238 : 5351 : int num = 1 + prov.ConsumeIntegralInRange<int>(0, 3);
239 [ + + ]: 16915 : for (int k = 0; k < num; ++k) {
240 : 11564 : values[k] = prov.ConsumeIntegral<int>();
241 : : }
242 [ - + + - ]: 5351 : test.insert_range(prov.ConsumeIntegralInRange<size_t>(0, test.size()), values, values + num);
243 : : break;
244 : : }
245 : 2358 : case 6: {
246 : 2358 : int num = 1 + prov.ConsumeIntegralInRange<int>(0, 15);
247 [ + - ]: 2358 : std::vector<int> values(num);
248 [ + + ]: 17395 : for (auto& v : values) {
249 : 15037 : v = prov.ConsumeIntegral<int>();
250 : : }
251 [ + - + - ]: 2358 : test.resize_uninitialized(values);
252 : 2358 : break;
253 : 2358 : }
254 : 3904 : case 7:
255 [ + - ]: 3904 : test.reserve(prov.ConsumeIntegralInRange<size_t>(0, 32767));
256 : : break;
257 : 6280 : case 8:
258 : 6280 : test.shrink_to_fit();
259 : : break;
260 : 3501 : case 9:
261 : 3501 : test.clear();
262 : : break;
263 : 8238 : case 10: {
264 : 8238 : auto n = prov.ConsumeIntegralInRange<size_t>(0, 32767);
265 : 8238 : auto value = prov.ConsumeIntegral<int>();
266 [ + - ]: 8238 : test.assign(n, value);
267 : : } break;
268 : 4038 : case 11:
269 : 4038 : test.swap();
270 : 4038 : break;
271 : 8394 : case 12:
272 [ + - ]: 8394 : test.copy();
273 : : break;
274 : 7880 : case 13:
275 : 7880 : test.move();
276 : : break;
277 : 2415 : case 14: {
278 [ - + ]: 2415 : auto pos = prov.ConsumeIntegralInRange<size_t>(0, test.size() - 1);
279 : 2415 : auto value = prov.ConsumeIntegral<int>();
280 : 2415 : test.update(pos, value);
281 : : } break;
282 : 2848 : case 15:
283 [ - + ]: 2848 : test.erase(prov.ConsumeIntegralInRange<size_t>(0, test.size() - 1));
284 : : break;
285 : 3430 : case 16:
286 : 3430 : test.pop_back();
287 : : break;
288 : : }
289 : : }
290 : :
291 [ + - ]: 165 : test.test();
292 : 165 : }
|