| // Copyright 2020 The Dawn & Tint Authors |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are met: |
| // |
| // 1. Redistributions of source code must retain the above copyright notice, this |
| // list of conditions and the following disclaimer. |
| // |
| // 2. Redistributions in binary form must reproduce the above copyright notice, |
| // this list of conditions and the following disclaimer in the documentation |
| // and/or other materials provided with the distribution. |
| // |
| // 3. Neither the name of the copyright holder nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE |
| // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL |
| // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR |
| // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER |
| // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, |
| // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #include "src/tint/utils/strconv/float_to_string.h" |
| |
| #include <cmath> |
| #include <cstring> |
| #include <iomanip> |
| |
| #include "src/tint/utils/ice/ice.h" |
| #include "src/tint/utils/text/string_stream.h" |
| |
| namespace tint::strconv { |
| |
| namespace { |
| |
| template <typename T> |
| struct Traits; |
| |
| template <> |
| struct Traits<float> { |
| using uint_t = uint32_t; |
| static constexpr int kExponentBias = 127; |
| static constexpr uint_t kExponentMask = 0x7f800000; |
| static constexpr uint_t kMantissaMask = 0x007fffff; |
| static constexpr uint_t kSignMask = 0x80000000; |
| static constexpr int kMantissaBits = 23; |
| }; |
| |
| template <> |
| struct Traits<double> { |
| using uint_t = uint64_t; |
| static constexpr int kExponentBias = 1023; |
| static constexpr uint_t kExponentMask = 0x7ff0000000000000; |
| static constexpr uint_t kMantissaMask = 0x000fffffffffffff; |
| static constexpr uint_t kSignMask = 0x8000000000000000; |
| static constexpr int kMantissaBits = 52; |
| }; |
| |
| template <typename F> |
| std::string ToString(F f) { |
| StringStream s; |
| s << f; |
| return s.str(); |
| } |
| |
| template <typename F> |
| std::string ToBitPreservingString(F f) { |
| using T = Traits<F>; |
| using uint_t = typename T::uint_t; |
| |
| // For the NaN case, avoid handling the number as a floating point value. |
| // Some machines will modify the top bit in the mantissa of a NaN. |
| |
| std::stringstream ss; |
| |
| typename T::uint_t float_bits = 0u; |
| static_assert(sizeof(float_bits) == sizeof(f)); |
| std::memcpy(&float_bits, &f, sizeof(float_bits)); |
| |
| // Handle the sign. |
| if (float_bits & T::kSignMask) { |
| // If `f` is -0.0 print -0.0. |
| ss << '-'; |
| // Strip sign bit. |
| float_bits = float_bits & (~T::kSignMask); |
| } |
| |
| switch (std::fpclassify(f)) { |
| case FP_ZERO: |
| case FP_NORMAL: |
| std::memcpy(&f, &float_bits, sizeof(float_bits)); |
| ss << ToString(f); |
| break; |
| |
| default: { |
| // Infinity, NaN, and Subnormal |
| // TODO(dneto): It's unclear how Infinity and NaN should be handled. |
| // See https://github.com/gpuweb/gpuweb/issues/1769 |
| |
| // std::hexfloat prints 'nan' and 'inf' instead of an explicit representation like we |
| // want. Split it out manually. |
| int mantissa_nibbles = (T::kMantissaBits + 3) / 4; |
| |
| const int biased_exponent = |
| static_cast<int>((float_bits & T::kExponentMask) >> T::kMantissaBits); |
| int exponent = biased_exponent - T::kExponentBias; |
| uint_t mantissa = float_bits & T::kMantissaMask; |
| |
| ss << "0x"; |
| |
| if (exponent == T::kExponentBias + 1) { |
| if (mantissa == 0) { |
| // Infinity case. |
| ss << "1p+" << exponent; |
| } else { |
| // NaN case. |
| // Emit the mantissa bits as if they are left-justified after the binary point. |
| // This is what SPIRV-Tools hex float emitter does, and it's a justifiable |
| // choice independent of the bit width of the mantissa. |
| mantissa <<= (4 - (T::kMantissaBits % 4)); |
| // Remove trailing zeroes, for tidiness. |
| while (0 == (0xf & mantissa)) { |
| mantissa >>= 4; |
| mantissa_nibbles--; |
| } |
| ss << "1." << std::hex << std::setfill('0') << std::setw(mantissa_nibbles) |
| << mantissa << "p+" << std::dec << exponent; |
| } |
| } else { |
| // Subnormal, and not zero. |
| TINT_ASSERT(mantissa != 0); |
| const auto kTopBit = static_cast<uint_t>(1u) << T::kMantissaBits; |
| |
| // Shift left until we get 1.x |
| while (0 == (kTopBit & mantissa)) { |
| mantissa <<= 1; |
| exponent--; |
| } |
| // Emit the leading 1, and remove it from the mantissa. |
| ss << "1"; |
| mantissa = mantissa ^ kTopBit; |
| exponent++; |
| |
| // Left-justify mantissa to whole nibble. |
| mantissa <<= (4 - (T::kMantissaBits % 4)); |
| |
| // Emit the fractional part. |
| if (mantissa) { |
| // Remove trailing zeroes, for tidiness |
| while (0 == (0xf & mantissa)) { |
| mantissa >>= 4; |
| mantissa_nibbles--; |
| } |
| ss << "." << std::hex << std::setfill('0') << std::setw(mantissa_nibbles) |
| << mantissa; |
| } |
| // Emit the exponent |
| ss << "p" << std::showpos << std::dec << exponent; |
| } |
| } |
| } |
| return ss.str(); |
| } |
| |
| } // namespace |
| |
| std::string FloatToString(float f) { |
| return ToString(f); |
| } |
| |
| std::string FloatToBitPreservingString(float f) { |
| return ToBitPreservingString(f); |
| } |
| |
| std::string DoubleToString(double f) { |
| return ToString(f); |
| } |
| |
| std::string DoubleToBitPreservingString(double f) { |
| return ToBitPreservingString(f); |
| } |
| |
| } // namespace tint::strconv |