decimal128.hpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. /*************************************************************************
  2. *
  3. * Copyright 2019 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_DECIMAL_HPP
  19. #define REALM_DECIMAL_HPP
  20. #include <realm/string_data.hpp>
  21. #include <string>
  22. #include <cstring>
  23. #include "null.hpp"
  24. namespace realm {
  25. class Decimal128 {
  26. public:
  27. // Indicates if constructing a Decimal128 from a double should round the double to 15 digits
  28. // or 7 digits. This will make 'string -> (float/double) -> Decimal128 -> string' give the
  29. // expected result.
  30. enum class RoundTo { Digits7 = 0, Digits15 = 1 };
  31. struct Bid64 {
  32. Bid64(uint64_t x)
  33. : w(x)
  34. {
  35. }
  36. uint64_t w;
  37. };
  38. struct Bid128 {
  39. uint64_t w[2];
  40. };
  41. Decimal128() noexcept;
  42. explicit Decimal128(int64_t) noexcept;
  43. explicit Decimal128(uint64_t) noexcept;
  44. explicit Decimal128(int) noexcept;
  45. explicit Decimal128(double, RoundTo = RoundTo::Digits15) noexcept;
  46. explicit Decimal128(float val) noexcept
  47. : Decimal128(double(val), RoundTo::Digits7)
  48. {
  49. }
  50. Decimal128(Bid128 coefficient, int exponent, bool sign) noexcept;
  51. explicit Decimal128(Bid64) noexcept;
  52. explicit Decimal128(StringData) noexcept;
  53. explicit Decimal128(Bid128 val) noexcept
  54. {
  55. m_value = val;
  56. }
  57. Decimal128(null) noexcept;
  58. static Decimal128 nan(const char*) noexcept;
  59. static bool is_valid_str(StringData) noexcept;
  60. bool is_null() const noexcept;
  61. bool is_nan() const noexcept;
  62. bool to_int(int64_t& i) const noexcept;
  63. bool operator==(const Decimal128& rhs) const noexcept;
  64. bool operator!=(const Decimal128& rhs) const noexcept;
  65. bool operator<(const Decimal128& rhs) const noexcept;
  66. bool operator>(const Decimal128& rhs) const noexcept;
  67. bool operator<=(const Decimal128& rhs) const noexcept;
  68. bool operator>=(const Decimal128& rhs) const noexcept;
  69. int compare(const Decimal128& rhs) const noexcept;
  70. Decimal128 operator*(int64_t mul) const noexcept;
  71. Decimal128 operator*(size_t mul) const noexcept;
  72. Decimal128 operator*(int mul) const noexcept;
  73. Decimal128 operator*(Decimal128 mul) const noexcept;
  74. Decimal128& operator*=(Decimal128 mul) noexcept
  75. {
  76. return *this = *this * mul;
  77. }
  78. Decimal128 operator/(int64_t div) const noexcept;
  79. Decimal128 operator/(size_t div) const noexcept;
  80. Decimal128 operator/(int div) const noexcept;
  81. Decimal128 operator/(Decimal128 div) const noexcept;
  82. Decimal128& operator/=(Decimal128 div) noexcept
  83. {
  84. return *this = *this / div;
  85. }
  86. Decimal128& operator+=(Decimal128) noexcept;
  87. Decimal128 operator+(Decimal128 rhs) const noexcept
  88. {
  89. auto ret(*this);
  90. ret += rhs;
  91. return ret;
  92. }
  93. Decimal128& operator-=(Decimal128) noexcept;
  94. Decimal128 operator-(Decimal128 rhs) const noexcept
  95. {
  96. auto ret(*this);
  97. ret -= rhs;
  98. return ret;
  99. }
  100. std::string to_string() const noexcept;
  101. Bid64 to_bid64() const;
  102. const Bid128* raw() const noexcept
  103. {
  104. return &m_value;
  105. }
  106. Bid128* raw() noexcept
  107. {
  108. return &m_value;
  109. }
  110. void unpack(Bid128& coefficient, int& exponent, bool& sign) const noexcept;
  111. private:
  112. // The high word of a Decimal128 consists of 49 bit coefficient, 14 bit exponent and a sign bit
  113. static constexpr int DECIMAL_EXPONENT_BIAS_128 = 6176;
  114. static constexpr int DECIMAL_COEFF_HIGH_BITS = 49;
  115. static constexpr int DECIMAL_EXP_BITS = 14;
  116. static constexpr uint64_t MASK_COEFF = (1ull << DECIMAL_COEFF_HIGH_BITS) - 1;
  117. static constexpr uint64_t MASK_EXP = ((1ull << DECIMAL_EXP_BITS) - 1) << DECIMAL_COEFF_HIGH_BITS;
  118. static constexpr uint64_t MASK_SIGN = 1ull << (DECIMAL_COEFF_HIGH_BITS + DECIMAL_EXP_BITS);
  119. Bid128 m_value;
  120. uint64_t get_coefficient_high() const noexcept
  121. {
  122. return m_value.w[1] & MASK_COEFF;
  123. }
  124. uint64_t get_coefficient_low() const noexcept
  125. {
  126. return m_value.w[0];
  127. }
  128. };
  129. inline std::ostream& operator<<(std::ostream& ostr, const Decimal128& id)
  130. {
  131. ostr << id.to_string();
  132. return ostr;
  133. }
  134. } // namespace realm
  135. namespace std {
  136. template <>
  137. struct numeric_limits<realm::Decimal128> {
  138. static constexpr bool is_integer = false;
  139. static realm::Decimal128 lowest() noexcept
  140. {
  141. return realm::Decimal128("-Inf");
  142. }
  143. static realm::Decimal128 max() noexcept
  144. {
  145. return realm::Decimal128("+Inf");
  146. }
  147. };
  148. } // namespace std
  149. #endif /* REALM_DECIMAL_HPP */