LCOV - code coverage report
Current view: top level - src - blockfilter.cpp (source / functions) Coverage Total Hit
Test: fuzz_coverage.info Lines: 56.4 % 133 75
Test Date: 2025-01-22 04:09:46 Functions: 68.4 % 19 13
Branches: 32.6 % 138 45

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2018-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 <mutex>
       6                 :             : #include <set>
       7                 :             : 
       8                 :             : #include <blockfilter.h>
       9                 :             : #include <crypto/siphash.h>
      10                 :             : #include <hash.h>
      11                 :             : #include <primitives/block.h>
      12                 :             : #include <primitives/transaction.h>
      13                 :             : #include <script/script.h>
      14                 :             : #include <streams.h>
      15                 :             : #include <undo.h>
      16                 :             : #include <util/golombrice.h>
      17                 :             : #include <util/string.h>
      18                 :             : 
      19                 :             : using util::Join;
      20                 :             : 
      21                 :             : static const std::map<BlockFilterType, std::string> g_filter_types = {
      22                 :             :     {BlockFilterType::BASIC, "basic"},
      23                 :             : };
      24                 :             : 
      25                 :      339188 : uint64_t GCSFilter::HashToRange(const Element& element) const
      26                 :             : {
      27                 :      339188 :     uint64_t hash = CSipHasher(m_params.m_siphash_k0, m_params.m_siphash_k1)
      28                 :      339188 :         .Write(element)
      29                 :      339188 :         .Finalize();
      30                 :      339188 :     return FastRange64(hash, m_F);
      31                 :             : }
      32                 :             : 
      33                 :         629 : std::vector<uint64_t> GCSFilter::BuildHashedSet(const ElementSet& elements) const
      34                 :             : {
      35                 :         629 :     std::vector<uint64_t> hashed_elements;
      36         [ +  - ]:         629 :     hashed_elements.reserve(elements.size());
      37         [ +  + ]:      339188 :     for (const Element& element : elements) {
      38   [ +  -  +  - ]:      338559 :         hashed_elements.push_back(HashToRange(element));
      39                 :             :     }
      40                 :         629 :     std::sort(hashed_elements.begin(), hashed_elements.end());
      41                 :         629 :     return hashed_elements;
      42                 :           0 : }
      43                 :             : 
      44                 :         783 : GCSFilter::GCSFilter(const Params& params)
      45                 :         783 :     : m_params(params), m_N(0), m_F(0), m_encoded{0}
      46                 :         783 : {}
      47                 :             : 
      48                 :         740 : GCSFilter::GCSFilter(const Params& params, std::vector<unsigned char> encoded_filter, bool skip_decode_check)
      49         [ +  + ]:         740 :     : m_params(params), m_encoded(std::move(encoded_filter))
      50                 :             : {
      51         [ +  + ]:         740 :     SpanReader stream{m_encoded};
      52                 :             : 
      53         [ +  + ]:         740 :     uint64_t N = ReadCompactSize(stream);
      54                 :         719 :     m_N = static_cast<uint32_t>(N);
      55         [ -  + ]:         719 :     if (m_N != N) {
      56         [ #  # ]:           0 :         throw std::ios_base::failure("N must be <2^32");
      57                 :             :     }
      58                 :         719 :     m_F = static_cast<uint64_t>(m_N) * static_cast<uint64_t>(m_params.m_M);
      59                 :             : 
      60         [ +  - ]:         719 :     if (skip_decode_check) return;
      61                 :             : 
      62                 :             :     // Verify that the encoded filter contains exactly N elements. If it has too much or too little
      63                 :             :     // data, a std::ios_base::failure exception will be raised.
      64                 :         719 :     BitStreamReader bitreader{stream};
      65         [ +  + ]:      567242 :     for (uint64_t i = 0; i < m_N; ++i) {
      66         [ +  + ]:      566558 :         GolombRiceDecode(bitreader, m_params.m_P);
      67                 :             :     }
      68         [ +  + ]:         684 :     if (!stream.empty()) {
      69         [ +  - ]:           9 :         throw std::ios_base::failure("encoded_filter contains excess data");
      70                 :             :     }
      71                 :          65 : }
      72                 :             : 
      73                 :           0 : GCSFilter::GCSFilter(const Params& params, const ElementSet& elements)
      74         [ #  # ]:           0 :     : m_params(params)
      75                 :             : {
      76         [ #  # ]:           0 :     size_t N = elements.size();
      77                 :           0 :     m_N = static_cast<uint32_t>(N);
      78         [ #  # ]:           0 :     if (m_N != N) {
      79         [ #  # ]:           0 :         throw std::invalid_argument("N must be <2^32");
      80                 :             :     }
      81                 :           0 :     m_F = static_cast<uint64_t>(m_N) * static_cast<uint64_t>(m_params.m_M);
      82                 :             : 
      83         [ #  # ]:           0 :     VectorWriter stream{m_encoded, 0};
      84                 :             : 
      85         [ #  # ]:           0 :     WriteCompactSize(stream, m_N);
      86                 :             : 
      87         [ #  # ]:           0 :     if (elements.empty()) {
      88                 :             :         return;
      89                 :             :     }
      90                 :             : 
      91         [ #  # ]:           0 :     BitStreamWriter bitwriter{stream};
      92                 :             : 
      93                 :           0 :     uint64_t last_value = 0;
      94   [ #  #  #  # ]:           0 :     for (uint64_t value : BuildHashedSet(elements)) {
      95                 :           0 :         uint64_t delta = value - last_value;
      96         [ #  # ]:           0 :         GolombRiceEncode(bitwriter, m_params.m_P, delta);
      97                 :           0 :         last_value = value;
      98                 :           0 :     }
      99                 :             : 
     100         [ #  # ]:           0 :     bitwriter.Flush();
     101                 :           0 : }
     102                 :             : 
     103                 :        1258 : bool GCSFilter::MatchInternal(const uint64_t* element_hashes, size_t size) const
     104                 :             : {
     105                 :        1258 :     SpanReader stream{m_encoded};
     106                 :             : 
     107                 :             :     // Seek forward by size of N
     108                 :        1258 :     uint64_t N = ReadCompactSize(stream);
     109         [ -  + ]:        1258 :     assert(N == m_N);
     110                 :             : 
     111                 :        1258 :     BitStreamReader bitreader{stream};
     112                 :             : 
     113                 :        1258 :     uint64_t value = 0;
     114                 :        1258 :     size_t hashes_index = 0;
     115         [ +  + ]:      835333 :     for (uint32_t i = 0; i < m_N; ++i) {
     116                 :      834873 :         uint64_t delta = GolombRiceDecode(bitreader, m_params.m_P);
     117                 :      834873 :         value += delta;
     118                 :             : 
     119                 :     1426995 :         while (true) {
     120         [ +  + ]:     1130934 :             if (hashes_index == size) {
     121                 :             :                 return false;
     122         [ +  + ]:     1130143 :             } else if (element_hashes[hashes_index] == value) {
     123                 :             :                 return true;
     124         [ +  + ]:     1130136 :             } else if (element_hashes[hashes_index] > value) {
     125                 :             :                 break;
     126                 :             :             }
     127                 :             : 
     128                 :      296061 :             hashes_index++;
     129                 :             :         }
     130                 :             :     }
     131                 :             : 
     132                 :             :     return false;
     133                 :             : }
     134                 :             : 
     135                 :         629 : bool GCSFilter::Match(const Element& element) const
     136                 :             : {
     137                 :         629 :     uint64_t query = HashToRange(element);
     138                 :         629 :     return MatchInternal(&query, 1);
     139                 :             : }
     140                 :             : 
     141                 :         629 : bool GCSFilter::MatchAny(const ElementSet& elements) const
     142                 :             : {
     143                 :         629 :     const std::vector<uint64_t> queries = BuildHashedSet(elements);
     144         [ +  - ]:         629 :     return MatchInternal(queries.data(), queries.size());
     145                 :         629 : }
     146                 :             : 
     147                 :         780 : const std::string& BlockFilterTypeName(BlockFilterType filter_type)
     148                 :             : {
     149   [ +  +  +  - ]:         865 :     static std::string unknown_retval;
     150                 :         780 :     auto it = g_filter_types.find(filter_type);
     151         [ +  - ]:         780 :     return it != g_filter_types.end() ? it->second : unknown_retval;
     152                 :             : }
     153                 :             : 
     154                 :         966 : bool BlockFilterTypeByName(const std::string& name, BlockFilterType& filter_type) {
     155         [ +  + ]:        1929 :     for (const auto& entry : g_filter_types) {
     156         [ +  + ]:         966 :         if (entry.second == name) {
     157                 :           3 :             filter_type = entry.first;
     158                 :           3 :             return true;
     159                 :             :         }
     160                 :             :     }
     161                 :             :     return false;
     162                 :             : }
     163                 :             : 
     164                 :           0 : const std::set<BlockFilterType>& AllBlockFilterTypes()
     165                 :             : {
     166   [ #  #  #  # ]:           0 :     static std::set<BlockFilterType> types;
     167                 :             : 
     168                 :           0 :     static std::once_flag flag;
     169                 :           0 :     std::call_once(flag, []() {
     170         [ #  # ]:           0 :             for (const auto& entry : g_filter_types) {
     171                 :           0 :                 types.insert(entry.first);
     172                 :             :             }
     173                 :           0 :         });
     174                 :             : 
     175                 :           0 :     return types;
     176                 :             : }
     177                 :             : 
     178                 :        1653 : const std::string& ListBlockFilterTypes()
     179                 :             : {
     180   [ +  +  +  -  :        1823 :     static std::string type_list{Join(g_filter_types, ", ", [](const auto& entry) { return entry.second; })};
             +  -  +  - ]
     181                 :             : 
     182                 :        1653 :     return type_list;
     183                 :             : }
     184                 :             : 
     185                 :           0 : static GCSFilter::ElementSet BasicFilterElements(const CBlock& block,
     186                 :             :                                                  const CBlockUndo& block_undo)
     187                 :             : {
     188                 :           0 :     GCSFilter::ElementSet elements;
     189                 :             : 
     190         [ #  # ]:           0 :     for (const CTransactionRef& tx : block.vtx) {
     191         [ #  # ]:           0 :         for (const CTxOut& txout : tx->vout) {
     192                 :           0 :             const CScript& script = txout.scriptPubKey;
     193   [ #  #  #  #  :           0 :             if (script.empty() || script[0] == OP_RETURN) continue;
             #  #  #  # ]
     194         [ #  # ]:           0 :             elements.emplace(script.begin(), script.end());
     195                 :             :         }
     196                 :             :     }
     197                 :             : 
     198         [ #  # ]:           0 :     for (const CTxUndo& tx_undo : block_undo.vtxundo) {
     199         [ #  # ]:           0 :         for (const Coin& prevout : tx_undo.vprevout) {
     200                 :           0 :             const CScript& script = prevout.out.scriptPubKey;
     201   [ #  #  #  # ]:           0 :             if (script.empty()) continue;
     202   [ #  #  #  # ]:           0 :             elements.emplace(script.begin(), script.end());
     203                 :             :         }
     204                 :             :     }
     205                 :             : 
     206                 :           0 :     return elements;
     207                 :           0 : }
     208                 :             : 
     209                 :           0 : BlockFilter::BlockFilter(BlockFilterType filter_type, const uint256& block_hash,
     210                 :           0 :                          std::vector<unsigned char> filter, bool skip_decode_check)
     211                 :           0 :     : m_filter_type(filter_type), m_block_hash(block_hash)
     212                 :             : {
     213         [ #  # ]:           0 :     GCSFilter::Params params;
     214   [ #  #  #  # ]:           0 :     if (!BuildParams(params)) {
     215         [ #  # ]:           0 :         throw std::invalid_argument("unknown filter_type");
     216                 :             :     }
     217         [ #  # ]:           0 :     m_filter = GCSFilter(params, std::move(filter), skip_decode_check);
     218                 :           0 : }
     219                 :             : 
     220                 :           0 : BlockFilter::BlockFilter(BlockFilterType filter_type, const CBlock& block, const CBlockUndo& block_undo)
     221                 :           0 :     : m_filter_type(filter_type), m_block_hash(block.GetHash())
     222                 :             : {
     223         [ #  # ]:           0 :     GCSFilter::Params params;
     224   [ #  #  #  # ]:           0 :     if (!BuildParams(params)) {
     225         [ #  # ]:           0 :         throw std::invalid_argument("unknown filter_type");
     226                 :             :     }
     227   [ #  #  #  # ]:           0 :     m_filter = GCSFilter(params, BasicFilterElements(block, block_undo));
     228                 :           0 : }
     229                 :             : 
     230                 :         745 : bool BlockFilter::BuildParams(GCSFilter::Params& params) const
     231                 :             : {
     232         [ +  + ]:         745 :     switch (m_filter_type) {
     233                 :         740 :     case BlockFilterType::BASIC:
     234                 :         740 :         params.m_siphash_k0 = m_block_hash.GetUint64(0);
     235                 :         740 :         params.m_siphash_k1 = m_block_hash.GetUint64(1);
     236                 :         740 :         params.m_P = BASIC_FILTER_P;
     237                 :         740 :         params.m_M = BASIC_FILTER_M;
     238                 :         740 :         return true;
     239                 :             :     case BlockFilterType::INVALID:
     240                 :             :         return false;
     241                 :             :     }
     242                 :             : 
     243                 :             :     return false;
     244                 :             : }
     245                 :             : 
     246                 :        1258 : uint256 BlockFilter::GetHash() const
     247                 :             : {
     248                 :        1258 :     return Hash(GetEncodedFilter());
     249                 :             : }
     250                 :             : 
     251                 :         629 : uint256 BlockFilter::ComputeHeader(const uint256& prev_header) const
     252                 :             : {
     253                 :         629 :     return Hash(GetHash(), prev_header);
     254                 :             : }
        

Generated by: LCOV version 2.0-1