// Copyright (c) 2020-2022 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 #include #include #include #include #include #include #include namespace { uint64_t HashToRange(const std::vector& element, const uint64_t f) { const uint64_t hash = CSipHasher(0x0706050403020100ULL, 0x0F0E0D0C0B0A0908ULL) .Write(element) .Finalize(); return FastRange64(hash, f); } std::vector BuildHashedSet(const std::unordered_set, ByteVectorHash>& elements, const uint64_t f) { std::vector hashed_elements; hashed_elements.reserve(elements.size()); for (const std::vector& element : elements) { hashed_elements.push_back(HashToRange(element, f)); } std::sort(hashed_elements.begin(), hashed_elements.end()); return hashed_elements; } } // namespace FUZZ_TARGET(golomb_rice) { SeedRandomStateForTest(SeedRand::ZEROS); FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size()); std::vector golomb_rice_data; std::vector encoded_deltas; { std::unordered_set, ByteVectorHash> elements; const int n = fuzzed_data_provider.ConsumeIntegralInRange(0, 512); for (int i = 0; i < n; ++i) { elements.insert(ConsumeRandomLengthByteVector(fuzzed_data_provider, 16)); } VectorWriter stream{golomb_rice_data, 0}; WriteCompactSize(stream, static_cast(elements.size())); BitStreamWriter bitwriter{stream}; if (!elements.empty()) { uint64_t last_value = 0; for (const uint64_t value : BuildHashedSet(elements, static_cast(elements.size()) * static_cast(BASIC_FILTER_M))) { const uint64_t delta = value - last_value; encoded_deltas.push_back(delta); GolombRiceEncode(bitwriter, BASIC_FILTER_P, delta); last_value = value; } } bitwriter.Flush(); } std::vector decoded_deltas; { SpanReader stream{golomb_rice_data}; BitStreamReader bitreader{stream}; const uint32_t n = static_cast(ReadCompactSize(stream)); for (uint32_t i = 0; i < n; ++i) { decoded_deltas.push_back(GolombRiceDecode(bitreader, BASIC_FILTER_P)); } } assert(encoded_deltas == decoded_deltas); { const std::vector random_bytes = ConsumeRandomLengthByteVector(fuzzed_data_provider, 1024); SpanReader stream{random_bytes}; uint32_t n; try { n = static_cast(ReadCompactSize(stream)); } catch (const std::ios_base::failure&) { return; } BitStreamReader bitreader{stream}; for (uint32_t i = 0; i < std::min(n, 1024); ++i) { try { (void)GolombRiceDecode(bitreader, BASIC_FILTER_P); } catch (const std::ios_base::failure&) { } } } }