Coverage Report

Created: 2024-10-29 12:15

/root/bitcoin/src/common/bloom.h
Line
Count
Source (jump to first uncovered line)
1
// Copyright (c) 2012-2021 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
#ifndef BITCOIN_COMMON_BLOOM_H
6
#define BITCOIN_COMMON_BLOOM_H
7
8
#include <serialize.h>
9
#include <span.h>
10
11
#include <vector>
12
13
class COutPoint;
14
class CTransaction;
15
16
//! 20,000 items with fp rate < 0.1% or 10,000 items and <0.0001%
17
static constexpr unsigned int MAX_BLOOM_FILTER_SIZE = 36000; // bytes
18
static constexpr unsigned int MAX_HASH_FUNCS = 50;
19
20
/**
21
 * First two bits of nFlags control how much IsRelevantAndUpdate actually updates
22
 * The remaining bits are reserved
23
 */
24
enum bloomflags
25
{
26
    BLOOM_UPDATE_NONE = 0,
27
    BLOOM_UPDATE_ALL = 1,
28
    // Only adds outpoints to the filter if the output is a pay-to-pubkey/pay-to-multisig script
29
    BLOOM_UPDATE_P2PUBKEY_ONLY = 2,
30
    BLOOM_UPDATE_MASK = 3,
31
};
32
33
/**
34
 * BloomFilter is a probabilistic filter which SPV clients provide
35
 * so that we can filter the transactions we send them.
36
 *
37
 * This allows for significantly more efficient transaction and block downloads.
38
 *
39
 * Because bloom filters are probabilistic, a SPV node can increase the false-
40
 * positive rate, making us send it transactions which aren't actually its,
41
 * allowing clients to trade more bandwidth for more privacy by obfuscating which
42
 * keys are controlled by them.
43
 */
44
class CBloomFilter
45
{
46
private:
47
    std::vector<unsigned char> vData;
48
    unsigned int nHashFuncs;
49
    unsigned int nTweak;
50
    unsigned char nFlags;
51
52
    unsigned int Hash(unsigned int nHashNum, Span<const unsigned char> vDataToHash) const;
53
54
public:
55
    /**
56
     * Creates a new bloom filter which will provide the given fp rate when filled with the given number of elements
57
     * Note that if the given parameters will result in a filter outside the bounds of the protocol limits,
58
     * the filter created will be as close to the given parameters as possible within the protocol limits.
59
     * This will apply if nFPRate is very low or nElements is unreasonably high.
60
     * nTweak is a constant which is added to the seed value passed to the hash function
61
     * It should generally always be a random value (and is largely only exposed for unit testing)
62
     * nFlags should be one of the BLOOM_UPDATE_* enums (not _MASK)
63
     */
64
    CBloomFilter(const unsigned int nElements, const double nFPRate, const unsigned int nTweak, unsigned char nFlagsIn);
65
0
    CBloomFilter() : nHashFuncs(0), nTweak(0), nFlags(0) {}
66
67
0
    SERIALIZE_METHODS(CBloomFilter, obj) { READWRITE(obj.vData, obj.nHashFuncs, obj.nTweak, obj.nFlags); }
Unexecuted instantiation: _ZN12CBloomFilter16SerializationOpsI10DataStreamS_17ActionUnserializeEEvRT0_RT_T1_
Unexecuted instantiation: _ZN12CBloomFilter16SerializationOpsI10DataStreamKS_15ActionSerializeEEvRT0_RT_T1_
68
69
    void insert(Span<const unsigned char> vKey);
70
    void insert(const COutPoint& outpoint);
71
72
    bool contains(Span<const unsigned char> vKey) const;
73
    bool contains(const COutPoint& outpoint) const;
74
75
    //! True if the size is <= MAX_BLOOM_FILTER_SIZE and the number of hash functions is <= MAX_HASH_FUNCS
76
    //! (catch a filter which was just deserialized which was too big)
77
    bool IsWithinSizeConstraints() const;
78
79
    //! Also adds any outputs which match the filter to the filter (to match their spending txes)
80
    bool IsRelevantAndUpdate(const CTransaction& tx);
81
};
82
83
/**
84
 * RollingBloomFilter is a probabilistic "keep track of most recently inserted" set.
85
 * Construct it with the number of items to keep track of, and a false-positive
86
 * rate. Unlike CBloomFilter, by default nTweak is set to a cryptographically
87
 * secure random value for you. Similarly rather than clear() the method
88
 * reset() is provided, which also changes nTweak to decrease the impact of
89
 * false-positives.
90
 *
91
 * contains(item) will always return true if item was one of the last N to 1.5*N
92
 * insert()'ed ... but may also return true for items that were not inserted.
93
 *
94
 * It needs around 1.8 bytes per element per factor 0.1 of false positive rate.
95
 * For example, if we want 1000 elements, we'd need:
96
 * - ~1800 bytes for a false positive rate of 0.1
97
 * - ~3600 bytes for a false positive rate of 0.01
98
 * - ~5400 bytes for a false positive rate of 0.001
99
 *
100
 * If we make these simplifying assumptions:
101
 * - logFpRate / log(0.5) doesn't get rounded or clamped in the nHashFuncs calculation
102
 * - nElements is even, so that nEntriesPerGeneration == nElements / 2
103
 *
104
 * Then we get a more accurate estimate for filter bytes:
105
 *
106
 *     3/(log(256)*log(2)) * log(1/fpRate) * nElements
107
 */
108
class CRollingBloomFilter
109
{
110
public:
111
    CRollingBloomFilter(const unsigned int nElements, const double nFPRate);
112
113
    void insert(Span<const unsigned char> vKey);
114
    bool contains(Span<const unsigned char> vKey) const;
115
116
    void reset();
117
118
private:
119
    int nEntriesPerGeneration;
120
    int nEntriesThisGeneration;
121
    int nGeneration;
122
    std::vector<uint64_t> data;
123
    unsigned int nTweak;
124
    int nHashFuncs;
125
};
126
127
#endif // BITCOIN_COMMON_BLOOM_H