// Copyright (c) 2021 The Bitcoin Core developers // Distributed under the MIT software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include #include #include #include #include #include #include #include using node::MakeMinisketch32; FUZZ_TARGET(minisketch) { FuzzedDataProvider fuzzed_data_provider{buffer.data(), buffer.size()}; const auto capacity{fuzzed_data_provider.ConsumeIntegralInRange(1, 200)}; Minisketch sketch_a{Assert(MakeMinisketch32(capacity))}; Minisketch sketch_b{Assert(MakeMinisketch32(capacity))}; // Fill two sets and keep the difference in a map std::map diff; LIMITED_WHILE(fuzzed_data_provider.ConsumeBool(), 10000) { const auto entry{fuzzed_data_provider.ConsumeIntegralInRange(1, std::numeric_limits::max() - 1)}; const auto KeepDiff{[&] { bool& mut{diff[entry]}; mut = !mut; }}; CallOneOf( fuzzed_data_provider, [&] { sketch_a.Add(entry); KeepDiff(); }, [&] { sketch_b.Add(entry); KeepDiff(); }, [&] { sketch_a.Add(entry); sketch_b.Add(entry); }); } const auto num_diff{std::accumulate(diff.begin(), diff.end(), size_t{0}, [](auto n, const auto& e) { return n + e.second; })}; Minisketch sketch_ar{MakeMinisketch32(capacity)}; Minisketch sketch_br{MakeMinisketch32(capacity)}; sketch_ar.Deserialize(sketch_a.Serialize()); sketch_br.Deserialize(sketch_b.Serialize()); Minisketch sketch_diff{std::move(fuzzed_data_provider.ConsumeBool() ? sketch_a : sketch_ar)}; sketch_diff.Merge(fuzzed_data_provider.ConsumeBool() ? sketch_b : sketch_br); if (capacity >= num_diff) { const auto max_elements{fuzzed_data_provider.ConsumeIntegralInRange(num_diff, capacity)}; const auto dec{*Assert(sketch_diff.Decode(max_elements))}; Assert(dec.size() == num_diff); for (auto d : dec) { Assert(diff.at(d)); } } }