// Copyright 2021 The Tint Authors. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef SRC_TINT_NUMBER_H_ #define SRC_TINT_NUMBER_H_ #include #include #include #include #include #include "src/tint/utils/compiler_macros.h" #include "src/tint/utils/result.h" #include "src/tint/utils/string_stream.h" #include "src/tint/utils/traits.h" // Forward declaration namespace tint { /// Number wraps a integer or floating point number, enforcing explicit casting. template struct Number; } // namespace tint namespace tint::detail { /// Base template for IsNumber template struct IsNumber : std::false_type {}; /// Specialization for IsNumber template struct IsNumber> : std::true_type {}; /// An empty structure used as a unique template type for Number when /// specializing for the f16 type. struct NumberKindF16 {}; /// Helper for obtaining the underlying type for a Number. template struct NumberUnwrapper { /// When T is not a Number, then type defined to be T. using type = T; }; /// NumberUnwrapper specialization for Number. template struct NumberUnwrapper> { /// The Number's underlying type. using type = typename Number::type; }; } // namespace tint::detail namespace tint { /// Evaluates to true iff T is a Number template constexpr bool IsNumber = detail::IsNumber::value; /// Resolves to the underlying type for a Number. template using UnwrapNumber = typename detail::NumberUnwrapper::type; /// Evaluates to true iff T or Number is a floating-point type or is NumberKindF16. template constexpr bool IsFloatingPoint = std::is_floating_point_v> || std::is_same_v, detail::NumberKindF16>; /// Evaluates to true iff T or Number is an integral type. template constexpr bool IsIntegral = std::is_integral_v>; /// Evaluates to true iff T or Number is a signed integer type. template constexpr bool IsSignedIntegral = std::is_integral_v> && std::is_signed_v>; /// Evaluates to true iff T or Number is an unsigned integer type. template constexpr bool IsUnsignedIntegral = std::is_integral_v> && std::is_unsigned_v>; /// Evaluates to true iff T is an integer type, floating-point type or is NumberKindF16. template constexpr bool IsNumeric = IsIntegral || IsFloatingPoint; /// Returns the bit width of T template constexpr size_t BitWidth = sizeof(UnwrapNumber) * 8; /// NumberBase is a CRTP base class for Number template struct NumberBase { /// @returns value of type `Number` with the highest value for that type. static NumberT Highest() { return NumberT(NumberT::kHighestValue); } /// @returns value of type `Number` with the lowest value for that type. static NumberT Lowest() { return NumberT(NumberT::kLowestValue); } /// @returns value of type `Number` with the smallest value for that type. static NumberT Smallest() { return NumberT(NumberT::kSmallestValue); } /// @returns value of type `Number` that represents NaN for that type. static NumberT NaN() { return NumberT(std::numeric_limits>::quiet_NaN()); } /// @returns value of type `Number` that represents infinity for that type. static NumberT Inf() { return NumberT(std::numeric_limits>::infinity()); } }; /// Number wraps a integer or floating point number, enforcing explicit casting. template struct Number : NumberBase> { static_assert(IsNumeric, "Number constructed with non-numeric type"); /// type is the underlying type of the Number using type = T; /// Number of bits in the number. static constexpr size_t kNumBits = sizeof(T) * 8; /// Highest finite representable value of this type. static constexpr type kHighestValue = std::numeric_limits::max(); /// Lowest finite representable value of this type. static constexpr type kLowestValue = std::numeric_limits::lowest(); /// Smallest positive normal value of this type. static constexpr type kSmallestValue = std::is_integral_v ? 0 : std::numeric_limits::min(); /// Smallest positive subnormal value of this type, 0 for integral type. static constexpr type kSmallestSubnormalValue = std::is_integral_v ? 0 : std::numeric_limits::denorm_min(); /// Constructor. The value is zero-initialized. Number() = default; /// Constructor. /// @param v the value to initialize this Number to template explicit Number(U v) : value(static_cast(v)) {} /// Constructor. /// @param v the value to initialize this Number to template explicit Number(Number v) : value(static_cast(v.value)) {} /// Conversion operator /// @returns the value as T operator T() const { return value; } /// Negation operator /// @returns the negative value of the number Number operator-() const { return Number(-value); } /// Assignment operator /// @param v the new value /// @returns this Number so calls can be chained Number& operator=(T v) { value = v; return *this; } /// The number value type value = {}; }; /// Writes the number to the ostream. /// @param out the stream to write to /// @param num the Number /// @return the stream so calls can be chained template inline utils::StringStream& operator<<(utils::StringStream& out, Number num) { return out << num.value; } /// The partial specification of Number for f16 type, storing the f16 value as float, /// and enforcing proper explicit casting. template <> struct Number : NumberBase> { /// C++ does not have a native float16 type, so we use a 32-bit float instead. using type = float; /// Number of bits in the number. static constexpr size_t kNumBits = 16; /// Highest finite representable value of this type. static constexpr type kHighestValue = 65504.0f; // 2¹⁵ × (1 + 1023/1024) /// Lowest finite representable value of this type. static constexpr type kLowestValue = -65504.0f; /// Smallest positive normal value of this type. /// binary16 0_00001_0000000000, value is 2⁻¹⁴. static constexpr type kSmallestValue = 0x1p-14f; /// Smallest positive subnormal value of this type. /// binary16 0_00000_0000000001, value is 2⁻¹⁴ * 2⁻¹⁰ = 2⁻²⁴. static constexpr type kSmallestSubnormalValue = 0x1p-24f; /// Constructor. The value is zero-initialized. Number() = default; /// Constructor. /// @param v the value to initialize this Number to template explicit Number(U v) : value(Quantize(static_cast(v))) {} /// Constructor. /// @param v the value to initialize this Number to template explicit Number(Number v) : value(Quantize(static_cast(v.value))) {} /// Conversion operator /// @returns the value as the internal representation type of F16 operator float() const { return value; } /// Negation operator /// @returns the negative value of the number Number operator-() const { return Number(-value); } /// Assignment operator with parameter as native floating point type /// @param v the new value /// @returns this Number so calls can be chained Number& operator=(type v) { value = Quantize(v); return *this; } /// Get the binary16 bit pattern in type uint16_t of this value. /// @returns the binary16 bit pattern, in type uint16_t, of the stored quantized f16 value. If /// the value is NaN, the returned value will be 0x7e00u. If the value is positive infinity, the /// returned value will be 0x7c00u. If the input value is negative infinity, the returned value /// will be 0xfc00u. uint16_t BitsRepresentation() const; /// Creates an f16 value from the uint16_t bit representation. /// @param bits the bits to convert from /// @returns the binary16 value based off the provided bit pattern. static Number FromBits(uint16_t bits); /// @param value the input float32 value /// @returns the float32 value quantized to the smaller float16 value, through truncation of the /// mantissa bits (no rounding). If the float32 value is too large (positive or negative) to be /// represented by a float16 value, then the returned value will be positive or negative /// infinity. static type Quantize(type value); /// The number value, stored as float type value = {}; }; /// `AInt` is a type alias to `Number`. using AInt = Number; /// `AFloat` is a type alias to `Number`. using AFloat = Number; /// `i32` is a type alias to `Number`. using i32 = Number; /// `u32` is a type alias to `Number`. using u32 = Number; /// `f32` is a type alias to `Number` using f32 = Number; /// `f16` is a type alias to `Number`, which should be IEEE 754 binary16. /// However since C++ don't have native binary16 type, the value is stored as float. using f16 = Number; template >* = nullptr> inline const auto kPi = T(UnwrapNumber(3.14159265358979323846)); /// True iff T is an abstract number type template constexpr bool IsAbstract = std::is_same_v || std::is_same_v; /// @returns the friendly name of Number type T template >* = nullptr> const char* FriendlyName() { if constexpr (std::is_same_v) { return "abstract-int"; } else if constexpr (std::is_same_v) { return "abstract-float"; } else if constexpr (std::is_same_v) { return "i32"; } else if constexpr (std::is_same_v) { return "u32"; } else if constexpr (std::is_same_v) { return "f32"; } else if constexpr (std::is_same_v) { return "f16"; } else { static_assert(!sizeof(T), "Unhandled type"); } } /// @returns the friendly name of T when T is bool template >* = nullptr> const char* FriendlyName() { return "bool"; } /// Enumerator of failure reasons when converting from one number to another. enum class ConversionFailure { kExceedsPositiveLimit, // The value was too big (+'ve) to fit in the target type kExceedsNegativeLimit, // The value was too big (-'ve) to fit in the target type }; /// Writes the conversion failure message to the ostream. /// @param out the stream to write to /// @param failure the ConversionFailure /// @return the stream so calls can be chained utils::StringStream& operator<<(utils::StringStream& out, ConversionFailure failure); /// Converts a number from one type to another, checking that the value fits in the target type. /// @returns the resulting value of the conversion, or a failure reason. template utils::Result CheckedConvert(Number num) { // Use the highest-precision integer or floating-point type to perform the comparisons. using T = std::conditional_t> || IsFloatingPoint, AFloat::type, AInt::type>; const auto value = static_cast(num.value); if (value > static_cast(TO::kHighestValue)) { return ConversionFailure::kExceedsPositiveLimit; } if (value < static_cast(TO::kLowestValue)) { return ConversionFailure::kExceedsNegativeLimit; } return TO(value); // Success } /// Equality operator. /// @param a the LHS number /// @param b the RHS number /// @returns true if the numbers `a` and `b` are exactly equal. Also considers sign bit. template bool operator==(Number a, Number b) { // Use the highest-precision integer or floating-point type to perform the comparisons. using T = std::conditional_t || IsFloatingPoint, AFloat::type, AInt::type>; auto va = static_cast(a.value); auto vb = static_cast(b.value); if constexpr (IsFloatingPoint) { if (std::signbit(va) != std::signbit(vb)) { return false; } } return std::equal_to()(va, vb); } /// Inequality operator. /// @param a the LHS number /// @param b the RHS number /// @returns true if the numbers `a` and `b` are exactly unequal. Also considers sign bit. template bool operator!=(Number a, Number b) { return !(a == b); } /// Equality operator. /// @param a the LHS number /// @param b the RHS number /// @returns true if the numbers `a` and `b` are exactly equal. template std::enable_if_t, bool> operator==(Number a, B b) { return a == Number(b); } /// Inequality operator. /// @param a the LHS number /// @param b the RHS number /// @returns true if the numbers `a` and `b` are exactly unequal. template std::enable_if_t, bool> operator!=(Number a, B b) { return !(a == b); } /// Equality operator. /// @param a the LHS number /// @param b the RHS number /// @returns true if the numbers `a` and `b` are exactly equal. template std::enable_if_t, bool> operator==(A a, Number b) { return Number(a) == b; } /// Inequality operator. /// @param a the LHS number /// @param b the RHS number /// @returns true if the numbers `a` and `b` are exactly unequal. template std::enable_if_t, bool> operator!=(A a, Number b) { return !(a == b); } /// Define 'TINT_HAS_OVERFLOW_BUILTINS' if the compiler provide overflow checking builtins. /// If the compiler does not support these builtins, then these are emulated with algorithms /// described in: /// https://wiki.sei.cmu.edu/confluence/display/c/INT32-C.+Ensure+that+operations+on+signed+integers+do+not+result+in+overflow #if defined(__GNUC__) && __GNUC__ >= 5 #define TINT_HAS_OVERFLOW_BUILTINS #elif defined(__clang__) #if __has_builtin(__builtin_add_overflow) && __has_builtin(__builtin_mul_overflow) #define TINT_HAS_OVERFLOW_BUILTINS #endif #endif // https://gcc.gnu.org/bugzilla/show_bug.cgi?id=80635 TINT_BEGIN_DISABLE_WARNING(MAYBE_UNINITIALIZED); /// @returns a + b, or an empty optional if the resulting value overflowed the AInt inline std::optional CheckedAdd(AInt a, AInt b) { int64_t result; #ifdef TINT_HAS_OVERFLOW_BUILTINS if (__builtin_add_overflow(a.value, b.value, &result)) { return {}; } #else // TINT_HAS_OVERFLOW_BUILTINS if (a.value >= 0) { if (b.value > AInt::kHighestValue - a.value) { return {}; } } else { if (b.value < AInt::kLowestValue - a.value) { return {}; } } result = a.value + b.value; #endif // TINT_HAS_OVERFLOW_BUILTINS return AInt(result); } /// @returns a + b, or an empty optional if the resulting value overflowed the float value template >> inline std::optional CheckedAdd(FloatingPointT a, FloatingPointT b) { auto result = FloatingPointT{a.value + b.value}; if (!std::isfinite(result.value)) { return {}; } return result; } /// @returns a - b, or an empty optional if the resulting value overflowed the AInt inline std::optional CheckedSub(AInt a, AInt b) { int64_t result; #ifdef TINT_HAS_OVERFLOW_BUILTINS if (__builtin_sub_overflow(a.value, b.value, &result)) { return {}; } #else // TINT_HAS_OVERFLOW_BUILTINS if (b.value >= 0) { if (a.value < AInt::kLowestValue + b.value) { return {}; } } else { if (a.value > AInt::kHighestValue + b.value) { return {}; } } result = a.value - b.value; #endif // TINT_HAS_OVERFLOW_BUILTINS return AInt(result); } /// @returns a + b, or an empty optional if the resulting value overflowed the float value template >> inline std::optional CheckedSub(FloatingPointT a, FloatingPointT b) { auto result = FloatingPointT{a.value - b.value}; if (!std::isfinite(result.value)) { return {}; } return result; } /// @returns a * b, or an empty optional if the resulting value overflowed the AInt inline std::optional CheckedMul(AInt a, AInt b) { int64_t result; #ifdef TINT_HAS_OVERFLOW_BUILTINS if (__builtin_mul_overflow(a.value, b.value, &result)) { return {}; } #else // TINT_HAS_OVERFLOW_BUILTINS if (a > 0) { if (b > 0) { if (a > (AInt::kHighestValue / b)) { return {}; } } else { if (b < (AInt::kLowestValue / a)) { return {}; } } } else { if (b > 0) { if (a < (AInt::kLowestValue / b)) { return {}; } } else { if ((a != 0) && (b < (AInt::kHighestValue / a))) { return {}; } } } result = a.value * b.value; #endif // TINT_HAS_OVERFLOW_BUILTINS return AInt(result); } /// @returns a * b, or an empty optional if the resulting value overflowed the float value template >> inline std::optional CheckedMul(FloatingPointT a, FloatingPointT b) { auto result = FloatingPointT{a.value * b.value}; if (!std::isfinite(result.value)) { return {}; } return result; } /// @returns a / b, or an empty optional if the resulting value overflowed the AInt inline std::optional CheckedDiv(AInt a, AInt b) { if (b == 0) { return {}; } if (b == -1 && a == AInt::Lowest()) { return {}; } return AInt{a.value / b.value}; } /// @returns a / b, or an empty optional if the resulting value overflowed the float value template >> inline std::optional CheckedDiv(FloatingPointT a, FloatingPointT b) { if (b == FloatingPointT{0.0} || b == FloatingPointT{-0.0}) { return {}; } auto result = FloatingPointT{a.value / b.value}; if (!std::isfinite(result.value)) { return {}; } return result; } namespace detail { /// @returns the remainder of e1 / e2 template inline T Mod(T e1, T e2) { if constexpr (IsIntegral) { return e1 % e2; } else { return e1 - e2 * std::trunc(e1 / e2); } } } // namespace detail /// @returns the remainder of a / b, or an empty optional if the resulting value overflowed the AInt inline std::optional CheckedMod(AInt a, AInt b) { if (b == 0) { return {}; } if (b == -1 && a == AInt::Lowest()) { return {}; } return AInt{detail::Mod(a.value, b.value)}; } /// @returns the remainder of a / b, or an empty optional if the resulting value overflowed the /// float value template >> inline std::optional CheckedMod(FloatingPointT a, FloatingPointT b) { if (b == FloatingPointT{0.0} || b == FloatingPointT{-0.0}) { return {}; } auto result = FloatingPointT{detail::Mod(a.value, b.value)}; if (!std::isfinite(result.value)) { return {}; } return result; } /// @returns a * b + c, or an empty optional if the value overflowed the AInt inline std::optional CheckedMadd(AInt a, AInt b, AInt c) { if (auto mul = CheckedMul(a, b)) { return CheckedAdd(mul.value(), c); } return {}; } /// @returns the value of `base` raised to the power `exp`, or an empty optional if the operation /// cannot be performed. template >> inline std::optional CheckedPow(FloatingPointT base, FloatingPointT exp) { static_assert(IsNumber); if ((base < 0) || (base == 0 && exp <= 0)) { return {}; } auto result = FloatingPointT{std::pow(base.value, exp.value)}; if (!std::isfinite(result.value)) { return {}; } return result; } TINT_END_DISABLE_WARNING(MAYBE_UNINITIALIZED); } // namespace tint namespace tint::number_suffixes { /// Literal suffix for abstract integer literals inline AInt operator""_a(unsigned long long int value) { // NOLINT return AInt(static_cast(value)); } /// Literal suffix for abstract float literals inline AFloat operator""_a(long double value) { // NOLINT return AFloat(static_cast(value)); } /// Literal suffix for i32 literals inline i32 operator""_i(unsigned long long int value) { // NOLINT return i32(static_cast(value)); } /// Literal suffix for u32 literals inline u32 operator""_u(unsigned long long int value) { // NOLINT return u32(static_cast(value)); } /// Literal suffix for f32 literals inline f32 operator""_f(long double value) { // NOLINT return f32(static_cast(value)); } /// Literal suffix for f32 literals inline f32 operator""_f(unsigned long long int value) { // NOLINT return f32(static_cast(value)); } /// Literal suffix for f16 literals inline f16 operator""_h(long double value) { // NOLINT return f16(static_cast(value)); } /// Literal suffix for f16 literals inline f16 operator""_h(unsigned long long int value) { // NOLINT return f16(static_cast(value)); } } // namespace tint::number_suffixes namespace std { /// Custom std::hash specialization for tint::Number template class hash> { public: /// @param n the Number /// @return the hash value inline std::size_t operator()(const tint::Number& n) const { return std::hash()(n.value); } }; } // namespace std #endif // SRC_TINT_NUMBER_H_