// Copyright (c) 2020-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 #include #include #include #include #include #include namespace { uint64_t MapIntoRange(const uint64_t x, const uint64_t n) { const uint64_t x_hi = x >> 32; const uint64_t x_lo = x & 0xFFFFFFFF; const uint64_t n_hi = n >> 32; const uint64_t n_lo = n & 0xFFFFFFFF; const uint64_t ac = x_hi * n_hi; const uint64_t ad = x_hi * n_lo; const uint64_t bc = x_lo * n_hi; const uint64_t bd = x_lo * n_lo; const uint64_t mid34 = (bd >> 32) + (bc & 0xFFFFFFFF) + (ad & 0xFFFFFFFF); const uint64_t upper64 = ac + (bc >> 32) + (ad >> 32) + (mid34 >> 32); return upper64; } uint64_t HashToRange(const std::vector& element, const uint64_t f) { const uint64_t hash = CSipHasher(0x0706050403020100ULL, 0x0F0E0D0C0B0A0908ULL) .Write(element.data(), element.size()) .Finalize(); return MapIntoRange(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) { 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)); } CVectorWriter stream(SER_NETWORK, 0, 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{SER_NETWORK, 0, 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{SER_NETWORK, 0, 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&) { } } } }