LCOV - code coverage report
Current view: top level - src/util - feefrac.h (source / functions) Coverage Total Hit
Test: total_coverage.info Lines: 100.0 % 31 31
Test Date: 2025-01-19 05:08:01 Functions: 100.0 % 2 2
Branches: 45.5 % 156 71

             Branch data     Line data    Source code
       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                 :             :     {
      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                 :    43723445 :     static inline __int128 Mul(int64_t a, int32_t b) noexcept
      55                 :             :     {
      56                 :             :         // If __int128 is available, use 128-bit wide multiply.
      57                 :    43723447 :         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                 :          30 :     constexpr inline FeeFrac() noexcept : fee{0}, size{0} {}
      68                 :             : 
      69                 :             :     /** Construct a FeeFrac with specified fee and size. */
      70   [ +  +  #  #  :    39745935 :     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                 :          20 :     bool inline IsEmpty() const noexcept {
      77         [ +  - ]:          20 :         return size == 0;
      78                 :             :     }
      79                 :             : 
      80                 :             :     /** Add fee and size of another FeeFrac to this one. */
      81                 :         276 :     void inline operator+=(const FeeFrac& other) noexcept
      82                 :             :     {
      83                 :         276 :         fee += other.fee;
      84                 :         276 :         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                 :         452 :     friend inline FeeFrac operator+(const FeeFrac& a, const FeeFrac& b) noexcept
      96                 :             :     {
      97                 :         452 :         return {a.fee + b.fee, a.size + b.size};
      98                 :             :     }
      99                 :             : 
     100                 :             :     /** Subtract both fee and size. */
     101                 :         338 :     friend inline FeeFrac operator-(const FeeFrac& a, const FeeFrac& b) noexcept
     102                 :             :     {
     103   [ +  +  +  + ]:         249 :         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                 :    19868295 :     friend inline bool operator==(const FeeFrac& a, const FeeFrac& b) noexcept
     108                 :             :     {
     109   [ +  +  +  +  :    19868295 :         return a.fee == b.fee && a.size == b.size;
             +  +  +  + ]
           [ +  -  -  +  
          +  -  -  +  +  
          -  -  +  -  +  
          -  -  -  +  -  
          -  +  -  -  +  
          +  -  -  +  -  
           +  -  - ][ +  
          +  +  +  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
          #  #  #  #  #  
                      # ]
     110                 :             :     }
     111                 :             : 
     112                 :             :     /** Compare two FeeFracs just by feerate. */
     113                 :         189 :     friend inline std::weak_ordering FeeRateCompare(const FeeFrac& a, const FeeFrac& b) noexcept
     114                 :             :     {
     115                 :         189 :         auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
     116   [ +  +  +  + ]:         189 :         return cross_a <=> cross_b;
     117                 :             :     }
     118                 :             : 
     119                 :             :     /** Check if a FeeFrac object has strictly lower feerate than another. */
     120                 :           4 :     friend inline bool operator<<(const FeeFrac& a, const FeeFrac& b) noexcept
     121                 :             :     {
     122         [ +  - ]:           1 :         auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
     123   [ +  -  +  -  :           4 :         return cross_a < cross_b;
             +  -  +  - ]
     124                 :             :     }
     125                 :             : 
     126                 :             :     /** Check if a FeeFrac object has strictly higher feerate than another. */
     127                 :          69 :     friend inline bool operator>>(const FeeFrac& a, const FeeFrac& b) noexcept
     128                 :             :     {
     129         [ +  - ]:          67 :         auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
     130   [ +  +  +  -  :          69 :         return cross_a > cross_b;
           +  - ][ +  + ]
     131                 :             :     }
     132                 :             : 
     133                 :             :     /** Compare two FeeFracs. <, >, <=, and >= are auto-generated from this. */
     134                 :    43723183 :     friend inline std::strong_ordering operator<=>(const FeeFrac& a, const FeeFrac& b) noexcept
     135                 :             :     {
     136                 :    43723183 :         auto cross_a = Mul(a.fee, b.size), cross_b = Mul(b.fee, a.size);
     137   [ +  +  +  +  :    43723183 :         if (cross_a == cross_b) return b.size <=> a.size;
                   +  + ]
     138         [ +  + ]:        4456 :         return cross_a <=> cross_b;
     139                 :             :     }
     140                 :             : 
     141                 :             :     /** Swap two FeeFracs. */
     142                 :         152 :     friend inline void swap(FeeFrac& a, FeeFrac& b) noexcept
     143                 :             :     {
     144                 :           7 :         std::swap(a.fee, b.fee);
     145                 :           7 :         std::swap(a.size, b.size);
     146                 :             :     }
     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
        

Generated by: LCOV version 2.0-1