//===-- Definition of bfloat16 data type. -----------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #ifndef LLVM_LIBC_SRC___SUPPORT_FPUTIL_BFLOAT16_H #define LLVM_LIBC_SRC___SUPPORT_FPUTIL_BFLOAT16_H #include "hdr/stdint_proxy.h" #include "src/__support/CPP/bit.h" #include "src/__support/CPP/type_traits.h" #include "src/__support/FPUtil/cast.h" #include "src/__support/FPUtil/comparison_operations.h" #include "src/__support/FPUtil/dyadic_float.h" #include "src/__support/FPUtil/generic/add_sub.h" #include "src/__support/FPUtil/generic/div.h" #include "src/__support/FPUtil/generic/mul.h" #include "src/__support/macros/config.h" #include "src/__support/macros/properties/types.h" namespace LIBC_NAMESPACE_DECL { namespace fputil { struct BFloat16 { uint16_t bits; LIBC_INLINE BFloat16() = default; template LIBC_INLINE constexpr explicit BFloat16(T value) : bits(static_cast(0U)) { if constexpr (cpp::is_floating_point_v) { bits = fputil::cast(value).bits; } else if constexpr (cpp::is_integral_v) { Sign sign = Sign::POS; if constexpr (cpp::is_signed_v) { if (value < 0) { sign = Sign::NEG; value = -value; } } fputil::DyadicFloat>::digits> xd(sign, 0, value); bits = xd.template as().bits; } else if constexpr (cpp::is_convertible_v) { bits = value.operator BFloat16().bits; } else { bits = fputil::cast(static_cast(value)).bits; } } template () == fputil::FPType::IEEE754_Binary32, int> = 0> LIBC_INLINE constexpr operator float() const { uint32_t x_bits = static_cast(bits) << 16U; return cpp::bit_cast(x_bits); } template , int> = 0> LIBC_INLINE constexpr explicit operator T() const { return static_cast(static_cast(*this)); } LIBC_INLINE bool operator==(BFloat16 other) const { return fputil::equals(*this, other); } LIBC_INLINE bool operator!=(BFloat16 other) const { return !fputil::equals(*this, other); } LIBC_INLINE bool operator<(BFloat16 other) const { return fputil::less_than(*this, other); } LIBC_INLINE bool operator<=(BFloat16 other) const { return fputil::less_than_or_equals(*this, other); } LIBC_INLINE bool operator>(BFloat16 other) const { return fputil::greater_than(*this, other); } LIBC_INLINE bool operator>=(BFloat16 other) const { return fputil::greater_than_or_equals(*this, other); } LIBC_INLINE constexpr BFloat16 operator-() const { fputil::FPBits result(*this); result.set_sign(result.is_pos() ? Sign::NEG : Sign::POS); return result.get_val(); } LIBC_INLINE BFloat16 operator+(BFloat16 other) const { return fputil::generic::add(*this, other); } LIBC_INLINE BFloat16 operator-(BFloat16 other) const { return fputil::generic::sub(*this, other); } LIBC_INLINE BFloat16 operator*(BFloat16 other) const { return fputil::generic::mul(*this, other); } LIBC_INLINE BFloat16 operator/(BFloat16 other) const { return fputil::generic::div(*this, other); } }; // struct BFloat16 } // namespace fputil } // namespace LIBC_NAMESPACE_DECL #endif // LLVM_LIBC_SRC___SUPPORT_FPUTIL_BFLOAT16_H