/root/bitcoin/src/util/feefrac.h
Line | Count | Source (jump to first uncovered line) |
1 | | // Copyright (c) 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_UTIL_FEEFRAC_H |
6 | | #define BITCOIN_UTIL_FEEFRAC_H |
7 | | |
8 | | #include <stdint.h> |
9 | | #include <compare> |
10 | | #include <vector> |
11 | | #include <span.h> |
12 | | #include <util/check.h> |
13 | | |
14 | | /** Data structure storing a fee and size, ordered by increasing fee/size. |
15 | | * |
16 | | * The size of a FeeFrac cannot be zero unless the fee is also zero. |
17 | | * |
18 | | * FeeFracs have a total ordering, first by increasing feerate (ratio of fee over size), and then |
19 | | * by decreasing size. The empty FeeFrac (fee and size both 0) sorts last. So for example, the |
20 | | * following FeeFracs are in sorted order: |
21 | | * |
22 | | * - fee=0 size=1 (feerate 0) |
23 | | * - fee=1 size=2 (feerate 0.5) |
24 | | * - fee=2 size=3 (feerate 0.667...) |
25 | | * - fee=2 size=2 (feerate 1) |
26 | | * - fee=1 size=1 (feerate 1) |
27 | | * - fee=3 size=2 (feerate 1.5) |
28 | | * - fee=2 size=1 (feerate 2) |
29 | | * - fee=0 size=0 (undefined feerate) |
30 | | * |
31 | | * A FeeFrac is considered "better" if it sorts after another, by this ordering. All standard |
32 | | * comparison operators (<=>, ==, !=, >, <, >=, <=) respect this ordering. |
33 | | * |
34 | | * The FeeRateCompare, and >> and << operators only compare feerate and treat equal feerate but |
35 | | * different size as equivalent. The empty FeeFrac is neither lower or higher in feerate than any |
36 | | * other. |
37 | | */ |
38 | | struct FeeFrac |
39 | | { |
40 | | /** Fallback version for Mul (see below). |
41 | | * |
42 | | * Separate to permit testing on platforms where it isn't actually needed. |
43 | | */ |
44 | | static inline std::pair<int64_t, uint32_t> MulFallback(int64_t a, int32_t b) noexcept |
45 | 0 | { |
46 | | // Otherwise, emulate 96-bit multiplication using two 64-bit multiplies. |
47 | 0 | int64_t low = int64_t{static_cast<uint32_t>(a)} * b; |
48 | 0 | int64_t high = (a >> 32) * b; |
49 | 0 | return {high + (low >> 32), static_cast<uint32_t>(low)}; |
50 | 0 | } |
51 | | |
52 | | // Compute a * b, returning an unspecified but totally ordered type. |
53 | | #ifdef __SIZEOF_INT128__ |
54 | | static inline __int128 Mul(int64_t a, int32_t b) noexcept |
55 | 0 | { |
56 | | // If __int128 is available, use 128-bit wide multiply. |
57 | 0 | return __int128{a} * b; |
58 | 0 | } |
59 | | #else |
60 | | static constexpr auto Mul = MulFallback; |
61 | | #endif |
62 | | |
63 | | int64_t fee; |
64 | | int32_t size; |
65 | | |
66 | | /** Construct an IsEmpty() FeeFrac. */ |
67 | 0 | constexpr inline FeeFrac() noexcept : fee{0}, size{0} {} |
68 | | |
69 | | /** Construct a FeeFrac with specified fee and size. */ |
70 | 0 | constexpr inline FeeFrac(int64_t f, int32_t s) noexcept : fee{f}, size{s} {} |
71 | | |
72 | | constexpr inline FeeFrac(const FeeFrac&) noexcept = default; |
73 | | constexpr inline FeeFrac& operator=(const FeeFrac&) noexcept = default; |
74 | | |
75 | | /** Check if this is empty (size and fee are 0). */ |
76 | 0 | bool inline IsEmpty() const noexcept { |
77 | 0 | return size == 0; |
78 | 0 | } |
79 | | |
80 | | /** Add fee and size of another FeeFrac to this one. */ |
81 | | void inline operator+=(const FeeFrac& other) noexcept |
82 | 0 | { |
83 | 0 | fee += other.fee; |
84 | 0 | size += other.size; |
85 | 0 | } |
86 | | |
87 | | /** Subtract fee and size of another FeeFrac from this one. */ |
88 | | void inline operator-=(const FeeFrac& other) noexcept |
89 | 0 | { |
90 | 0 | fee -= other.fee; |
91 | 0 | size -= other.size; |
92 | 0 | } |
93 | | |
94 | | /** Sum fee and size. */ |
95 | | friend inline FeeFrac operator+(const FeeFrac& a, const FeeFrac& b) noexcept |
96 | 0 | { |
97 | 0 | return {a.fee + b.fee, a.size + b.size}; |
98 | 0 | } |
99 | | |
100 | | /** Subtract both fee and size. */ |
101 | | friend inline FeeFrac operator-(const FeeFrac& a, const FeeFrac& b) noexcept |
102 | 0 | { |
103 | 0 | return {a.fee - b.fee, a.size - b.size}; |
104 | 0 | } |
105 | | |
106 | | /** Check if two FeeFrac objects are equal (both same fee and same size). */ |
107 | | friend inline bool operator==(const FeeFrac& a, const FeeFrac& b) noexcept |
108 | 0 | { |
109 | 0 | return a.fee == b.fee && a.size == b.size; |
110 | 0 | } |
111 | | |
112 | | /** Compare two FeeFracs just by feerate. */ |
113 | | friend inline std::weak_ordering FeeRateCompare(const FeeFrac& a, const FeeFrac& b) noexcept |
114 | 0 | { |
115 | 0 | auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size); |
116 | 0 | return cross_a <=> cross_b; |
117 | 0 | } |
118 | | |
119 | | /** Check if a FeeFrac object has strictly lower feerate than another. */ |
120 | | friend inline bool operator<<(const FeeFrac& a, const FeeFrac& b) noexcept |
121 | 0 | { |
122 | 0 | auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size); |
123 | 0 | return cross_a < cross_b; |
124 | 0 | } |
125 | | |
126 | | /** Check if a FeeFrac object has strictly higher feerate than another. */ |
127 | | friend inline bool operator>>(const FeeFrac& a, const FeeFrac& b) noexcept |
128 | 0 | { |
129 | 0 | auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size); |
130 | 0 | return cross_a > cross_b; |
131 | 0 | } |
132 | | |
133 | | /** Compare two FeeFracs. <, >, <=, and >= are auto-generated from this. */ |
134 | | friend inline std::strong_ordering operator<=>(const FeeFrac& a, const FeeFrac& b) noexcept |
135 | 0 | { |
136 | 0 | auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size); |
137 | 0 | if (cross_a == cross_b) return b.size <=> a.size; |
138 | 0 | return cross_a <=> cross_b; |
139 | 0 | } |
140 | | |
141 | | /** Swap two FeeFracs. */ |
142 | | friend inline void swap(FeeFrac& a, FeeFrac& b) noexcept |
143 | 0 | { |
144 | 0 | std::swap(a.fee, b.fee); |
145 | 0 | std::swap(a.size, b.size); |
146 | 0 | } |
147 | | }; |
148 | | |
149 | | /** Compare the feerate diagrams implied by the provided sorted chunks data. |
150 | | * |
151 | | * The implied diagram for each starts at (0, 0), then contains for each chunk the cumulative fee |
152 | | * and size up to that chunk, and then extends infinitely to the right with a horizontal line. |
153 | | * |
154 | | * The caller must guarantee that the sum of the FeeFracs in either of the chunks' data set do not |
155 | | * overflow (so sum fees < 2^63, and sum sizes < 2^31). |
156 | | */ |
157 | | std::partial_ordering CompareChunks(Span<const FeeFrac> chunks0, Span<const FeeFrac> chunks1); |
158 | | |
159 | | #endif // BITCOIN_UTIL_FEEFRAC_H |