/ src / util / feefrac.h
feefrac.h
  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 CompareFeeFrac, 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      {
 46          // Otherwise, emulate 96-bit multiplication using two 64-bit multiplies.
 47          int64_t low = int64_t{static_cast<uint32_t>(a)} * b;
 48          int64_t high = (a >> 32) * b;
 49          return {high + (low >> 32), static_cast<uint32_t>(low)};
 50      }
 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      {
 56          // If __int128 is available, use 128-bit wide multiply.
 57          return __int128{a} * b;
 58      }
 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      inline FeeFrac() noexcept : fee{0}, size{0} {}
 68  
 69      /** Construct a FeeFrac with specified fee and size. */
 70      inline FeeFrac(int64_t f, int32_t s) noexcept : fee{f}, size{s} {}
 71  
 72      inline FeeFrac(const FeeFrac&) noexcept = default;
 73      inline FeeFrac& operator=(const FeeFrac&) noexcept = default;
 74  
 75      /** Check if this is empty (size and fee are 0). */
 76      bool inline IsEmpty() const noexcept {
 77          return size == 0;
 78      }
 79  
 80      /** Add fee and size of another FeeFrac to this one. */
 81      void inline operator+=(const FeeFrac& other) noexcept
 82      {
 83          fee += other.fee;
 84          size += other.size;
 85      }
 86  
 87      /** Subtract fee and size of another FeeFrac from this one. */
 88      void inline operator-=(const FeeFrac& other) noexcept
 89      {
 90          fee -= other.fee;
 91          size -= other.size;
 92      }
 93  
 94      /** Sum fee and size. */
 95      friend inline FeeFrac operator+(const FeeFrac& a, const FeeFrac& b) noexcept
 96      {
 97          return {a.fee + b.fee, a.size + b.size};
 98      }
 99  
100      /** Subtract both fee and size. */
101      friend inline FeeFrac operator-(const FeeFrac& a, const FeeFrac& b) noexcept
102      {
103          return {a.fee - b.fee, a.size - b.size};
104      }
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      {
109          return a.fee == b.fee && a.size == b.size;
110      }
111  
112      /** Compare two FeeFracs just by feerate. */
113      friend inline std::weak_ordering FeeRateCompare(const FeeFrac& a, const FeeFrac& b) noexcept
114      {
115          auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
116          return cross_a <=> cross_b;
117      }
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      {
122          auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
123          return cross_a < cross_b;
124      }
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      {
129          auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
130          return cross_a > cross_b;
131      }
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      {
136          auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
137          if (cross_a == cross_b) return b.size <=> a.size;
138          return cross_a <=> cross_b;
139      }
140  
141      /** Swap two FeeFracs. */
142      friend inline void swap(FeeFrac& a, FeeFrac& b) noexcept
143      {
144          std::swap(a.fee, b.fee);
145          std::swap(a.size, b.size);
146      }
147  };
148  
149  /** Takes the pre-computed and topologically-valid chunks and generates a fee diagram which starts at FeeFrac of (0, 0) */
150  std::vector<FeeFrac> BuildDiagramFromChunks(Span<const FeeFrac> chunks);
151  
152  /** Compares two feerate diagrams. The shorter one is implicitly
153   * extended with a horizontal straight line.
154   *
155   * A feerate diagram consists of a list of (fee, size) points with the property that size
156   * is strictly increasing and that the first entry is (0, 0).
157   */
158  std::partial_ordering CompareFeerateDiagram(Span<const FeeFrac> dia0, Span<const FeeFrac> dia1);
159  
160  #endif // BITCOIN_UTIL_FEEFRAC_H