76.88% Lines (143/186)
100.00% Functions (9/9)
| TLA | Baseline | Branch | ||||||
|---|---|---|---|---|---|---|---|---|
| Line | Hits | Code | Line | Hits | Code | |||
| 1 | // Copyright 2020-2023 Daniel Lemire | 1 | // Copyright 2020-2023 Daniel Lemire | |||||
| 2 | // Copyright 2023 Matt Borland | 2 | // Copyright 2023 Matt Borland | |||||
| 3 | // Distributed under the Boost Software License, Version 1.0. | 3 | // Distributed under the Boost Software License, Version 1.0. | |||||
| 4 | // https://www.boost.org/LICENSE_1_0.txt | 4 | // https://www.boost.org/LICENSE_1_0.txt | |||||
| 5 | // | 5 | // | |||||
| 6 | // Derivative of: https://github.com/fastfloat/fast_float | 6 | // Derivative of: https://github.com/fastfloat/fast_float | |||||
| 7 | 7 | |||||||
| 8 | #ifndef BOOST_JSON_DETAIL_CHARCONV_DETAIL_FASTFLOAT_DIGIT_COMPARISON_HPP | 8 | #ifndef BOOST_JSON_DETAIL_CHARCONV_DETAIL_FASTFLOAT_DIGIT_COMPARISON_HPP | |||||
| 9 | #define BOOST_JSON_DETAIL_CHARCONV_DETAIL_FASTFLOAT_DIGIT_COMPARISON_HPP | 9 | #define BOOST_JSON_DETAIL_CHARCONV_DETAIL_FASTFLOAT_DIGIT_COMPARISON_HPP | |||||
| 10 | 10 | |||||||
| 11 | #include <boost/json/detail/charconv/detail/fast_float/float_common.hpp> | 11 | #include <boost/json/detail/charconv/detail/fast_float/float_common.hpp> | |||||
| 12 | #include <boost/json/detail/charconv/detail/fast_float/bigint.hpp> | 12 | #include <boost/json/detail/charconv/detail/fast_float/bigint.hpp> | |||||
| 13 | #include <boost/json/detail/charconv/detail/fast_float/ascii_number.hpp> | 13 | #include <boost/json/detail/charconv/detail/fast_float/ascii_number.hpp> | |||||
| 14 | #include <algorithm> | 14 | #include <algorithm> | |||||
| 15 | #include <cstdint> | 15 | #include <cstdint> | |||||
| 16 | #include <cstring> | 16 | #include <cstring> | |||||
| 17 | #include <iterator> | 17 | #include <iterator> | |||||
| 18 | 18 | |||||||
| 19 | namespace boost { namespace json { namespace detail { namespace charconv { namespace detail { namespace fast_float { | 19 | namespace boost { namespace json { namespace detail { namespace charconv { namespace detail { namespace fast_float { | |||||
| 20 | 20 | |||||||
| 21 | // 1e0 to 1e19 | 21 | // 1e0 to 1e19 | |||||
| 22 | constexpr static uint64_t powers_of_ten_uint64[] = { | 22 | constexpr static uint64_t powers_of_ten_uint64[] = { | |||||
| 23 | 1UL, 10UL, 100UL, 1000UL, 10000UL, 100000UL, 1000000UL, 10000000UL, 100000000UL, | 23 | 1UL, 10UL, 100UL, 1000UL, 10000UL, 100000UL, 1000000UL, 10000000UL, 100000000UL, | |||||
| 24 | 1000000000UL, 10000000000UL, 100000000000UL, 1000000000000UL, 10000000000000UL, | 24 | 1000000000UL, 10000000000UL, 100000000000UL, 1000000000000UL, 10000000000000UL, | |||||
| 25 | 100000000000000UL, 1000000000000000UL, 10000000000000000UL, 100000000000000000UL, | 25 | 100000000000000UL, 1000000000000000UL, 10000000000000000UL, 100000000000000000UL, | |||||
| 26 | 1000000000000000000UL, 10000000000000000000UL}; | 26 | 1000000000000000000UL, 10000000000000000000UL}; | |||||
| 27 | 27 | |||||||
| 28 | // calculate the exponent, in scientific notation, of the number. | 28 | // calculate the exponent, in scientific notation, of the number. | |||||
| 29 | // this algorithm is not even close to optimized, but it has no practical | 29 | // this algorithm is not even close to optimized, but it has no practical | |||||
| 30 | // effect on performance: in order to have a faster algorithm, we'd need | 30 | // effect on performance: in order to have a faster algorithm, we'd need | |||||
| 31 | // to slow down performance for faster algorithms, and this is still fast. | 31 | // to slow down performance for faster algorithms, and this is still fast. | |||||
| 32 | template <typename UC> | 32 | template <typename UC> | |||||
| 33 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | 33 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | |||||
| 34 | int32_t scientific_exponent(parsed_number_string_t<UC> & num) noexcept { | 34 | int32_t scientific_exponent(parsed_number_string_t<UC> & num) noexcept { | |||||
| HITCBC | 35 | 2986 | uint64_t mantissa = num.mantissa; | 35 | 2986 | uint64_t mantissa = num.mantissa; | ||
| HITCBC | 36 | 2986 | int32_t exponent = int32_t(num.exponent); | 36 | 2986 | int32_t exponent = int32_t(num.exponent); | ||
| HITCBC | 37 | 14930 | while (mantissa >= 10000) { | 37 | 14930 | while (mantissa >= 10000) { | ||
| HITCBC | 38 | 11944 | mantissa /= 10000; | 38 | 11944 | mantissa /= 10000; | ||
| HITCBC | 39 | 11944 | exponent += 4; | 39 | 11944 | exponent += 4; | ||
| 40 | } | 40 | } | |||||
| HITCBC | 41 | 5972 | while (mantissa >= 100) { | 41 | 5972 | while (mantissa >= 100) { | ||
| HITCBC | 42 | 2986 | mantissa /= 100; | 42 | 2986 | mantissa /= 100; | ||
| HITCBC | 43 | 2986 | exponent += 2; | 43 | 2986 | exponent += 2; | ||
| 44 | } | 44 | } | |||||
| HITCBC | 45 | 2986 | while (mantissa >= 10) { | 45 | 2986 | while (mantissa >= 10) { | ||
| MISUBC | 46 | ✗ | mantissa /= 10; | 46 | ✗ | mantissa /= 10; | ||
| MISUBC | 47 | ✗ | exponent += 1; | 47 | ✗ | exponent += 1; | ||
| 48 | } | 48 | } | |||||
| HITCBC | 49 | 2986 | return exponent; | 49 | 2986 | return exponent; | ||
| 50 | } | 50 | } | |||||
| 51 | 51 | |||||||
| 52 | // this converts a native floating-point number to an extended-precision float. | 52 | // this converts a native floating-point number to an extended-precision float. | |||||
| 53 | template <typename T> | 53 | template <typename T> | |||||
| 54 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 54 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 55 | adjusted_mantissa to_extended(T value) noexcept { | 55 | adjusted_mantissa to_extended(T value) noexcept { | |||||
| 56 | using equiv_uint = typename binary_format<T>::equiv_uint; | 56 | using equiv_uint = typename binary_format<T>::equiv_uint; | |||||
| HITCBC | 57 | 1611 | constexpr equiv_uint exponent_mask = binary_format<T>::exponent_mask(); | 57 | 1611 | constexpr equiv_uint exponent_mask = binary_format<T>::exponent_mask(); | ||
| HITCBC | 58 | 1611 | constexpr equiv_uint mantissa_mask = binary_format<T>::mantissa_mask(); | 58 | 1611 | constexpr equiv_uint mantissa_mask = binary_format<T>::mantissa_mask(); | ||
| HITCBC | 59 | 1611 | constexpr equiv_uint hidden_bit_mask = binary_format<T>::hidden_bit_mask(); | 59 | 1611 | constexpr equiv_uint hidden_bit_mask = binary_format<T>::hidden_bit_mask(); | ||
| 60 | 60 | |||||||
| HITCBC | 61 | 1611 | adjusted_mantissa am; | 61 | 1611 | adjusted_mantissa am; | ||
| HITCBC | 62 | 1611 | int32_t bias = binary_format<T>::mantissa_explicit_bits() - binary_format<T>::minimum_exponent(); | 62 | 1611 | int32_t bias = binary_format<T>::mantissa_explicit_bits() - binary_format<T>::minimum_exponent(); | ||
| 63 | equiv_uint bits; | 63 | equiv_uint bits; | |||||
| 64 | #ifdef BOOST_JSON_HAS_BIT_CAST | 64 | #ifdef BOOST_JSON_HAS_BIT_CAST | |||||
| 65 | bits = std::bit_cast<equiv_uint>(value); | 65 | bits = std::bit_cast<equiv_uint>(value); | |||||
| 66 | #else | 66 | #else | |||||
| HITCBC | 67 | 1611 | ::memcpy(&bits, &value, sizeof(T)); | 67 | 1611 | ::memcpy(&bits, &value, sizeof(T)); | ||
| 68 | #endif | 68 | #endif | |||||
| HITCBC | 69 | 1611 | if ((bits & exponent_mask) == 0) { | 69 | 1611 | if ((bits & exponent_mask) == 0) { | ||
| 70 | // denormal | 70 | // denormal | |||||
| MISUBC | 71 | ✗ | am.power2 = 1 - bias; | 71 | ✗ | am.power2 = 1 - bias; | ||
| MISUBC | 72 | ✗ | am.mantissa = bits & mantissa_mask; | 72 | ✗ | am.mantissa = bits & mantissa_mask; | ||
| 73 | } else { | 73 | } else { | |||||
| 74 | // normal | 74 | // normal | |||||
| HITCBC | 75 | 1611 | am.power2 = int32_t((bits & exponent_mask) >> binary_format<T>::mantissa_explicit_bits()); | 75 | 1611 | am.power2 = int32_t((bits & exponent_mask) >> binary_format<T>::mantissa_explicit_bits()); | ||
| HITCBC | 76 | 1611 | am.power2 -= bias; | 76 | 1611 | am.power2 -= bias; | ||
| HITCBC | 77 | 1611 | am.mantissa = (bits & mantissa_mask) | hidden_bit_mask; | 77 | 1611 | am.mantissa = (bits & mantissa_mask) | hidden_bit_mask; | ||
| 78 | } | 78 | } | |||||
| 79 | 79 | |||||||
| HITCBC | 80 | 1611 | return am; | 80 | 1611 | return am; | ||
| 81 | } | 81 | } | |||||
| 82 | 82 | |||||||
| 83 | // get the extended precision value of the halfway point between b and b+u. | 83 | // get the extended precision value of the halfway point between b and b+u. | |||||
| 84 | // we are given a native float that represents b, so we need to adjust it | 84 | // we are given a native float that represents b, so we need to adjust it | |||||
| 85 | // halfway between b and b+u. | 85 | // halfway between b and b+u. | |||||
| 86 | template <typename T> | 86 | template <typename T> | |||||
| 87 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 87 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 88 | adjusted_mantissa to_extended_halfway(T value) noexcept { | 88 | adjusted_mantissa to_extended_halfway(T value) noexcept { | |||||
| HITCBC | 89 | 1611 | adjusted_mantissa am = to_extended(value); | 89 | 1611 | adjusted_mantissa am = to_extended(value); | ||
| HITCBC | 90 | 1611 | am.mantissa <<= 1; | 90 | 1611 | am.mantissa <<= 1; | ||
| HITCBC | 91 | 1611 | am.mantissa += 1; | 91 | 1611 | am.mantissa += 1; | ||
| HITCBC | 92 | 1611 | am.power2 -= 1; | 92 | 1611 | am.power2 -= 1; | ||
| HITCBC | 93 | 1611 | return am; | 93 | 1611 | return am; | ||
| 94 | } | 94 | } | |||||
| 95 | 95 | |||||||
| 96 | // round an extended-precision float to the nearest machine float. | 96 | // round an extended-precision float to the nearest machine float. | |||||
| 97 | template <typename T, typename callback> | 97 | template <typename T, typename callback> | |||||
| 98 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | 98 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | |||||
| 99 | void round(adjusted_mantissa& am, callback cb) noexcept { | 99 | void round(adjusted_mantissa& am, callback cb) noexcept { | |||||
| HITCBC | 100 | 4597 | int32_t mantissa_shift = 64 - binary_format<T>::mantissa_explicit_bits() - 1; | 100 | 4597 | int32_t mantissa_shift = 64 - binary_format<T>::mantissa_explicit_bits() - 1; | ||
| HITCBC | 101 | 4597 | if (-am.power2 >= mantissa_shift) { | 101 | 4597 | if (-am.power2 >= mantissa_shift) { | ||
| 102 | // have a denormal float | 102 | // have a denormal float | |||||
| MISUBC | 103 | ✗ | int32_t shift = -am.power2 + 1; | 103 | ✗ | int32_t shift = -am.power2 + 1; | ||
| MISUBC | 104 | ✗ | cb(am, std::min<int32_t>(shift, 64)); | 104 | ✗ | cb(am, std::min<int32_t>(shift, 64)); | ||
| 105 | // check for round-up: if rounding-nearest carried us to the hidden bit. | 105 | // check for round-up: if rounding-nearest carried us to the hidden bit. | |||||
| MISUBC | 106 | ✗ | am.power2 = (am.mantissa < (uint64_t(1) << binary_format<T>::mantissa_explicit_bits())) ? 0 : 1; | 106 | ✗ | am.power2 = (am.mantissa < (uint64_t(1) << binary_format<T>::mantissa_explicit_bits())) ? 0 : 1; | ||
| MISUBC | 107 | ✗ | return; | 107 | ✗ | return; | ||
| 108 | } | 108 | } | |||||
| 109 | 109 | |||||||
| 110 | // have a normal float, use the default shift. | 110 | // have a normal float, use the default shift. | |||||
| HITCBC | 111 | 4597 | cb(am, mantissa_shift); | 111 | 4597 | cb(am, mantissa_shift); | ||
| 112 | 112 | |||||||
| 113 | // check for carry | 113 | // check for carry | |||||
| HITCBC | 114 | 4597 | if (am.mantissa >= (uint64_t(2) << binary_format<T>::mantissa_explicit_bits())) { | 114 | 4597 | if (am.mantissa >= (uint64_t(2) << binary_format<T>::mantissa_explicit_bits())) { | ||
| MISUBC | 115 | ✗ | am.mantissa = (uint64_t(1) << binary_format<T>::mantissa_explicit_bits()); | 115 | ✗ | am.mantissa = (uint64_t(1) << binary_format<T>::mantissa_explicit_bits()); | ||
| MISUBC | 116 | ✗ | am.power2++; | 116 | ✗ | am.power2++; | ||
| 117 | } | 117 | } | |||||
| 118 | 118 | |||||||
| 119 | // check for infinite: we could have carried to an infinite power | 119 | // check for infinite: we could have carried to an infinite power | |||||
| HITCBC | 120 | 4597 | am.mantissa &= ~(uint64_t(1) << binary_format<T>::mantissa_explicit_bits()); | 120 | 4597 | am.mantissa &= ~(uint64_t(1) << binary_format<T>::mantissa_explicit_bits()); | ||
| HITCBC | 121 | 4597 | if (am.power2 >= binary_format<T>::infinite_power()) { | 121 | 4597 | if (am.power2 >= binary_format<T>::infinite_power()) { | ||
| MISUBC | 122 | ✗ | am.power2 = binary_format<T>::infinite_power(); | 122 | ✗ | am.power2 = binary_format<T>::infinite_power(); | ||
| MISUBC | 123 | ✗ | am.mantissa = 0; | 123 | ✗ | am.mantissa = 0; | ||
| 124 | } | 124 | } | |||||
| 125 | } | 125 | } | |||||
| 126 | 126 | |||||||
| 127 | template <typename callback> | 127 | template <typename callback> | |||||
| 128 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | 128 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | |||||
| 129 | void round_nearest_tie_even(adjusted_mantissa& am, int32_t shift, callback cb) noexcept { | 129 | void round_nearest_tie_even(adjusted_mantissa& am, int32_t shift, callback cb) noexcept { | |||||
| HITCBC | 130 | 2986 | const uint64_t mask | 130 | 2986 | const uint64_t mask | ||
| 131 | = (shift == 64) | 131 | = (shift == 64) | |||||
| MISUBC | 132 | ✗ | ? UINT64_MAX | 132 | ✗ | ? UINT64_MAX | ||
| HITCBC | 133 | 2986 | : (uint64_t(1) << shift) - 1; | 133 | 2986 | : (uint64_t(1) << shift) - 1; | ||
| HITCBC | 134 | 2986 | const uint64_t halfway | 134 | 2986 | const uint64_t halfway | ||
| 135 | = (shift == 0) | 135 | = (shift == 0) | |||||
| HITCBC | 136 | 2986 | ? 0 | 136 | 2986 | ? 0 | ||
| HITCBC | 137 | 2986 | : uint64_t(1) << (shift - 1); | 137 | 2986 | : uint64_t(1) << (shift - 1); | ||
| HITCBC | 138 | 2986 | uint64_t truncated_bits = am.mantissa & mask; | 138 | 2986 | uint64_t truncated_bits = am.mantissa & mask; | ||
| HITCBC | 139 | 2986 | bool is_above = truncated_bits > halfway; | 139 | 2986 | bool is_above = truncated_bits > halfway; | ||
| HITCBC | 140 | 2986 | bool is_halfway = truncated_bits == halfway; | 140 | 2986 | bool is_halfway = truncated_bits == halfway; | ||
| 141 | 141 | |||||||
| 142 | // shift digits into position | 142 | // shift digits into position | |||||
| HITCBC | 143 | 2986 | if (shift == 64) { | 143 | 2986 | if (shift == 64) { | ||
| MISUBC | 144 | ✗ | am.mantissa = 0; | 144 | ✗ | am.mantissa = 0; | ||
| 145 | } else { | 145 | } else { | |||||
| HITCBC | 146 | 2986 | am.mantissa >>= shift; | 146 | 2986 | am.mantissa >>= shift; | ||
| 147 | } | 147 | } | |||||
| HITCBC | 148 | 2986 | am.power2 += shift; | 148 | 2986 | am.power2 += shift; | ||
| 149 | 149 | |||||||
| HITCBC | 150 | 2986 | bool is_odd = (am.mantissa & 1) == 1; | 150 | 2986 | bool is_odd = (am.mantissa & 1) == 1; | ||
| HITCBC | 151 | 2986 | am.mantissa += uint64_t(cb(is_odd, is_halfway, is_above)); | 151 | 2986 | am.mantissa += uint64_t(cb(is_odd, is_halfway, is_above)); | ||
| HITCBC | 152 | 2986 | } | 152 | 2986 | } | ||
| 153 | 153 | |||||||
| 154 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | 154 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | |||||
| 155 | void round_down(adjusted_mantissa& am, int32_t shift) noexcept { | 155 | void round_down(adjusted_mantissa& am, int32_t shift) noexcept { | |||||
| HITCBC | 156 | 1611 | if (shift == 64) { | 156 | 1611 | if (shift == 64) { | ||
| MISUBC | 157 | ✗ | am.mantissa = 0; | 157 | ✗ | am.mantissa = 0; | ||
| 158 | } else { | 158 | } else { | |||||
| HITCBC | 159 | 1611 | am.mantissa >>= shift; | 159 | 1611 | am.mantissa >>= shift; | ||
| 160 | } | 160 | } | |||||
| HITCBC | 161 | 1611 | am.power2 += shift; | 161 | 1611 | am.power2 += shift; | ||
| HITCBC | 162 | 1611 | } | 162 | 1611 | } | ||
| 163 | template <typename UC> | 163 | template <typename UC> | |||||
| 164 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 164 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 165 | void skip_zeros(UC const * & first, UC const * last) noexcept { | 165 | void skip_zeros(UC const * & first, UC const * last) noexcept { | |||||
| 166 | uint64_t val; | 166 | uint64_t val; | |||||
| HITCBC | 167 | 5972 | while (!cpp20_and_in_constexpr() && std::distance(first, last) >= int_cmp_len<UC>()) { | 167 | 5972 | while (!cpp20_and_in_constexpr() && std::distance(first, last) >= int_cmp_len<UC>()) { | ||
| HITCBC | 168 | 2986 | ::memcpy(&val, first, sizeof(uint64_t)); | 168 | 2986 | ::memcpy(&val, first, sizeof(uint64_t)); | ||
| HITCBC | 169 | 2986 | if (val != int_cmp_zeros<UC>()) { | 169 | 2986 | if (val != int_cmp_zeros<UC>()) { | ||
| HITCBC | 170 | 2986 | break; | 170 | 2986 | break; | ||
| 171 | } | 171 | } | |||||
| MISUBC | 172 | ✗ | first += int_cmp_len<UC>(); | 172 | ✗ | first += int_cmp_len<UC>(); | ||
| 173 | } | 173 | } | |||||
| HITCBC | 174 | 2986 | while (first != last) { | 174 | 2986 | while (first != last) { | ||
| HITCBC | 175 | 2986 | if (*first != UC('0')) { | 175 | 2986 | if (*first != UC('0')) { | ||
| HITCBC | 176 | 2986 | break; | 176 | 2986 | break; | ||
| 177 | } | 177 | } | |||||
| MISUBC | 178 | ✗ | first++; | 178 | ✗ | first++; | ||
| 179 | } | 179 | } | |||||
| HITCBC | 180 | 2986 | } | 180 | 2986 | } | ||
| 181 | 181 | |||||||
| 182 | // determine if any non-zero digits were truncated. | 182 | // determine if any non-zero digits were truncated. | |||||
| 183 | // all characters must be valid digits. | 183 | // all characters must be valid digits. | |||||
| 184 | template <typename UC> | 184 | template <typename UC> | |||||
| 185 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 185 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 186 | bool is_truncated(UC const * first, UC const * last) noexcept { | 186 | bool is_truncated(UC const * first, UC const * last) noexcept { | |||||
| 187 | // do 8-bit optimizations, can just compare to 8 literal 0s. | 187 | // do 8-bit optimizations, can just compare to 8 literal 0s. | |||||
| 188 | uint64_t val; | 188 | uint64_t val; | |||||
| MISUBC | 189 | ✗ | while (!cpp20_and_in_constexpr() && std::distance(first, last) >= int_cmp_len<UC>()) { | 189 | ✗ | while (!cpp20_and_in_constexpr() && std::distance(first, last) >= int_cmp_len<UC>()) { | ||
| MISUBC | 190 | ✗ | ::memcpy(&val, first, sizeof(uint64_t)); | 190 | ✗ | ::memcpy(&val, first, sizeof(uint64_t)); | ||
| MISUBC | 191 | ✗ | if (val != int_cmp_zeros<UC>()) { | 191 | ✗ | if (val != int_cmp_zeros<UC>()) { | ||
| MISUBC | 192 | ✗ | return true; | 192 | ✗ | return true; | ||
| 193 | } | 193 | } | |||||
| MISUBC | 194 | ✗ | first += int_cmp_len<UC>(); | 194 | ✗ | first += int_cmp_len<UC>(); | ||
| 195 | } | 195 | } | |||||
| MISUBC | 196 | ✗ | while (first != last) { | 196 | ✗ | while (first != last) { | ||
| MISUBC | 197 | ✗ | if (*first != UC('0')) { | 197 | ✗ | if (*first != UC('0')) { | ||
| MISUBC | 198 | ✗ | return true; | 198 | ✗ | return true; | ||
| 199 | } | 199 | } | |||||
| MISUBC | 200 | ✗ | ++first; | 200 | ✗ | ++first; | ||
| 201 | } | 201 | } | |||||
| MISUBC | 202 | ✗ | return false; | 202 | ✗ | return false; | ||
| 203 | } | 203 | } | |||||
| 204 | template <typename UC> | 204 | template <typename UC> | |||||
| 205 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 205 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 206 | bool is_truncated(span<const UC> s) noexcept { | 206 | bool is_truncated(span<const UC> s) noexcept { | |||||
| MISUBC | 207 | ✗ | return is_truncated(s.ptr, s.ptr + s.len()); | 207 | ✗ | return is_truncated(s.ptr, s.ptr + s.len()); | ||
| 208 | } | 208 | } | |||||
| 209 | 209 | |||||||
| 210 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 210 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 211 | void parse_eight_digits(const char16_t*& , limb& , size_t& , size_t& ) noexcept { | 211 | void parse_eight_digits(const char16_t*& , limb& , size_t& , size_t& ) noexcept { | |||||
| 212 | // currently unused | 212 | // currently unused | |||||
| 213 | } | 213 | } | |||||
| 214 | 214 | |||||||
| 215 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 215 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 216 | void parse_eight_digits(const char32_t*& , limb& , size_t& , size_t& ) noexcept { | 216 | void parse_eight_digits(const char32_t*& , limb& , size_t& , size_t& ) noexcept { | |||||
| 217 | // currently unused | 217 | // currently unused | |||||
| 218 | } | 218 | } | |||||
| 219 | 219 | |||||||
| 220 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 220 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 221 | void parse_eight_digits(const char*& p, limb& value, size_t& counter, size_t& count) noexcept { | 221 | void parse_eight_digits(const char*& p, limb& value, size_t& counter, size_t& count) noexcept { | |||||
| HITCBC | 222 | 11932 | value = value * 100000000 + parse_eight_digits_unrolled(p); | 222 | 11932 | value = value * 100000000 + parse_eight_digits_unrolled(p); | ||
| HITCBC | 223 | 11932 | p += 8; | 223 | 11932 | p += 8; | ||
| HITCBC | 224 | 11932 | counter += 8; | 224 | 11932 | counter += 8; | ||
| HITCBC | 225 | 11932 | count += 8; | 225 | 11932 | count += 8; | ||
| HITCBC | 226 | 11932 | } | 226 | 11932 | } | ||
| 227 | 227 | |||||||
| 228 | template <typename UC> | 228 | template <typename UC> | |||||
| 229 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | 229 | BOOST_FORCEINLINE BOOST_JSON_CXX14_CONSTEXPR_NO_INLINE | |||||
| 230 | void parse_one_digit(UC const *& p, limb& value, size_t& counter, size_t& count) noexcept { | 230 | void parse_one_digit(UC const *& p, limb& value, size_t& counter, size_t& count) noexcept { | |||||
| HITCBC | 231 | 21125 | value = value * 10 + limb(*p - UC('0')); | 231 | 21125 | value = value * 10 + limb(*p - UC('0')); | ||
| HITCBC | 232 | 21125 | p++; | 232 | 21125 | p++; | ||
| HITCBC | 233 | 21125 | counter++; | 233 | 21125 | counter++; | ||
| HITCBC | 234 | 21125 | count++; | 234 | 21125 | count++; | ||
| HITCBC | 235 | 21125 | } | 235 | 21125 | } | ||
| 236 | 236 | |||||||
| 237 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 237 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 238 | void add_native(bigint& big, limb power, limb value) noexcept { | 238 | void add_native(bigint& big, limb power, limb value) noexcept { | |||||
| MISUBC | 239 | ✗ | big.mul(power); | 239 | ✗ | big.mul(power); | ||
| HITCBC | 240 | 9437 | big.add(value); | 240 | 9437 | big.add(value); | ||
| HITCBC | 241 | 9437 | } | 241 | 9437 | } | ||
| 242 | 242 | |||||||
| 243 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 243 | BOOST_FORCEINLINE BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| 244 | void round_up_bigint(bigint& big, size_t& count) noexcept { | 244 | void round_up_bigint(bigint& big, size_t& count) noexcept { | |||||
| 245 | // need to round-up the digits, but need to avoid rounding | 245 | // need to round-up the digits, but need to avoid rounding | |||||
| 246 | // ....9999 to ...10000, which could cause a false halfway point. | 246 | // ....9999 to ...10000, which could cause a false halfway point. | |||||
| 247 | add_native(big, 10, 1); | 247 | add_native(big, 10, 1); | |||||
| MISUBC | 248 | ✗ | count++; | 248 | ✗ | count++; | ||
| MISUBC | 249 | ✗ | } | 249 | ✗ | } | ||
| 250 | 250 | |||||||
| 251 | // parse the significant digits into a big integer | 251 | // parse the significant digits into a big integer | |||||
| 252 | template <typename UC> | 252 | template <typename UC> | |||||
| 253 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 253 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| HITCBC | 254 | 2986 | void parse_mantissa(bigint& result, parsed_number_string_t<UC>& num, size_t max_digits, size_t& digits) noexcept { | 254 | 2986 | void parse_mantissa(bigint& result, parsed_number_string_t<UC>& num, size_t max_digits, size_t& digits) noexcept { | ||
| 255 | // try to minimize the number of big integer and scalar multiplication. | 255 | // try to minimize the number of big integer and scalar multiplication. | |||||
| 256 | // therefore, try to parse 8 digits at a time, and multiply by the largest | 256 | // therefore, try to parse 8 digits at a time, and multiply by the largest | |||||
| 257 | // scalar value (9 or 19 digits) for each step. | 257 | // scalar value (9 or 19 digits) for each step. | |||||
| HITCBC | 258 | 2986 | size_t counter = 0; | 258 | 2986 | size_t counter = 0; | ||
| HITCBC | 259 | 2986 | digits = 0; | 259 | 2986 | digits = 0; | ||
| HITCBC | 260 | 2986 | limb value = 0; | 260 | 2986 | limb value = 0; | ||
| 261 | #ifdef BOOST_JSON_FASTFLOAT_64BIT_LIMB | 261 | #ifdef BOOST_JSON_FASTFLOAT_64BIT_LIMB | |||||
| HITCBC | 262 | 2986 | constexpr size_t step = 19; | 262 | 2986 | constexpr size_t step = 19; | ||
| 263 | #else | 263 | #else | |||||
| 264 | constexpr size_t step = 9; | 264 | constexpr size_t step = 9; | |||||
| 265 | #endif | 265 | #endif | |||||
| 266 | 266 | |||||||
| 267 | // process all integer digits. | 267 | // process all integer digits. | |||||
| HITCBC | 268 | 2986 | UC const * p = num.integer.ptr; | 268 | 2986 | UC const * p = num.integer.ptr; | ||
| HITCBC | 269 | 2986 | UC const * pend = p + num.integer.len(); | 269 | 2986 | UC const * pend = p + num.integer.len(); | ||
| 270 | skip_zeros(p, pend); | 270 | skip_zeros(p, pend); | |||||
| 271 | // process all digits, in increments of step per loop | 271 | // process all digits, in increments of step per loop | |||||
| HITCBC | 272 | 8073 | while (p != pend) { | 272 | 8073 | while (p != pend) { | ||
| 273 | if (std::is_same<UC,char>::value) { | 273 | if (std::is_same<UC,char>::value) { | |||||
| HITCBC | 274 | 22118 | while ((std::distance(p, pend) >= 8) && (step - counter >= 8) && (max_digits - digits >= 8)) { | 274 | 22118 | while ((std::distance(p, pend) >= 8) && (step - counter >= 8) && (max_digits - digits >= 8)) { | ||
| 275 | parse_eight_digits(p, value, counter, digits); | 275 | parse_eight_digits(p, value, counter, digits); | |||||
| 276 | } | 276 | } | |||||
| 277 | } | 277 | } | |||||
| HITCBC | 278 | 16053 | while (counter < step && p != pend && digits < max_digits) { | 278 | 16053 | while (counter < step && p != pend && digits < max_digits) { | ||
| 279 | parse_one_digit(p, value, counter, digits); | 279 | parse_one_digit(p, value, counter, digits); | |||||
| 280 | } | 280 | } | |||||
| HITCBC | 281 | 5087 | if (digits == max_digits) { | 281 | 5087 | if (digits == max_digits) { | ||
| 282 | // add the temporary value, then check if we've truncated any digits | 282 | // add the temporary value, then check if we've truncated any digits | |||||
| MISUBC | 283 | ✗ | add_native(result, limb(powers_of_ten_uint64[counter]), value); | 283 | ✗ | add_native(result, limb(powers_of_ten_uint64[counter]), value); | ||
| MISUBC | 284 | ✗ | bool truncated = is_truncated(p, pend); | 284 | ✗ | bool truncated = is_truncated(p, pend); | ||
| MISUBC | 285 | ✗ | if (num.fraction.ptr != nullptr) { | 285 | ✗ | if (num.fraction.ptr != nullptr) { | ||
| MISUBC | 286 | ✗ | truncated |= is_truncated(num.fraction); | 286 | ✗ | truncated |= is_truncated(num.fraction); | ||
| 287 | } | 287 | } | |||||
| MISUBC | 288 | ✗ | if (truncated) { | 288 | ✗ | if (truncated) { | ||
| 289 | round_up_bigint(result, digits); | 289 | round_up_bigint(result, digits); | |||||
| 290 | } | 290 | } | |||||
| MISUBC | 291 | ✗ | return; | 291 | ✗ | return; | ||
| 292 | } else { | 292 | } else { | |||||
| HITCBC | 293 | 5087 | add_native(result, limb(powers_of_ten_uint64[counter]), value); | 293 | 5087 | add_native(result, limb(powers_of_ten_uint64[counter]), value); | ||
| HITCBC | 294 | 5087 | counter = 0; | 294 | 5087 | counter = 0; | ||
| HITCBC | 295 | 5087 | value = 0; | 295 | 5087 | value = 0; | ||
| 296 | } | 296 | } | |||||
| 297 | } | 297 | } | |||||
| 298 | 298 | |||||||
| 299 | // add our fraction digits, if they're available. | 299 | // add our fraction digits, if they're available. | |||||
| HITCBC | 300 | 2986 | if (num.fraction.ptr != nullptr) { | 300 | 2986 | if (num.fraction.ptr != nullptr) { | ||
| HITCBC | 301 | 2980 | p = num.fraction.ptr; | 301 | 2980 | p = num.fraction.ptr; | ||
| HITCBC | 302 | 2980 | pend = p + num.fraction.len(); | 302 | 2980 | pend = p + num.fraction.len(); | ||
| HITCBC | 303 | 2980 | if (digits == 0) { | 303 | 2980 | if (digits == 0) { | ||
| 304 | skip_zeros(p, pend); | 304 | skip_zeros(p, pend); | |||||
| 305 | } | 305 | } | |||||
| 306 | // process all digits, in increments of step per loop | 306 | // process all digits, in increments of step per loop | |||||
| HITCBC | 307 | 7330 | while (p != pend) { | 307 | 7330 | while (p != pend) { | ||
| 308 | if (std::is_same<UC,char>::value) { | 308 | if (std::is_same<UC,char>::value) { | |||||
| HITCBC | 309 | 20620 | while ((std::distance(p, pend) >= 8) && (step - counter >= 8) && (max_digits - digits >= 8)) { | 309 | 20620 | while ((std::distance(p, pend) >= 8) && (step - counter >= 8) && (max_digits - digits >= 8)) { | ||
| 310 | parse_eight_digits(p, value, counter, digits); | 310 | parse_eight_digits(p, value, counter, digits); | |||||
| 311 | } | 311 | } | |||||
| 312 | } | 312 | } | |||||
| HITCBC | 313 | 14509 | while (counter < step && p != pend && digits < max_digits) { | 313 | 14509 | while (counter < step && p != pend && digits < max_digits) { | ||
| 314 | parse_one_digit(p, value, counter, digits); | 314 | parse_one_digit(p, value, counter, digits); | |||||
| 315 | } | 315 | } | |||||
| HITCBC | 316 | 4350 | if (digits == max_digits) { | 316 | 4350 | if (digits == max_digits) { | ||
| 317 | // add the temporary value, then check if we've truncated any digits | 317 | // add the temporary value, then check if we've truncated any digits | |||||
| MISUBC | 318 | ✗ | add_native(result, limb(powers_of_ten_uint64[counter]), value); | 318 | ✗ | add_native(result, limb(powers_of_ten_uint64[counter]), value); | ||
| MISUBC | 319 | ✗ | bool truncated = is_truncated(p, pend); | 319 | ✗ | bool truncated = is_truncated(p, pend); | ||
| MISUBC | 320 | ✗ | if (truncated) { | 320 | ✗ | if (truncated) { | ||
| 321 | round_up_bigint(result, digits); | 321 | round_up_bigint(result, digits); | |||||
| 322 | } | 322 | } | |||||
| MISUBC | 323 | ✗ | return; | 323 | ✗ | return; | ||
| 324 | } else { | 324 | } else { | |||||
| HITCBC | 325 | 4350 | add_native(result, limb(powers_of_ten_uint64[counter]), value); | 325 | 4350 | add_native(result, limb(powers_of_ten_uint64[counter]), value); | ||
| HITCBC | 326 | 4350 | counter = 0; | 326 | 4350 | counter = 0; | ||
| HITCBC | 327 | 4350 | value = 0; | 327 | 4350 | value = 0; | ||
| 328 | } | 328 | } | |||||
| 329 | } | 329 | } | |||||
| 330 | } | 330 | } | |||||
| 331 | 331 | |||||||
| HITCBC | 332 | 2986 | if (counter != 0) { | 332 | 2986 | if (counter != 0) { | ||
| MISUBC | 333 | ✗ | add_native(result, limb(powers_of_ten_uint64[counter]), value); | 333 | ✗ | add_native(result, limb(powers_of_ten_uint64[counter]), value); | ||
| 334 | } | 334 | } | |||||
| 335 | } | 335 | } | |||||
| 336 | 336 | |||||||
| 337 | template <typename T> | 337 | template <typename T> | |||||
| 338 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 338 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| HITCBC | 339 | 1375 | adjusted_mantissa positive_digit_comp(bigint& bigmant, int32_t exponent) noexcept { | 339 | 1375 | adjusted_mantissa positive_digit_comp(bigint& bigmant, int32_t exponent) noexcept { | ||
| HITCBC | 340 | 1375 | bigmant.pow10(uint32_t(exponent)); | 340 | 1375 | bigmant.pow10(uint32_t(exponent)); | ||
| HITCBC | 341 | 1375 | adjusted_mantissa answer; | 341 | 1375 | adjusted_mantissa answer; | ||
| 342 | bool truncated; | 342 | bool truncated; | |||||
| HITCBC | 343 | 1375 | answer.mantissa = bigmant.hi64(truncated); | 343 | 1375 | answer.mantissa = bigmant.hi64(truncated); | ||
| HITCBC | 344 | 1375 | int bias = binary_format<T>::mantissa_explicit_bits() - binary_format<T>::minimum_exponent(); | 344 | 1375 | int bias = binary_format<T>::mantissa_explicit_bits() - binary_format<T>::minimum_exponent(); | ||
| HITCBC | 345 | 1375 | answer.power2 = bigmant.bit_length() - 64 + bias; | 345 | 1375 | answer.power2 = bigmant.bit_length() - 64 + bias; | ||
| 346 | 346 | |||||||
| HITCBC | 347 | 1375 | round<T>(answer, [truncated](adjusted_mantissa& a, int32_t shift) { | 347 | 1375 | round<T>(answer, [truncated](adjusted_mantissa& a, int32_t shift) { | ||
| HITCBC | 348 | 1375 | round_nearest_tie_even(a, shift, [truncated](bool is_odd, bool is_halfway, bool is_above) -> bool { | 348 | 1375 | round_nearest_tie_even(a, shift, [truncated](bool is_odd, bool is_halfway, bool is_above) -> bool { | ||
| HITCBC | 349 | 1375 | return is_above || (is_halfway && truncated) || (is_odd && is_halfway); | 349 | 1375 | return is_above || (is_halfway && truncated) || (is_odd && is_halfway); | ||
| 350 | }); | 350 | }); | |||||
| 351 | }); | 351 | }); | |||||
| 352 | 352 | |||||||
| HITCBC | 353 | 1375 | return answer; | 353 | 1375 | return answer; | ||
| 354 | } | 354 | } | |||||
| 355 | 355 | |||||||
| 356 | // the scaling here is quite simple: we have, for the real digits `m * 10^e`, | 356 | // the scaling here is quite simple: we have, for the real digits `m * 10^e`, | |||||
| 357 | // and for the theoretical digits `n * 2^f`. Since `e` is always negative, | 357 | // and for the theoretical digits `n * 2^f`. Since `e` is always negative, | |||||
| 358 | // to scale them identically, we do `n * 2^f * 5^-f`, so we now have `m * 2^e`. | 358 | // to scale them identically, we do `n * 2^f * 5^-f`, so we now have `m * 2^e`. | |||||
| 359 | // we then need to scale by `2^(f- e)`, and then the two significant digits | 359 | // we then need to scale by `2^(f- e)`, and then the two significant digits | |||||
| 360 | // are of the same magnitude. | 360 | // are of the same magnitude. | |||||
| 361 | template <typename T> | 361 | template <typename T> | |||||
| 362 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 362 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| HITCBC | 363 | 1611 | adjusted_mantissa negative_digit_comp(bigint& bigmant, adjusted_mantissa am, int32_t exponent) noexcept { | 363 | 1611 | adjusted_mantissa negative_digit_comp(bigint& bigmant, adjusted_mantissa am, int32_t exponent) noexcept { | ||
| HITCBC | 364 | 1611 | bigint& real_digits = bigmant; | 364 | 1611 | bigint& real_digits = bigmant; | ||
| HITCBC | 365 | 1611 | int32_t real_exp = exponent; | 365 | 1611 | int32_t real_exp = exponent; | ||
| 366 | 366 | |||||||
| 367 | // get the value of `b`, rounded down, and get a bigint representation of b+h | 367 | // get the value of `b`, rounded down, and get a bigint representation of b+h | |||||
| HITCBC | 368 | 1611 | adjusted_mantissa am_b = am; | 368 | 1611 | adjusted_mantissa am_b = am; | ||
| 369 | // gcc7 buf: use a lambda to remove the noexcept qualifier bug with -Wnoexcept-type. | 369 | // gcc7 buf: use a lambda to remove the noexcept qualifier bug with -Wnoexcept-type. | |||||
| HITCBC | 370 | 3222 | round<T>(am_b, [](adjusted_mantissa&a, int32_t shift) { round_down(a, shift); }); | 370 | 3222 | round<T>(am_b, [](adjusted_mantissa&a, int32_t shift) { round_down(a, shift); }); | ||
| 371 | T b; | 371 | T b; | |||||
| HITCBC | 372 | 1611 | to_float(false, am_b, b); | 372 | 1611 | to_float(false, am_b, b); | ||
| HITCBC | 373 | 1611 | adjusted_mantissa theor = to_extended_halfway(b); | 373 | 1611 | adjusted_mantissa theor = to_extended_halfway(b); | ||
| HITCBC | 374 | 1611 | bigint theor_digits(theor.mantissa); | 374 | 1611 | bigint theor_digits(theor.mantissa); | ||
| HITCBC | 375 | 1611 | int32_t theor_exp = theor.power2; | 375 | 1611 | int32_t theor_exp = theor.power2; | ||
| 376 | 376 | |||||||
| 377 | // scale real digits and theor digits to be same power. | 377 | // scale real digits and theor digits to be same power. | |||||
| HITCBC | 378 | 1611 | int32_t pow2_exp = theor_exp - real_exp; | 378 | 1611 | int32_t pow2_exp = theor_exp - real_exp; | ||
| HITCBC | 379 | 1611 | uint32_t pow5_exp = uint32_t(-real_exp); | 379 | 1611 | uint32_t pow5_exp = uint32_t(-real_exp); | ||
| HITCBC | 380 | 1611 | if (pow5_exp != 0) { | 380 | 1611 | if (pow5_exp != 0) { | ||
| HITCBC | 381 | 1611 | theor_digits.pow5(pow5_exp); | 381 | 1611 | theor_digits.pow5(pow5_exp); | ||
| 382 | } | 382 | } | |||||
| HITCBC | 383 | 1611 | if (pow2_exp > 0) { | 383 | 1611 | if (pow2_exp > 0) { | ||
| HITCBC | 384 | 140 | theor_digits.pow2(uint32_t(pow2_exp)); | 384 | 140 | theor_digits.pow2(uint32_t(pow2_exp)); | ||
| HITCBC | 385 | 1471 | } else if (pow2_exp < 0) { | 385 | 1471 | } else if (pow2_exp < 0) { | ||
| HITCBC | 386 | 1469 | real_digits.pow2(uint32_t(-pow2_exp)); | 386 | 1469 | real_digits.pow2(uint32_t(-pow2_exp)); | ||
| 387 | } | 387 | } | |||||
| 388 | 388 | |||||||
| 389 | // compare digits, and use it to director rounding | 389 | // compare digits, and use it to director rounding | |||||
| HITCBC | 390 | 1611 | int ord = real_digits.compare(theor_digits); | 390 | 1611 | int ord = real_digits.compare(theor_digits); | ||
| HITCBC | 391 | 1611 | adjusted_mantissa answer = am; | 391 | 1611 | adjusted_mantissa answer = am; | ||
| HITCBC | 392 | 1611 | round<T>(answer, [ord](adjusted_mantissa& a, int32_t shift) { | 392 | 1611 | round<T>(answer, [ord](adjusted_mantissa& a, int32_t shift) { | ||
| HITCBC | 393 | 1611 | round_nearest_tie_even(a, shift, [ord](bool is_odd, bool, bool) -> bool { | 393 | 1611 | round_nearest_tie_even(a, shift, [ord](bool is_odd, bool, bool) -> bool { | ||
| HITCBC | 394 | 1611 | if (ord > 0) { | 394 | 1611 | if (ord > 0) { | ||
| HITCBC | 395 | 767 | return true; | 395 | 767 | return true; | ||
| HITCBC | 396 | 844 | } else if (ord < 0) { | 396 | 844 | } else if (ord < 0) { | ||
| HITCBC | 397 | 844 | return false; | 397 | 844 | return false; | ||
| 398 | } else { | 398 | } else { | |||||
| MISUBC | 399 | ✗ | return is_odd; | 399 | ✗ | return is_odd; | ||
| 400 | } | 400 | } | |||||
| 401 | }); | 401 | }); | |||||
| 402 | }); | 402 | }); | |||||
| 403 | 403 | |||||||
| HITCBC | 404 | 1611 | return answer; | 404 | 1611 | return answer; | ||
| 405 | } | 405 | } | |||||
| 406 | 406 | |||||||
| 407 | // parse the significant digits as a big integer to unambiguously round | 407 | // parse the significant digits as a big integer to unambiguously round | |||||
| 408 | // the significant digits. here, we are trying to determine how to round | 408 | // the significant digits. here, we are trying to determine how to round | |||||
| 409 | // an extended float representation close to `b+h`, halfway between `b` | 409 | // an extended float representation close to `b+h`, halfway between `b` | |||||
| 410 | // (the float rounded-down) and `b+u`, the next positive float. this | 410 | // (the float rounded-down) and `b+u`, the next positive float. this | |||||
| 411 | // algorithm is always correct, and uses one of two approaches. when | 411 | // algorithm is always correct, and uses one of two approaches. when | |||||
| 412 | // the exponent is positive relative to the significant digits (such as | 412 | // the exponent is positive relative to the significant digits (such as | |||||
| 413 | // 1234), we create a big-integer representation, get the high 64-bits, | 413 | // 1234), we create a big-integer representation, get the high 64-bits, | |||||
| 414 | // determine if any lower bits are truncated, and use that to direct | 414 | // determine if any lower bits are truncated, and use that to direct | |||||
| 415 | // rounding. in case of a negative exponent relative to the significant | 415 | // rounding. in case of a negative exponent relative to the significant | |||||
| 416 | // digits (such as 1.2345), we create a theoretical representation of | 416 | // digits (such as 1.2345), we create a theoretical representation of | |||||
| 417 | // `b` as a big-integer type, scaled to the same binary exponent as | 417 | // `b` as a big-integer type, scaled to the same binary exponent as | |||||
| 418 | // the actual digits. we then compare the big integer representations | 418 | // the actual digits. we then compare the big integer representations | |||||
| 419 | // of both, and use that to direct rounding. | 419 | // of both, and use that to direct rounding. | |||||
| 420 | template <typename T, typename UC> | 420 | template <typename T, typename UC> | |||||
| 421 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | 421 | inline BOOST_JSON_FASTFLOAT_CONSTEXPR20 | |||||
| HITCBC | 422 | 2986 | adjusted_mantissa digit_comp(parsed_number_string_t<UC>& num, adjusted_mantissa am) noexcept { | 422 | 2986 | adjusted_mantissa digit_comp(parsed_number_string_t<UC>& num, adjusted_mantissa am) noexcept { | ||
| 423 | // remove the invalid exponent bias | 423 | // remove the invalid exponent bias | |||||
| HITCBC | 424 | 2986 | am.power2 -= invalid_am_bias; | 424 | 2986 | am.power2 -= invalid_am_bias; | ||
| 425 | 425 | |||||||
| HITCBC | 426 | 2986 | int32_t sci_exp = scientific_exponent(num); | 426 | 2986 | int32_t sci_exp = scientific_exponent(num); | ||
| HITCBC | 427 | 2986 | size_t max_digits = binary_format<T>::max_digits(); | 427 | 2986 | size_t max_digits = binary_format<T>::max_digits(); | ||
| HITCBC | 428 | 2986 | size_t digits = 0; | 428 | 2986 | size_t digits = 0; | ||
| HITCBC | 429 | 2986 | bigint bigmant; | 429 | 2986 | bigint bigmant; | ||
| HITCBC | 430 | 2986 | parse_mantissa(bigmant, num, max_digits, digits); | 430 | 2986 | parse_mantissa(bigmant, num, max_digits, digits); | ||
| 431 | // can't underflow, since digits is at most max_digits. | 431 | // can't underflow, since digits is at most max_digits. | |||||
| HITCBC | 432 | 2986 | int32_t exponent = sci_exp + 1 - int32_t(digits); | 432 | 2986 | int32_t exponent = sci_exp + 1 - int32_t(digits); | ||
| HITCBC | 433 | 2986 | if (exponent >= 0) { | 433 | 2986 | if (exponent >= 0) { | ||
| HITCBC | 434 | 1375 | return positive_digit_comp<T>(bigmant, exponent); | 434 | 1375 | return positive_digit_comp<T>(bigmant, exponent); | ||
| 435 | } else { | 435 | } else { | |||||
| HITCBC | 436 | 1611 | return negative_digit_comp<T>(bigmant, am, exponent); | 436 | 1611 | return negative_digit_comp<T>(bigmant, am, exponent); | ||
| 437 | } | 437 | } | |||||
| 438 | } | 438 | } | |||||
| 439 | 439 | |||||||
| 440 | }}}}}} // namespace fast_float | 440 | }}}}}} // namespace fast_float | |||||
| 441 | 441 | |||||||
| 442 | #endif | 442 | #endif | |||||