mirror of
https://github.com/pytorch/pytorch.git
synced 2025-10-20 21:14:14 +08:00
Move Float8 variations to headeronly (#159415)
This PR is a big copy pasta from `c10/util/Float8*` -> `torch/headeronly/util/` which is why we are breaking PR sanity :C (sorry @albanD!). Why is it not a clean copy paste? - For BC reasons, we have to keep the old c10 file around so that OSS devs relying on those files can still get the same APIs - Because we reexpose APIs that are headeronly through torch::headeronly, so there is an extra chunk of code in the new torch::headeronly files to do that. Outside of the copy paste, I: - changed the tests to call torch::headeronly instead of c10 - updated header_only_apis.txt - added `// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)` to pass lint (which was previously skipped for -inl.h files) Pull Request resolved: https://github.com/pytorch/pytorch/pull/159415 Approved by: https://github.com/albanD
This commit is contained in:
committed by
PyTorch MergeBot
parent
9f753f8c0d
commit
5e2ef2a465
@ -1,274 +1 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <cstdint>
|
||||
#include <limits>
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn::Float8_e4m3fn(float value)
|
||||
: x(detail::fp8e4m3fn_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn::operator float() const {
|
||||
return detail::fp8e4m3fn_to_fp32_value(x);
|
||||
}
|
||||
|
||||
/// Special values helper
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e4m3fn::isnan() const {
|
||||
return (x & 0b01111111) == 0b01111111;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn
|
||||
operator+(const Float8_e4m3fn& a, const Float8_e4m3fn& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn
|
||||
operator-(const Float8_e4m3fn& a, const Float8_e4m3fn& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn
|
||||
operator*(const Float8_e4m3fn& a, const Float8_e4m3fn& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(
|
||||
const Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(const Float8_e4m3fn& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator+=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator-=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator*=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator/=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e4m3fn a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e4m3fn a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e4m3fn a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e4m3fn a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e4m3fn b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e4m3fn b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e4m3fn b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e4m3fn b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e4m3fn& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e4m3fn& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e4m3fn& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e4m3fn& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e4m3fn a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e4m3fn a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e4m3fn a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e4m3fn a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e4m3fn b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e4m3fn b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e4m3fn b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e4m3fn b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(Float8_e4m3fn a, int b) {
|
||||
return a + static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(Float8_e4m3fn a, int b) {
|
||||
return a - static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(Float8_e4m3fn a, int b) {
|
||||
return a * static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(Float8_e4m3fn a, int b) {
|
||||
return a / static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(int a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(int a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(int a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(int a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(Float8_e4m3fn a, int64_t b) {
|
||||
return a + static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(Float8_e4m3fn a, int64_t b) {
|
||||
return a - static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(Float8_e4m3fn a, int64_t b) {
|
||||
return a * static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(Float8_e4m3fn a, int64_t b) {
|
||||
return a / static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(int64_t a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(int64_t a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(int64_t a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(int64_t a, Float8_e4m3fn b) {
|
||||
return static_cast<Float8_e4m3fn>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e4m3fn to float.
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e4m3fn> {
|
||||
public:
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 4;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 3;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -5;
|
||||
static constexpr int min_exponent10 = -1;
|
||||
static constexpr int max_exponent = 8;
|
||||
static constexpr int max_exponent10 = 2;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before = false;
|
||||
|
||||
static constexpr c10::Float8_e4m3fn min() {
|
||||
return c10::Float8_e4m3fn(0x08, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn lowest() {
|
||||
return c10::Float8_e4m3fn(0xFE, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn max() {
|
||||
return c10::Float8_e4m3fn(0x7E, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn epsilon() {
|
||||
return c10::Float8_e4m3fn(0x20, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn round_error() {
|
||||
return c10::Float8_e4m3fn(0x30, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn quiet_NaN() {
|
||||
return c10::Float8_e4m3fn(0x7F, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn denorm_min() {
|
||||
return c10::Float8_e4m3fn(0x01, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
#include <torch/headeronly/util/Float8_e4m3fn.h>
|
||||
|
@ -1,238 +1 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e4m3fn type (8-bit floating-point) including conversions
|
||||
/// to standard C types and basic arithmetic operations. Note that arithmetic
|
||||
/// operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration:
|
||||
/// s eeee mmm
|
||||
/// 1 sign bit
|
||||
/// 4 exponent bits
|
||||
/// 3 mantissa bits
|
||||
/// bias = 7
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2209.05433.pdf
|
||||
/// and inspired by Half implementation from pytorch/c10/util/Half.h
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/floating_point_utils.h>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
#include <cmath>
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#ifdef _MSC_VER
|
||||
#include <intrin.h>
|
||||
#endif
|
||||
|
||||
#include <climits>
|
||||
#include <iostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 8-bit floating-point number in fp8 E4M3FN format, in bit
|
||||
* representation, to a 32-bit floating-point number in IEEE single-precision
|
||||
* format, in bit representation.
|
||||
*
|
||||
* @note The implementation doesn't use any floating-point operations.
|
||||
*/
|
||||
inline C10_HOST_DEVICE float fp8e4m3fn_to_fp32_value(uint8_t input) {
|
||||
/*
|
||||
* Extend the fp8 E4M3FN number to 32 bits and shift to the
|
||||
* upper part of the 32-bit word:
|
||||
* +---+----+---+-----------------------------+
|
||||
* | S |EEEE|MMM|0000 0000 0000 0000 0000 0000|
|
||||
* +---+----+---+-----------------------------+
|
||||
* Bits 31 27-30 24-26 0-23
|
||||
*
|
||||
* S - sign bit, E - bits of the biased exponent, M - bits of the mantissa, 0
|
||||
* - zero bits.
|
||||
*/
|
||||
const uint32_t w = (uint32_t)input << 24;
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = w & UINT32_C(0x80000000);
|
||||
/*
|
||||
* Extract mantissa and biased exponent of the input number into the bits 0-30
|
||||
* of the 32-bit word:
|
||||
*
|
||||
* +---+----+---+-----------------------------+
|
||||
* | S |EEEE|MMM|0000 0000 0000 0000 0000 0000|
|
||||
* +---+----+---+-----------------------------+
|
||||
* Bits 31 27-30 24-26 0-23
|
||||
*/
|
||||
const uint32_t nonsign = w & UINT32_C(0x7FFFFFFF);
|
||||
/*
|
||||
* Renorm shift is the number of bits to shift mantissa left to make the
|
||||
* half-precision number normalized. If the initial number is normalized, some
|
||||
* of its high 5 bits (sign == 0 and 4-bit exponent) equals one. In this case
|
||||
* renorm_shift == 0. If the number is denormalize, renorm_shift > 0. Note
|
||||
* that if we shift denormalized nonsign by renorm_shift, the unit bit of
|
||||
* mantissa will shift into exponent, turning the biased exponent into 1, and
|
||||
* making mantissa normalized (i.e. without leading 1).
|
||||
*/
|
||||
#if defined(__CUDA_ARCH__) || defined(__HIP_DEVICE_COMPILE__)
|
||||
uint32_t renorm_shift = __clz(nonsign);
|
||||
#elif defined(__SYCL_DEVICE_ONLY__)
|
||||
// Note: zero is not a supported input into `__builtin_clz`
|
||||
uint32_t renorm_shift =
|
||||
nonsign != 0 ? __builtin_clz(nonsign) : sizeof(uint32_t) * CHAR_BIT;
|
||||
#elif defined(_MSC_VER) && !defined(__clang__)
|
||||
unsigned long nonsign_bsr;
|
||||
_BitScanReverse(&nonsign_bsr, (unsigned long)nonsign);
|
||||
uint32_t renorm_shift = (uint32_t)nonsign_bsr ^ 31;
|
||||
#else
|
||||
// Note: zero is not a supported input into `__builtin_clz`
|
||||
uint32_t renorm_shift =
|
||||
nonsign != 0 ? __builtin_clz(nonsign) : sizeof(uint32_t) * CHAR_BIT;
|
||||
#endif
|
||||
renorm_shift = renorm_shift > 4 ? renorm_shift - 4 : 0;
|
||||
/*
|
||||
* Iff fp8e4m3fn number has all exponent and mantissa bits set to 1,
|
||||
* the addition overflows it into bit 31, and the subsequent shift turns the
|
||||
* high 9 bits into 1. Thus inf_nan_mask == 0x7F800000 if the fp8e4m3fn number
|
||||
* is Nan, 0x00000000 otherwise
|
||||
*/
|
||||
const int32_t inf_nan_mask =
|
||||
((int32_t)(nonsign + 0x01000000) >> 8) & INT32_C(0x7F800000);
|
||||
/*
|
||||
* Iff nonsign is 0, it overflows into 0xFFFFFFFF, turning bit 31
|
||||
* into 1. Otherwise, bit 31 remains 0. The signed shift right by 31
|
||||
* broadcasts bit 31 into all bits of the zero_mask. Thus zero_mask ==
|
||||
* 0xFFFFFFFF if the half-precision number was zero (+0.0h or -0.0h)
|
||||
* 0x00000000 otherwise
|
||||
*/
|
||||
const int32_t zero_mask = (int32_t)(nonsign - 1) >> 31;
|
||||
/*
|
||||
* 1. Shift nonsign left by renorm_shift to normalize it (if the input
|
||||
* was denormal)
|
||||
* 2. Shift nonsign right by 4 so the exponent (4 bits originally)
|
||||
* becomes an 8-bit field and 3-bit mantissa shifts into the 3 high
|
||||
* bits of the 23-bit mantissa of IEEE single-precision number.
|
||||
* 3. Add 0x78 to the exponent (starting at bit 23) to compensate the
|
||||
* different in exponent bias (0x7F for single-precision number less 0x07
|
||||
* for fp8e4m3fn number).
|
||||
* 4. Subtract renorm_shift from the exponent (starting at bit 23) to
|
||||
* account for renormalization. As renorm_shift is less than 0x78, this
|
||||
* can be combined with step 3.
|
||||
* 5. Binary OR with inf_nan_mask to turn the exponent into 0xFF if the
|
||||
* input was NaN or infinity.
|
||||
* 6. Binary ANDNOT with zero_mask to turn the mantissa and exponent
|
||||
* into zero if the input was zero.
|
||||
* 7. Combine with the sign of the input number.
|
||||
*/
|
||||
uint32_t result = sign |
|
||||
((((nonsign << renorm_shift >> 4) + ((0x78 - renorm_shift) << 23)) |
|
||||
inf_nan_mask) &
|
||||
~zero_mask);
|
||||
return fp32_from_bits(result);
|
||||
}
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E4M3FN format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e4m3fn_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of 480.0f, which is the first value
|
||||
* not representable in fp8e4m3fn range:
|
||||
* 0 1111 111 - fp8e4m3fn
|
||||
* 0 10000111 11100000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fp8_max = UINT32_C(1087) << 20;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e4m3fn normal range
|
||||
* into denorm representation
|
||||
* magic number: ((127 - 7) + (23 - 3) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(141) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
|
||||
uint8_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fp8_max) {
|
||||
// NaN - all exponent and mantissa bits set to 1
|
||||
result = 0x7f;
|
||||
} else {
|
||||
if (f_bits < (UINT32_C(121) << 23)) {
|
||||
// Input number is smaller than 2^(-6), which is the smallest
|
||||
// fp8e4m3fn normal number
|
||||
f_bits =
|
||||
fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint8_t mant_odd = (f_bits >> 20) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(7 - 127) << 23) + 0x7FFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 20);
|
||||
}
|
||||
}
|
||||
|
||||
result |= static_cast<uint8_t>(sign >> 24);
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
struct alignas(1) Float8_e4m3fn {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e4m3fn() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e4m3fn(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(std::ostream& out, const Float8_e4m3fn& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
#include <c10/util/Float8_e4m3fn-inl.h> // IWYU pragma: keep
|
||||
#include <torch/headeronly/util/Float8_e4m3fn.h>
|
||||
|
@ -1,279 +1 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/Float8_fnuz_cvt.h>
|
||||
#include <cstring>
|
||||
#include <limits>
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz::Float8_e4m3fnuz(float value)
|
||||
: x(detail::fp8e4m3fnuz_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz::operator float() const {
|
||||
return detail::fp8_fnuz_to_fp32_value<4, 3>(x);
|
||||
}
|
||||
|
||||
/// Special values helper
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e4m3fnuz::isnan() const {
|
||||
return x == 0b10000000;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz
|
||||
operator+(const Float8_e4m3fnuz& a, const Float8_e4m3fnuz& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz
|
||||
operator-(const Float8_e4m3fnuz& a, const Float8_e4m3fnuz& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz
|
||||
operator*(const Float8_e4m3fnuz& a, const Float8_e4m3fnuz& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(
|
||||
const Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(const Float8_e4m3fnuz& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator+=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator-=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator*=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator/=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e4m3fnuz a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e4m3fnuz a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e4m3fnuz a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e4m3fnuz a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e4m3fnuz b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e4m3fnuz b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e4m3fnuz b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e4m3fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e4m3fnuz a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e4m3fnuz a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e4m3fnuz a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e4m3fnuz a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e4m3fnuz b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e4m3fnuz b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e4m3fnuz b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e4m3fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(Float8_e4m3fnuz a, int b) {
|
||||
return a + static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(Float8_e4m3fnuz a, int b) {
|
||||
return a - static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(Float8_e4m3fnuz a, int b) {
|
||||
return a * static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(Float8_e4m3fnuz a, int b) {
|
||||
return a / static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(int a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(int a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(int a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(int a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(Float8_e4m3fnuz a, int64_t b) {
|
||||
return a + static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(Float8_e4m3fnuz a, int64_t b) {
|
||||
return a - static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(Float8_e4m3fnuz a, int64_t b) {
|
||||
return a * static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(Float8_e4m3fnuz a, int64_t b) {
|
||||
return a / static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(int64_t a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(int64_t a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(int64_t a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(int64_t a, Float8_e4m3fnuz b) {
|
||||
return static_cast<Float8_e4m3fnuz>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e4m3fnuz to float.
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e4m3fnuz> {
|
||||
public:
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 4;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 3;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -6;
|
||||
static constexpr int min_exponent10 = -1;
|
||||
static constexpr int max_exponent = 8;
|
||||
static constexpr int max_exponent10 = 2;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before = false;
|
||||
|
||||
static constexpr c10::Float8_e4m3fnuz min() {
|
||||
return c10::Float8_e4m3fnuz(0x08, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz lowest() {
|
||||
return c10::Float8_e4m3fnuz(0xFF, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz max() {
|
||||
return c10::Float8_e4m3fnuz(0x7F, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz epsilon() {
|
||||
return c10::Float8_e4m3fnuz(0x28, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz round_error() {
|
||||
return c10::Float8_e4m3fnuz(0x38, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz infinity() {
|
||||
// NaN (no infinities)
|
||||
return c10::Float8_e4m3fnuz(0x80, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz quiet_NaN() {
|
||||
return c10::Float8_e4m3fnuz(0x80, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz denorm_min() {
|
||||
return c10::Float8_e4m3fnuz(0x01, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
#include <torch/headeronly/util/Float8_e4m3fnuz.h>
|
||||
|
@ -1,139 +1 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e4m3fnuz type (8-bit floating-point) including
|
||||
/// conversions to standard C types and basic arithmetic operations. Note that
|
||||
/// arithmetic operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration remains the same as Float8_e4m3fn:
|
||||
/// s eeee mmm
|
||||
/// 1 sign bit
|
||||
/// 4 exponent bits
|
||||
/// 3 mantissa bits
|
||||
/// The key differences versus Float8_e4m3fn are:
|
||||
/// bias = 8
|
||||
/// no infinities or negative zero
|
||||
/// NaN only when sign bit is 1, rest all 0s
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2206.02915.pdf and
|
||||
/// the existing Float8_e4m3fn implementation.
|
||||
|
||||
#include <c10/macros/Export.h>
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/floating_point_utils.h>
|
||||
#include <type_traits>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#include <iosfwd>
|
||||
#include <ostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E4M3FNUZ format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e4m3fnuz_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of 256.0f, which is the first value not representable
|
||||
* (i.e. the first value which would overflow in to the sign bit, resulting in
|
||||
* a NaN) in fp8e4m3fnuz range:
|
||||
* 1 0000 000 - fp8e4m3fnuz
|
||||
* 0 10000111 00000000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fnuz_max = UINT32_C(0x87) << 23;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e4m3fnuz normal range
|
||||
* into denorm representation
|
||||
* magic number: ((127 - 8) + (23 - 3) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(0x8C) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
|
||||
uint32_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fnuz_max) {
|
||||
// NaN -- sign bit set to 1, rest 0s.
|
||||
return 0x80;
|
||||
}
|
||||
|
||||
if (f_bits < (UINT32_C(0x78) << 23) /* 2^-7 in float32 */) {
|
||||
// Input exponent is less than -7, the smallest e4m3fnuz exponent, so the
|
||||
// number will become subnormal.
|
||||
f_bits = fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
if (result == 0) {
|
||||
// fnuz types don't have negative zero.
|
||||
return 0;
|
||||
}
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint8_t mant_odd = (f_bits >> 20) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(8 - 127) << 23) + 0x7FFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 20);
|
||||
}
|
||||
|
||||
result |= sign >> 24;
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
struct alignas(1) Float8_e4m3fnuz {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e4m3fnuz() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e4m3fnuz(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(
|
||||
std::ostream& out,
|
||||
const Float8_e4m3fnuz& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
#include <c10/util/Float8_e4m3fnuz-inl.h> // IWYU pragma: keep
|
||||
#include <torch/headeronly/util/Float8_e4m3fnuz.h>
|
||||
|
@ -1,286 +1 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <cstring>
|
||||
#include <limits>
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
#define EXP_WIDTH_FP8 5
|
||||
#define MAN_WIDTH_FP8 2
|
||||
#define EXP_BIAS_FP8 15
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2::Float8_e5m2(float value)
|
||||
: x(detail::fp8e5m2_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2::operator float() const {
|
||||
return detail::fp8e5m2_to_fp32_value(x);
|
||||
}
|
||||
|
||||
/// Special values helpers
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2::isnan() const {
|
||||
return (x & 0b01111111) > 0b01111100;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2::isinf() const {
|
||||
return (x & 0b01111111) == 0b01111100;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2
|
||||
operator+(const Float8_e5m2& a, const Float8_e5m2& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2
|
||||
operator-(const Float8_e5m2& a, const Float8_e5m2& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2
|
||||
operator*(const Float8_e5m2& a, const Float8_e5m2& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(
|
||||
const Float8_e5m2& a,
|
||||
const Float8_e5m2& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(const Float8_e5m2& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator+=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator-=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator*=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator/=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e5m2 a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e5m2 a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e5m2 a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e5m2 a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e5m2 b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e5m2 b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e5m2 b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e5m2 b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e5m2& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e5m2& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e5m2& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e5m2& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e5m2 a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e5m2 a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e5m2 a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e5m2 a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e5m2 b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e5m2 b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e5m2 b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e5m2 b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(Float8_e5m2 a, int b) {
|
||||
return a + static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(Float8_e5m2 a, int b) {
|
||||
return a - static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(Float8_e5m2 a, int b) {
|
||||
return a * static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(Float8_e5m2 a, int b) {
|
||||
return a / static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(int a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(int a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(int a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(int a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(Float8_e5m2 a, int64_t b) {
|
||||
return a + static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(Float8_e5m2 a, int64_t b) {
|
||||
return a - static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(Float8_e5m2 a, int64_t b) {
|
||||
return a * static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(Float8_e5m2 a, int64_t b) {
|
||||
return a / static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(int64_t a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(int64_t a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(int64_t a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(int64_t a, Float8_e5m2 b) {
|
||||
return static_cast<Float8_e5m2>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e5m2 to float.
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e5m2> {
|
||||
public:
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = true;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 3;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 2;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -13;
|
||||
static constexpr int min_exponent10 = -4;
|
||||
static constexpr int max_exponent = 16;
|
||||
static constexpr int max_exponent10 = 4;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before =
|
||||
numeric_limits<float>::tinyness_before;
|
||||
|
||||
static constexpr c10::Float8_e5m2 min() {
|
||||
return c10::Float8_e5m2(0x4, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 max() {
|
||||
return c10::Float8_e5m2(0x7B, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 lowest() {
|
||||
return c10::Float8_e5m2(0xFB, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 epsilon() {
|
||||
return c10::Float8_e5m2(0x34, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 round_error() {
|
||||
return c10::Float8_e5m2(0x38, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 infinity() {
|
||||
return c10::Float8_e5m2(0x7C, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 quiet_NaN() {
|
||||
return c10::Float8_e5m2(0x7F, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 denorm_min() {
|
||||
return c10::Float8_e5m2(0x01, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
#include <torch/headeronly/util/Float8_e5m2.h>
|
||||
|
@ -1,146 +1 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e5m2 type (8-bit floating-point) including conversions
|
||||
/// to standard C types and basic arithmetic operations. Note that arithmetic
|
||||
/// operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration:
|
||||
/// s eeeee mm
|
||||
/// 1 sign bit
|
||||
/// 5 exponent bits
|
||||
/// 2 mantissa bits
|
||||
/// bias = 15
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2209.05433.pdf
|
||||
/// and inspired by Half implementation from pytorch/c10/util/Half.h
|
||||
|
||||
#include <c10/util/Half.h>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 8-bit floating-point number in fp8 E5M2 format, in bit
|
||||
* representation, to a 32-bit floating-point number in IEEE single-precision
|
||||
* format, in bit representation.
|
||||
*
|
||||
* @note The implementation doesn't use any floating-point operations.
|
||||
*/
|
||||
inline C10_HOST_DEVICE float fp8e5m2_to_fp32_value(uint8_t input) {
|
||||
/*
|
||||
* Extend the fp8 E5M2 number to 32 bits and shift to the
|
||||
* upper part of the 32-bit word:
|
||||
* +---+----+---+-----------------------------+
|
||||
* | S |EEEEE|MM|0000 0000 0000 0000 0000 0000|
|
||||
* +---+----+---+-----------------------------+
|
||||
* Bits 31 26-30 24-25 0-23
|
||||
*
|
||||
* S - sign bit, E - bits of the biased exponent, M - bits of the mantissa, 0
|
||||
* - zero bits.
|
||||
*/
|
||||
uint16_t half_representation = input;
|
||||
half_representation <<= 8;
|
||||
return fp16_ieee_to_fp32_value(half_representation);
|
||||
}
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E5M2 format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e5m2_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of fp32 infinity
|
||||
* 0 11111111 00000000000000000000000
|
||||
*/
|
||||
constexpr uint32_t fp32_inf = UINT32_C(255) << 23;
|
||||
|
||||
/*
|
||||
* Binary representation of 65536.0f, which is the first value
|
||||
* not representable in fp8e5m2 range:
|
||||
* 0 11111 00 - fp8e5m2
|
||||
* 0 10001111 00000000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fp8_max = UINT32_C(143) << 23;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e5m2 normal range
|
||||
* into denorm representation
|
||||
* magic number: ((127 - 15) + (23 - 2) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(134) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
uint8_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fp8_max) {
|
||||
// NaN - all exponent and mantissa bits set to 1
|
||||
result = f_bits > fp32_inf ? UINT8_C(0x7F) : UINT8_C(0x7C);
|
||||
} else {
|
||||
if (f_bits < (UINT32_C(113) << 23)) {
|
||||
// Input number is smaller than 2^(-14), which is the smallest
|
||||
// fp8e5m2 normal number
|
||||
f_bits =
|
||||
fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint32_t mant_odd = (f_bits >> 21) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(15 - 127) << 23) + 0xFFFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 21);
|
||||
}
|
||||
}
|
||||
|
||||
result |= static_cast<uint8_t>(sign >> 24);
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
struct alignas(1) Float8_e5m2 {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e5m2() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e5m2(uint8_t bits, from_bits_t) : x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e5m2(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
inline C10_HOST_DEVICE bool isinf() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(std::ostream& out, const Float8_e5m2& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
#include <c10/util/Float8_e5m2-inl.h> // IWYU pragma: keep
|
||||
#include <torch/headeronly/util/Float8_e5m2.h>
|
||||
|
@ -1,285 +1 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/Float8_fnuz_cvt.h>
|
||||
#include <cstring>
|
||||
#include <limits>
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz::Float8_e5m2fnuz(float value)
|
||||
: x(detail::fp8e5m2fnuz_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz::operator float() const {
|
||||
return detail::fp8_fnuz_to_fp32_value<5, 2>(x);
|
||||
}
|
||||
|
||||
/// Special values helpers
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2fnuz::isnan() const {
|
||||
return x == 0b10000000;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2fnuz::isinf() const {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz
|
||||
operator+(const Float8_e5m2fnuz& a, const Float8_e5m2fnuz& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz
|
||||
operator-(const Float8_e5m2fnuz& a, const Float8_e5m2fnuz& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz
|
||||
operator*(const Float8_e5m2fnuz& a, const Float8_e5m2fnuz& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(
|
||||
const Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(const Float8_e5m2fnuz& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator+=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator-=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator*=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator/=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e5m2fnuz a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e5m2fnuz a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e5m2fnuz a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e5m2fnuz a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e5m2fnuz b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e5m2fnuz b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e5m2fnuz b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e5m2fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e5m2fnuz a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e5m2fnuz a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e5m2fnuz a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e5m2fnuz a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e5m2fnuz b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e5m2fnuz b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e5m2fnuz b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e5m2fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(Float8_e5m2fnuz a, int b) {
|
||||
return a + static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(Float8_e5m2fnuz a, int b) {
|
||||
return a - static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(Float8_e5m2fnuz a, int b) {
|
||||
return a * static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(Float8_e5m2fnuz a, int b) {
|
||||
return a / static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(int a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(int a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(int a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(int a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(Float8_e5m2fnuz a, int64_t b) {
|
||||
return a + static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(Float8_e5m2fnuz a, int64_t b) {
|
||||
return a - static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(Float8_e5m2fnuz a, int64_t b) {
|
||||
return a * static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(Float8_e5m2fnuz a, int64_t b) {
|
||||
return a / static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(int64_t a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(int64_t a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(int64_t a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(int64_t a, Float8_e5m2fnuz b) {
|
||||
return static_cast<Float8_e5m2fnuz>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e5m2fnuz to float.
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e5m2fnuz> {
|
||||
public:
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 3;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 2;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -14;
|
||||
static constexpr int min_exponent10 = -4;
|
||||
static constexpr int max_exponent = 16;
|
||||
static constexpr int max_exponent10 = 4;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before =
|
||||
numeric_limits<float>::tinyness_before;
|
||||
|
||||
static constexpr c10::Float8_e5m2fnuz min() {
|
||||
return c10::Float8_e5m2fnuz(0x04, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz max() {
|
||||
return c10::Float8_e5m2fnuz(0x7F, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz lowest() {
|
||||
return c10::Float8_e5m2fnuz(0xFF, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz epsilon() {
|
||||
return c10::Float8_e5m2fnuz(0x34, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz round_error() {
|
||||
return c10::Float8_e5m2fnuz(0x38, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz infinity() {
|
||||
return c10::Float8_e5m2fnuz(0x80, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
// TODO(future): we are mapping neg_zero to both inf and NaN, this is
|
||||
// surprising and we should figure out what to do about it.
|
||||
static constexpr c10::Float8_e5m2fnuz quiet_NaN() {
|
||||
return c10::Float8_e5m2fnuz(0x80, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz denorm_min() {
|
||||
return c10::Float8_e5m2fnuz(0x01, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
#include <torch/headeronly/util/Float8_e5m2fnuz.h>
|
||||
|
@ -1,138 +1 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e5m2fnuz type (8-bit floating-point) including
|
||||
/// conversions to standard C types and basic arithmetic operations. Note that
|
||||
/// arithmetic operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration remains the same as e5m2:
|
||||
/// s eeeee mm
|
||||
/// 1 sign bit
|
||||
/// 5 exponent bits
|
||||
/// 2 mantissa bits
|
||||
/// The key differences that e5m2fnuz brings are:
|
||||
/// bias = 16
|
||||
/// no infinities or negative zero
|
||||
/// NaN only when sign bit is 1, rest all 0s
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2206.02915.pdf and
|
||||
/// the existing Float8_e4m3fn implementation.
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/TypeSafeSignMath.h>
|
||||
#include <c10/util/floating_point_utils.h>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#include <iosfwd>
|
||||
#include <ostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E5M2 format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e5m2fnuz_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of 65536.0f, which is the first value not
|
||||
* representable (i.e. the first value which would overflow in to the sign
|
||||
* bit, resulting in a NaN) in fp8e4m3fnuz range:
|
||||
* 1 00000 00 - fp8e5m2fnuz
|
||||
* 0 10001111 00000000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fnuz_max = UINT32_C(0x8F) << 23;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e5m2fnuz normal range
|
||||
* into denormalized representation.
|
||||
* magic number: ((127 - 16) + (23 - 2) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(0x85) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
uint32_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fnuz_max) {
|
||||
// NaN -- sign bit set to 1, rest 0s
|
||||
return 0x80;
|
||||
}
|
||||
|
||||
if (f_bits < (UINT32_C(0x70) << 23) /* 2^-15 in float32 */) {
|
||||
// Input exponent is less than -15, the smallest e5m2fnuz exponent, so the
|
||||
// number will become subnormal.
|
||||
f_bits = fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
if (result == 0) {
|
||||
// fnuz types don't have negative zero.
|
||||
return 0;
|
||||
}
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint8_t mant_odd = (f_bits >> 21) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(16 - 127) << 23) + 0xFFFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 21);
|
||||
}
|
||||
|
||||
result |= sign >> 24;
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
struct alignas(1) Float8_e5m2fnuz {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e5m2fnuz() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e5m2fnuz(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
inline C10_HOST_DEVICE bool isinf() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(
|
||||
std::ostream& out,
|
||||
const Float8_e5m2fnuz& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
#include <c10/util/Float8_e5m2fnuz-inl.h> // IWYU pragma: keep
|
||||
#include <torch/headeronly/util/Float8_e5m2fnuz.h>
|
||||
|
@ -1,112 +1 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/floating_point_utils.h>
|
||||
#include <cstring>
|
||||
#include <limits>
|
||||
|
||||
// TODO(#146647): Can we remove the below warning?
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e8m0fnu::Float8_e8m0fnu(float value)
|
||||
: x(detail::fp8e8m0fnu_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e8m0fnu::operator float() const {
|
||||
// TODO(#146647): maybe rewrite without control flow
|
||||
|
||||
// if exponent is zero, need to special case to return 2^-127 instead of zero
|
||||
if (x == 0) {
|
||||
return c10::detail::fp32_from_bits(0x00400000);
|
||||
}
|
||||
|
||||
// if exponent is NaN, need to special case to return properly encoded NaN
|
||||
if (isnan()) {
|
||||
return c10::detail::fp32_from_bits(0x7f800001);
|
||||
}
|
||||
|
||||
// leave sign at 0, set the exponent bits, leave stored mantissa at 0
|
||||
uint32_t res = x << 23;
|
||||
|
||||
return c10::detail::fp32_from_bits(res);
|
||||
}
|
||||
|
||||
/// Special values helper
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e8m0fnu::isnan() const {
|
||||
return x == 0b11111111;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e8m0fnu to float.
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e8m0fnu> {
|
||||
public:
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_signed = false;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = false;
|
||||
static constexpr auto has_denorm_loss = false;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 1;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 1; // just a 2!
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -126;
|
||||
static constexpr int min_exponent10 = -38;
|
||||
static constexpr int max_exponent = 128;
|
||||
static constexpr int max_exponent10 = 38;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before = false;
|
||||
|
||||
static constexpr c10::Float8_e8m0fnu min() {
|
||||
// 2^-127
|
||||
return c10::Float8_e8m0fnu(0b00000000, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu lowest() {
|
||||
// 2^-127
|
||||
return c10::Float8_e8m0fnu(0b00000000, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu max() {
|
||||
// 254 biased, which is 127 unbiased, so 2^127
|
||||
return c10::Float8_e8m0fnu(0b11111110, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu epsilon() {
|
||||
// according to https://en.cppreference.com/w/cpp/types/numeric_limits, this
|
||||
// is "the difference between 1.0 and the next representable value of the
|
||||
// given floating-point type". The next representable value is 2.0, so the
|
||||
// difference is 1.0 which is 2^0. 0 unbiased is 127 biased.
|
||||
return c10::Float8_e8m0fnu(0b01111111, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu round_error() {
|
||||
// 0.5 in float, which is 2^-1, and -1 + 127 = 126
|
||||
return c10::Float8_e8m0fnu(0b01111110, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu quiet_NaN() {
|
||||
return c10::Float8_e8m0fnu(0b11111111, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
#include <torch/headeronly/util/Float8_e8m0fnu.h>
|
||||
|
@ -1,120 +1 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e8m0fnu type (8-bit floating-point) including
|
||||
/// conversions to standard C types
|
||||
/// Binary configuration :
|
||||
/// eeeeeeee
|
||||
/// no sign bits
|
||||
/// 8 exponent bits
|
||||
/// no mantissa bits
|
||||
///
|
||||
/// This is the E8M0 dtype from the OCP MX format spec
|
||||
/// (https://www.opencompute.org/documents/ocp-microscaling-formats-mx-v1-0-spec-final-pdf,
|
||||
/// Section 5.4.1)
|
||||
|
||||
#include <c10/macros/Export.h>
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <c10/util/floating_point_utils.h>
|
||||
#include <type_traits>
|
||||
|
||||
// TODO(#146647): do we need to special case OPENCL?
|
||||
#if defined(__cplusplus)
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#include <iosfwd>
|
||||
#include <ostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 e8m0fnu format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e8m0fnu_from_fp32_value(float f) {
|
||||
// TODO(#146647): maybe rewrite without control flow
|
||||
|
||||
uint32_t f_bits = c10::detail::fp32_to_bits(f);
|
||||
|
||||
// extract the exponent
|
||||
uint32_t exponent = (f_bits >> 23) & 0b11111111;
|
||||
|
||||
// special case float32 NaN and +-inf to map to e8m0 nan
|
||||
if (exponent == 0b11111111) {
|
||||
return exponent;
|
||||
}
|
||||
|
||||
// next, we use guard, round, sticky bits and the LSB to implement round to
|
||||
// nearest, with ties to even
|
||||
|
||||
// guard bit - bit 23, or 22 zero-indexed
|
||||
uint8_t g = (f_bits & 0x400000) > 0;
|
||||
// round bit - bit 22, or 21 zero-indexed
|
||||
uint8_t r = (f_bits & 0x200000) > 0;
|
||||
// sticky bit - bits 21 to 1, or 20 to 0 zero-indexed
|
||||
uint8_t s = (f_bits & 0x1FFFFF) > 0;
|
||||
// in casting to e8m0, LSB is the implied mantissa bit. It equals to 0 if the
|
||||
// original float32 is denormal, and to 1 if the original float32 is normal.
|
||||
uint8_t lsb = exponent > 0;
|
||||
|
||||
// implement the RNE logic
|
||||
bool round_up = false;
|
||||
|
||||
// if g == 0, round down (no-op)
|
||||
if (g == 1) {
|
||||
if ((r == 1) || (s == 1)) {
|
||||
// round up
|
||||
round_up = true;
|
||||
} else {
|
||||
if (lsb == 1) {
|
||||
// round up
|
||||
round_up = true;
|
||||
}
|
||||
// if lsb == 0, round down (no-op)
|
||||
}
|
||||
}
|
||||
|
||||
if (round_up) {
|
||||
// adjust exponent
|
||||
// note that if exponent was 255 we would have already returned earlier, so
|
||||
// we know we can add one safely without running out of bounds
|
||||
exponent++;
|
||||
}
|
||||
|
||||
return exponent;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
struct alignas(1) Float8_e8m0fnu {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e8m0fnu() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e8m0fnu(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e8m0fnu(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(
|
||||
std::ostream& out,
|
||||
const Float8_e8m0fnu& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
#include <c10/util/Float8_e8m0fnu-inl.h> // IWYU pragma: keep
|
||||
#include <torch/headeronly/util/Float8_e8m0fnu.h>
|
||||
|
@ -1,140 +1 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/macros/Macros.h>
|
||||
#include <limits>
|
||||
#include <type_traits>
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wstring-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wstring-conversion")
|
||||
#endif
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Returns false since we cannot have x < 0 if x is unsigned.
|
||||
template <typename T>
|
||||
inline constexpr bool is_negative(
|
||||
const T& /*x*/,
|
||||
std::true_type /*is_unsigned*/) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Returns true if a signed variable x < 0
|
||||
template <typename T>
|
||||
inline constexpr bool is_negative(const T& x, std::false_type /*is_unsigned*/) {
|
||||
return x < T(0);
|
||||
}
|
||||
|
||||
/// Returns true if x < 0
|
||||
/// NOTE: Will fail on an unsigned custom type
|
||||
/// For the most part it's possible to fix this if
|
||||
/// the custom type has a constexpr constructor.
|
||||
/// However, notably, c10::Half does not :-(
|
||||
template <typename T>
|
||||
inline constexpr bool is_negative(const T& x) {
|
||||
return is_negative(x, std::is_unsigned<T>());
|
||||
}
|
||||
|
||||
/// Returns the sign of an unsigned variable x as 0, 1
|
||||
template <typename T>
|
||||
inline constexpr int signum(const T& x, std::true_type /*is_unsigned*/) {
|
||||
return T(0) < x;
|
||||
}
|
||||
|
||||
/// Returns the sign of a signed variable x as -1, 0, 1
|
||||
template <typename T>
|
||||
inline constexpr int signum(const T& x, std::false_type /*is_unsigned*/) {
|
||||
return (T(0) < x) - (x < T(0));
|
||||
}
|
||||
|
||||
/// Returns the sign of x as -1, 0, 1
|
||||
/// NOTE: Will fail on an unsigned custom type
|
||||
/// For the most part it's possible to fix this if
|
||||
/// the custom type has a constexpr constructor.
|
||||
/// However, notably, c10::Half does not :-(
|
||||
template <typename T>
|
||||
inline constexpr int signum(const T& x) {
|
||||
return signum(x, std::is_unsigned<T>());
|
||||
}
|
||||
|
||||
/// Returns true if a and b are not both negative
|
||||
template <typename T, typename U>
|
||||
inline constexpr bool signs_differ(const T& a, const U& b) {
|
||||
return is_negative(a) != is_negative(b);
|
||||
}
|
||||
|
||||
// Suppress sign compare warning when compiling with GCC
|
||||
// as later does not account for short-circuit rule before
|
||||
// raising the warning, see https://godbolt.org/z/Tr3Msnz99
|
||||
#ifdef __GNUC__
|
||||
#pragma GCC diagnostic push
|
||||
#pragma GCC diagnostic ignored "-Wsign-compare"
|
||||
#endif
|
||||
|
||||
/// Returns true if x is greater than the greatest value of the type Limit
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool greater_than_max(const T& x) {
|
||||
constexpr bool can_overflow =
|
||||
std::numeric_limits<T>::digits > std::numeric_limits<Limit>::digits;
|
||||
return can_overflow && x > (std::numeric_limits<Limit>::max)();
|
||||
}
|
||||
|
||||
#ifdef __GNUC__
|
||||
#pragma GCC diagnostic pop
|
||||
#endif
|
||||
|
||||
/// Returns true if x < lowest(Limit). Standard comparison
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& x,
|
||||
std::false_type /*limit_is_unsigned*/,
|
||||
std::false_type /*x_is_unsigned*/) {
|
||||
return x < std::numeric_limits<Limit>::lowest();
|
||||
}
|
||||
|
||||
/// Returns false since all the limit is signed and therefore includes
|
||||
/// negative values but x cannot be negative because it is unsigned
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& /*x*/,
|
||||
std::false_type /*limit_is_unsigned*/,
|
||||
std::true_type /*x_is_unsigned*/) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Returns true if x < 0, where 0 is constructed from T.
|
||||
/// Limit is not signed, so its lower value is zero
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& x,
|
||||
std::true_type /*limit_is_unsigned*/,
|
||||
std::false_type /*x_is_unsigned*/) {
|
||||
return x < T(0);
|
||||
}
|
||||
|
||||
/// Returns false sign both types are unsigned
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& /*x*/,
|
||||
std::true_type /*limit_is_unsigned*/,
|
||||
std::true_type /*x_is_unsigned*/) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Returns true if x is less than the lowest value of type T
|
||||
/// NOTE: Will fail on an unsigned custom type
|
||||
/// For the most part it's possible to fix this if
|
||||
/// the custom type has a constexpr constructor.
|
||||
/// However, notably, c10::Half does not :
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(const T& x) {
|
||||
return less_than_lowest<Limit>(
|
||||
x, std::is_unsigned<Limit>(), std::is_unsigned<T>());
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
#include <torch/headeronly/util/TypeSafeSignMath.h>
|
||||
|
@ -1,13 +1,13 @@
|
||||
#include <gtest/gtest.h>
|
||||
|
||||
#include <c10/util/Float8_e4m3fn.h>
|
||||
#include <c10/util/Float8_e4m3fnuz.h>
|
||||
#include <c10/util/Float8_e5m2.h>
|
||||
#include <c10/util/Float8_e5m2fnuz.h>
|
||||
#include <c10/util/complex.h>
|
||||
#include <torch/headeronly/util/BFloat16.h>
|
||||
#include <torch/headeronly/util/Float4_e2m1fn_x2.h>
|
||||
|
||||
#include <torch/headeronly/util/Float8_e4m3fn.h>
|
||||
#include <torch/headeronly/util/Float8_e4m3fnuz.h>
|
||||
#include <torch/headeronly/util/Float8_e5m2.h>
|
||||
#include <torch/headeronly/util/Float8_e5m2fnuz.h>
|
||||
#include <torch/headeronly/util/Float8_e8m0fnu.h>
|
||||
#include <torch/headeronly/util/Half.h>
|
||||
#include <torch/headeronly/util/bits.h>
|
||||
#include <torch/headeronly/util/qint32.h>
|
||||
@ -31,12 +31,12 @@ TEST(TestDtype, TestBFloat16) {
|
||||
}
|
||||
|
||||
TEST(TestDtype, TestFloat8_e4m3fn) {
|
||||
c10::Float8_e4m3fn a = 1.0f;
|
||||
c10::Float8_e4m3fn b = 2.0f;
|
||||
c10::Float8_e4m3fn add = 3.0f;
|
||||
c10::Float8_e4m3fn sub = -1.0f;
|
||||
c10::Float8_e4m3fn mul = 2.0f;
|
||||
c10::Float8_e4m3fn div = 0.5f;
|
||||
torch::headeronly::Float8_e4m3fn a = 1.0f;
|
||||
torch::headeronly::Float8_e4m3fn b = 2.0f;
|
||||
torch::headeronly::Float8_e4m3fn add = 3.0f;
|
||||
torch::headeronly::Float8_e4m3fn sub = -1.0f;
|
||||
torch::headeronly::Float8_e4m3fn mul = 2.0f;
|
||||
torch::headeronly::Float8_e4m3fn div = 0.5f;
|
||||
|
||||
EXPECT_EQ(a + b, add);
|
||||
EXPECT_EQ(a - b, sub);
|
||||
@ -45,12 +45,12 @@ TEST(TestDtype, TestFloat8_e4m3fn) {
|
||||
}
|
||||
|
||||
TEST(TestDtype, TestFloat8_e4m3fuz) {
|
||||
c10::Float8_e4m3fnuz a = 1.0f;
|
||||
c10::Float8_e4m3fnuz b = 2.0f;
|
||||
c10::Float8_e4m3fnuz add = 3.0f;
|
||||
c10::Float8_e4m3fnuz sub = -1.0f;
|
||||
c10::Float8_e4m3fnuz mul = 2.0f;
|
||||
c10::Float8_e4m3fnuz div = 0.5f;
|
||||
torch::headeronly::Float8_e4m3fnuz a = 1.0f;
|
||||
torch::headeronly::Float8_e4m3fnuz b = 2.0f;
|
||||
torch::headeronly::Float8_e4m3fnuz add = 3.0f;
|
||||
torch::headeronly::Float8_e4m3fnuz sub = -1.0f;
|
||||
torch::headeronly::Float8_e4m3fnuz mul = 2.0f;
|
||||
torch::headeronly::Float8_e4m3fnuz div = 0.5f;
|
||||
|
||||
EXPECT_EQ(a + b, add);
|
||||
EXPECT_EQ(a - b, sub);
|
||||
@ -59,12 +59,12 @@ TEST(TestDtype, TestFloat8_e4m3fuz) {
|
||||
}
|
||||
|
||||
TEST(TestDtype, TestFloat8_e5m2) {
|
||||
c10::Float8_e5m2 a = 1.0f;
|
||||
c10::Float8_e5m2 b = 2.0f;
|
||||
c10::Float8_e5m2 add = 3.0f;
|
||||
c10::Float8_e5m2 sub = -1.0f;
|
||||
c10::Float8_e5m2 mul = 2.0f;
|
||||
c10::Float8_e5m2 div = 0.5f;
|
||||
torch::headeronly::Float8_e5m2 a = 1.0f;
|
||||
torch::headeronly::Float8_e5m2 b = 2.0f;
|
||||
torch::headeronly::Float8_e5m2 add = 3.0f;
|
||||
torch::headeronly::Float8_e5m2 sub = -1.0f;
|
||||
torch::headeronly::Float8_e5m2 mul = 2.0f;
|
||||
torch::headeronly::Float8_e5m2 div = 0.5f;
|
||||
|
||||
EXPECT_EQ(a + b, add);
|
||||
EXPECT_EQ(a - b, sub);
|
||||
@ -73,12 +73,12 @@ TEST(TestDtype, TestFloat8_e5m2) {
|
||||
}
|
||||
|
||||
TEST(TestDtype, TestFloat8_e5m2fnuz) {
|
||||
c10::Float8_e5m2fnuz a = 1.0f;
|
||||
c10::Float8_e5m2fnuz b = 2.0f;
|
||||
c10::Float8_e5m2fnuz add = 3.0f;
|
||||
c10::Float8_e5m2fnuz sub = -1.0f;
|
||||
c10::Float8_e5m2fnuz mul = 2.0f;
|
||||
c10::Float8_e5m2fnuz div = 0.5f;
|
||||
torch::headeronly::Float8_e5m2fnuz a = 1.0f;
|
||||
torch::headeronly::Float8_e5m2fnuz b = 2.0f;
|
||||
torch::headeronly::Float8_e5m2fnuz add = 3.0f;
|
||||
torch::headeronly::Float8_e5m2fnuz sub = -1.0f;
|
||||
torch::headeronly::Float8_e5m2fnuz mul = 2.0f;
|
||||
torch::headeronly::Float8_e5m2fnuz div = 0.5f;
|
||||
|
||||
EXPECT_EQ(a + b, add);
|
||||
EXPECT_EQ(a - b, sub);
|
||||
@ -86,6 +86,11 @@ TEST(TestDtype, TestFloat8_e5m2fnuz) {
|
||||
EXPECT_EQ(a / b, div);
|
||||
}
|
||||
|
||||
TEST(TestDtype, TestFloat8_e8m0fnu) {
|
||||
torch::headeronly::Float8_e8m0fnu a = 1.0f;
|
||||
ASSERT_FALSE(a.isnan());
|
||||
}
|
||||
|
||||
TEST(TestDtype, TestFloat4) {
|
||||
// not much you can do with this type, just make sure it compiles
|
||||
torch::headeronly::Float4_e2m1fn_x2 a(5);
|
||||
|
@ -6,7 +6,7 @@
|
||||
# c10/util/TypeCast.h
|
||||
convert
|
||||
|
||||
# c10/util/bit_cast.h, torch/headeronly/util/bit_cast.h
|
||||
# torch/headeronly/util/bit_cast.h
|
||||
bit_cast
|
||||
|
||||
# torch/headeronly/util/BFloat16.h
|
||||
@ -15,22 +15,23 @@ BFloat16
|
||||
# torch/headeronly/util/Float4_e2m1fn_x2.h
|
||||
Float4_e2m1fn_x2
|
||||
|
||||
# c10/util/Float8_e4m3fn.h
|
||||
# torch/headeronly/util/Float8_e4m3fn.h
|
||||
Float8_e4m3fn
|
||||
|
||||
# c10/util/Float8_e4m3fnuz.h
|
||||
# torch/headeronly/util/Float8_e4m3fnuz.h
|
||||
Float8_e4m3fnuz
|
||||
|
||||
# c10/util/Float8_e5m2.h
|
||||
# torch/headeronly/util/Float8_e5m2.h
|
||||
Float8_e5m2
|
||||
|
||||
# c10/util/Float8_e5m2fnuz.h
|
||||
# torch/headeronly/util/Float8_e5m2fnuz.h
|
||||
Float8_e5m2fnuz
|
||||
|
||||
# c10/util/Half.h
|
||||
Half
|
||||
# torch/headeronly/util/Float8_e8m0fnu.h
|
||||
Float8_e8m0fnu
|
||||
|
||||
# torch/headeronly/util/Half.h
|
||||
Half
|
||||
fp16_ieee_from_fp32_value
|
||||
fp16_ieee_to_fp32_value
|
||||
|
||||
|
531
torch/headeronly/util/Float8_e4m3fn.h
Normal file
531
torch/headeronly/util/Float8_e4m3fn.h
Normal file
@ -0,0 +1,531 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e4m3fn type (8-bit floating-point) including conversions
|
||||
/// to standard C types and basic arithmetic operations. Note that arithmetic
|
||||
/// operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration:
|
||||
/// s eeee mmm
|
||||
/// 1 sign bit
|
||||
/// 4 exponent bits
|
||||
/// 3 mantissa bits
|
||||
/// bias = 7
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2209.05433.pdf
|
||||
/// and inspired by Half implementation from pytorch/c10/util/Half.h
|
||||
|
||||
#include <torch/headeronly/macros/Macros.h>
|
||||
#include <torch/headeronly/util/floating_point_utils.h>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
#include <cmath>
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#ifdef _MSC_VER
|
||||
#include <intrin.h>
|
||||
#endif
|
||||
|
||||
#include <climits>
|
||||
#include <iostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
struct alignas(1) Float8_e4m3fn {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e4m3fn() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e4m3fn(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(std::ostream& out, const Float8_e4m3fn& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 8-bit floating-point number in fp8 E4M3FN format, in bit
|
||||
* representation, to a 32-bit floating-point number in IEEE single-precision
|
||||
* format, in bit representation.
|
||||
*
|
||||
* @note The implementation doesn't use any floating-point operations.
|
||||
*/
|
||||
inline C10_HOST_DEVICE float fp8e4m3fn_to_fp32_value(uint8_t input) {
|
||||
/*
|
||||
* Extend the fp8 E4M3FN number to 32 bits and shift to the
|
||||
* upper part of the 32-bit word:
|
||||
* +---+----+---+-----------------------------+
|
||||
* | S |EEEE|MMM|0000 0000 0000 0000 0000 0000|
|
||||
* +---+----+---+-----------------------------+
|
||||
* Bits 31 27-30 24-26 0-23
|
||||
*
|
||||
* S - sign bit, E - bits of the biased exponent, M - bits of the mantissa, 0
|
||||
* - zero bits.
|
||||
*/
|
||||
const uint32_t w = (uint32_t)input << 24;
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = w & UINT32_C(0x80000000);
|
||||
/*
|
||||
* Extract mantissa and biased exponent of the input number into the bits 0-30
|
||||
* of the 32-bit word:
|
||||
*
|
||||
* +---+----+---+-----------------------------+
|
||||
* | S |EEEE|MMM|0000 0000 0000 0000 0000 0000|
|
||||
* +---+----+---+-----------------------------+
|
||||
* Bits 31 27-30 24-26 0-23
|
||||
*/
|
||||
const uint32_t nonsign = w & UINT32_C(0x7FFFFFFF);
|
||||
/*
|
||||
* Renorm shift is the number of bits to shift mantissa left to make the
|
||||
* half-precision number normalized. If the initial number is normalized, some
|
||||
* of its high 5 bits (sign == 0 and 4-bit exponent) equals one. In this case
|
||||
* renorm_shift == 0. If the number is denormalize, renorm_shift > 0. Note
|
||||
* that if we shift denormalized nonsign by renorm_shift, the unit bit of
|
||||
* mantissa will shift into exponent, turning the biased exponent into 1, and
|
||||
* making mantissa normalized (i.e. without leading 1).
|
||||
*/
|
||||
#if defined(__CUDA_ARCH__) || defined(__HIP_DEVICE_COMPILE__)
|
||||
uint32_t renorm_shift = __clz(nonsign);
|
||||
#elif defined(__SYCL_DEVICE_ONLY__)
|
||||
// Note: zero is not a supported input into `__builtin_clz`
|
||||
uint32_t renorm_shift =
|
||||
nonsign != 0 ? __builtin_clz(nonsign) : sizeof(uint32_t) * CHAR_BIT;
|
||||
#elif defined(_MSC_VER) && !defined(__clang__)
|
||||
unsigned long nonsign_bsr;
|
||||
_BitScanReverse(&nonsign_bsr, (unsigned long)nonsign);
|
||||
uint32_t renorm_shift = (uint32_t)nonsign_bsr ^ 31;
|
||||
#else
|
||||
// Note: zero is not a supported input into `__builtin_clz`
|
||||
uint32_t renorm_shift =
|
||||
nonsign != 0 ? __builtin_clz(nonsign) : sizeof(uint32_t) * CHAR_BIT;
|
||||
#endif
|
||||
renorm_shift = renorm_shift > 4 ? renorm_shift - 4 : 0;
|
||||
/*
|
||||
* Iff fp8e4m3fn number has all exponent and mantissa bits set to 1,
|
||||
* the addition overflows it into bit 31, and the subsequent shift turns the
|
||||
* high 9 bits into 1. Thus inf_nan_mask == 0x7F800000 if the fp8e4m3fn number
|
||||
* is Nan, 0x00000000 otherwise
|
||||
*/
|
||||
const int32_t inf_nan_mask =
|
||||
((int32_t)(nonsign + 0x01000000) >> 8) & INT32_C(0x7F800000);
|
||||
/*
|
||||
* Iff nonsign is 0, it overflows into 0xFFFFFFFF, turning bit 31
|
||||
* into 1. Otherwise, bit 31 remains 0. The signed shift right by 31
|
||||
* broadcasts bit 31 into all bits of the zero_mask. Thus zero_mask ==
|
||||
* 0xFFFFFFFF if the half-precision number was zero (+0.0h or -0.0h)
|
||||
* 0x00000000 otherwise
|
||||
*/
|
||||
const int32_t zero_mask = (int32_t)(nonsign - 1) >> 31;
|
||||
/*
|
||||
* 1. Shift nonsign left by renorm_shift to normalize it (if the input
|
||||
* was denormal)
|
||||
* 2. Shift nonsign right by 4 so the exponent (4 bits originally)
|
||||
* becomes an 8-bit field and 3-bit mantissa shifts into the 3 high
|
||||
* bits of the 23-bit mantissa of IEEE single-precision number.
|
||||
* 3. Add 0x78 to the exponent (starting at bit 23) to compensate the
|
||||
* different in exponent bias (0x7F for single-precision number less 0x07
|
||||
* for fp8e4m3fn number).
|
||||
* 4. Subtract renorm_shift from the exponent (starting at bit 23) to
|
||||
* account for renormalization. As renorm_shift is less than 0x78, this
|
||||
* can be combined with step 3.
|
||||
* 5. Binary OR with inf_nan_mask to turn the exponent into 0xFF if the
|
||||
* input was NaN or infinity.
|
||||
* 6. Binary ANDNOT with zero_mask to turn the mantissa and exponent
|
||||
* into zero if the input was zero.
|
||||
* 7. Combine with the sign of the input number.
|
||||
*/
|
||||
uint32_t result = sign |
|
||||
((((nonsign << renorm_shift >> 4) + ((0x78 - renorm_shift) << 23)) |
|
||||
inf_nan_mask) &
|
||||
~zero_mask);
|
||||
return fp32_from_bits(result);
|
||||
}
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E4M3FN format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e4m3fn_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of 480.0f, which is the first value
|
||||
* not representable in fp8e4m3fn range:
|
||||
* 0 1111 111 - fp8e4m3fn
|
||||
* 0 10000111 11100000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fp8_max = UINT32_C(1087) << 20;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e4m3fn normal range
|
||||
* into denorm representation
|
||||
* magic number: ((127 - 7) + (23 - 3) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(141) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
|
||||
uint8_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fp8_max) {
|
||||
// NaN - all exponent and mantissa bits set to 1
|
||||
result = 0x7f;
|
||||
} else {
|
||||
if (f_bits < (UINT32_C(121) << 23)) {
|
||||
// Input number is smaller than 2^(-6), which is the smallest
|
||||
// fp8e4m3fn normal number
|
||||
f_bits =
|
||||
fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint8_t mant_odd = (f_bits >> 20) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(7 - 127) << 23) + 0x7FFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 20);
|
||||
}
|
||||
}
|
||||
|
||||
result |= static_cast<uint8_t>(sign >> 24);
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
// -------- below is copied from c10/util/Float8_e4m3fn-inl.h --------//
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn::Float8_e4m3fn(float value)
|
||||
: x(detail::fp8e4m3fn_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn::operator float() const {
|
||||
return detail::fp8e4m3fn_to_fp32_value(x);
|
||||
}
|
||||
|
||||
/// Special values helper
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e4m3fn::isnan() const {
|
||||
return (x & 0b01111111) == 0b01111111;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn
|
||||
operator+(const Float8_e4m3fn& a, const Float8_e4m3fn& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn
|
||||
operator-(const Float8_e4m3fn& a, const Float8_e4m3fn& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn
|
||||
operator*(const Float8_e4m3fn& a, const Float8_e4m3fn& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(
|
||||
const Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(const Float8_e4m3fn& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator+=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator-=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator*=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn& operator/=(
|
||||
Float8_e4m3fn& a,
|
||||
const Float8_e4m3fn& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e4m3fn a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e4m3fn a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e4m3fn a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e4m3fn a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e4m3fn b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e4m3fn b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e4m3fn b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e4m3fn b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e4m3fn& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e4m3fn& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e4m3fn& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e4m3fn& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e4m3fn a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e4m3fn a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e4m3fn a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e4m3fn a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e4m3fn b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e4m3fn b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e4m3fn b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e4m3fn b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(Float8_e4m3fn a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(Float8_e4m3fn a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(Float8_e4m3fn a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(Float8_e4m3fn a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(int a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(int a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(int a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(int a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(Float8_e4m3fn a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(Float8_e4m3fn a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(Float8_e4m3fn a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(Float8_e4m3fn a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e4m3fn>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator+(int64_t a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator-(int64_t a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator*(int64_t a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fn operator/(int64_t a, Float8_e4m3fn b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fn>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e4m3fn to float.
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace torch::headeronly {
|
||||
using c10::Float8_e4m3fn;
|
||||
using c10::operator<<;
|
||||
using c10::operator+;
|
||||
using c10::operator-;
|
||||
using c10::operator*;
|
||||
using c10::operator/;
|
||||
using c10::operator+=;
|
||||
using c10::operator-=;
|
||||
using c10::operator*=;
|
||||
using c10::operator/=;
|
||||
} // namespace torch::headeronly
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e4m3fn> {
|
||||
public:
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 4;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 3;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -5;
|
||||
static constexpr int min_exponent10 = -1;
|
||||
static constexpr int max_exponent = 8;
|
||||
static constexpr int max_exponent10 = 2;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before = false;
|
||||
|
||||
static constexpr c10::Float8_e4m3fn min() {
|
||||
return c10::Float8_e4m3fn(0x08, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn lowest() {
|
||||
return c10::Float8_e4m3fn(0xFE, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn max() {
|
||||
return c10::Float8_e4m3fn(0x7E, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn epsilon() {
|
||||
return c10::Float8_e4m3fn(0x20, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn round_error() {
|
||||
return c10::Float8_e4m3fn(0x30, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn quiet_NaN() {
|
||||
return c10::Float8_e4m3fn(0x7F, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fn denorm_min() {
|
||||
return c10::Float8_e4m3fn(0x01, c10::Float8_e4m3fn::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
442
torch/headeronly/util/Float8_e4m3fnuz.h
Normal file
442
torch/headeronly/util/Float8_e4m3fnuz.h
Normal file
@ -0,0 +1,442 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e4m3fnuz type (8-bit floating-point) including
|
||||
/// conversions to standard C types and basic arithmetic operations. Note that
|
||||
/// arithmetic operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration remains the same as Float8_e4m3fn:
|
||||
/// s eeee mmm
|
||||
/// 1 sign bit
|
||||
/// 4 exponent bits
|
||||
/// 3 mantissa bits
|
||||
/// The key differences versus Float8_e4m3fn are:
|
||||
/// bias = 8
|
||||
/// no infinities or negative zero
|
||||
/// NaN only when sign bit is 1, rest all 0s
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2206.02915.pdf and
|
||||
/// the existing Float8_e4m3fn implementation.
|
||||
|
||||
#include <torch/headeronly/macros/Macros.h>
|
||||
#include <torch/headeronly/util/Float8_fnuz_cvt.h>
|
||||
#include <torch/headeronly/util/floating_point_utils.h>
|
||||
|
||||
#include <limits>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#include <iosfwd>
|
||||
#include <ostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
struct alignas(1) Float8_e4m3fnuz {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e4m3fnuz() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e4m3fnuz(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(
|
||||
std::ostream& out,
|
||||
const Float8_e4m3fnuz& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E4M3FNUZ format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e4m3fnuz_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of 256.0f, which is the first value not representable
|
||||
* (i.e. the first value which would overflow in to the sign bit, resulting in
|
||||
* a NaN) in fp8e4m3fnuz range:
|
||||
* 1 0000 000 - fp8e4m3fnuz
|
||||
* 0 10000111 00000000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fnuz_max = UINT32_C(0x87) << 23;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e4m3fnuz normal range
|
||||
* into denorm representation
|
||||
* magic number: ((127 - 8) + (23 - 3) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(0x8C) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
|
||||
uint32_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fnuz_max) {
|
||||
// NaN -- sign bit set to 1, rest 0s.
|
||||
return 0x80;
|
||||
}
|
||||
|
||||
if (f_bits < (UINT32_C(0x78) << 23) /* 2^-7 in float32 */) {
|
||||
// Input exponent is less than -7, the smallest e4m3fnuz exponent, so the
|
||||
// number will become subnormal.
|
||||
f_bits = fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
if (result == 0) {
|
||||
// fnuz types don't have negative zero.
|
||||
return 0;
|
||||
}
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint8_t mant_odd = (f_bits >> 20) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(8 - 127) << 23) + 0x7FFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 20);
|
||||
}
|
||||
|
||||
result |= sign >> 24;
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
//------ below is copied from c10/util/Float8_e4m3fnuz-inl.h ------//
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz::Float8_e4m3fnuz(float value)
|
||||
: x(detail::fp8e4m3fnuz_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz::operator float() const {
|
||||
return torch::headeronly::detail::fp8_fnuz_to_fp32_value<4, 3>(x);
|
||||
}
|
||||
|
||||
/// Special values helper
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e4m3fnuz::isnan() const {
|
||||
return x == 0b10000000;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz
|
||||
operator+(const Float8_e4m3fnuz& a, const Float8_e4m3fnuz& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz
|
||||
operator-(const Float8_e4m3fnuz& a, const Float8_e4m3fnuz& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz
|
||||
operator*(const Float8_e4m3fnuz& a, const Float8_e4m3fnuz& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(
|
||||
const Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(const Float8_e4m3fnuz& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator+=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator-=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator*=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz& operator/=(
|
||||
Float8_e4m3fnuz& a,
|
||||
const Float8_e4m3fnuz& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e4m3fnuz a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e4m3fnuz a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e4m3fnuz a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e4m3fnuz a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e4m3fnuz b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e4m3fnuz b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e4m3fnuz b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e4m3fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e4m3fnuz& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e4m3fnuz a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e4m3fnuz a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e4m3fnuz a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e4m3fnuz a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e4m3fnuz b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e4m3fnuz b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e4m3fnuz b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e4m3fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(Float8_e4m3fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(Float8_e4m3fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(Float8_e4m3fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(Float8_e4m3fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(int a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(int a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(int a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(int a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(Float8_e4m3fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(Float8_e4m3fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(Float8_e4m3fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(Float8_e4m3fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e4m3fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator+(int64_t a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator-(int64_t a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator*(int64_t a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e4m3fnuz operator/(int64_t a, Float8_e4m3fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e4m3fnuz>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e4m3fnuz to float.
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace torch::headeronly {
|
||||
using c10::Float8_e4m3fnuz;
|
||||
using c10::operator+;
|
||||
using c10::operator-;
|
||||
using c10::operator*;
|
||||
using c10::operator/;
|
||||
using c10::operator+=;
|
||||
using c10::operator-=;
|
||||
using c10::operator*=;
|
||||
using c10::operator/=;
|
||||
using c10::operator<<;
|
||||
|
||||
namespace detail {
|
||||
using c10::detail::fp8e4m3fnuz_from_fp32_value;
|
||||
} // namespace detail
|
||||
|
||||
} // namespace torch::headeronly
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e4m3fnuz> {
|
||||
public:
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 4;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 3;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -6;
|
||||
static constexpr int min_exponent10 = -1;
|
||||
static constexpr int max_exponent = 8;
|
||||
static constexpr int max_exponent10 = 2;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before = false;
|
||||
|
||||
static constexpr c10::Float8_e4m3fnuz min() {
|
||||
return c10::Float8_e4m3fnuz(0x08, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz lowest() {
|
||||
return c10::Float8_e4m3fnuz(0xFF, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz max() {
|
||||
return c10::Float8_e4m3fnuz(0x7F, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz epsilon() {
|
||||
return c10::Float8_e4m3fnuz(0x28, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz round_error() {
|
||||
return c10::Float8_e4m3fnuz(0x38, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz infinity() {
|
||||
// NaN (no infinities)
|
||||
return c10::Float8_e4m3fnuz(0x80, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz quiet_NaN() {
|
||||
return c10::Float8_e4m3fnuz(0x80, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e4m3fnuz denorm_min() {
|
||||
return c10::Float8_e4m3fnuz(0x01, c10::Float8_e4m3fnuz::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
456
torch/headeronly/util/Float8_e5m2.h
Normal file
456
torch/headeronly/util/Float8_e5m2.h
Normal file
@ -0,0 +1,456 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e5m2 type (8-bit floating-point) including conversions
|
||||
/// to standard C types and basic arithmetic operations. Note that arithmetic
|
||||
/// operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration:
|
||||
/// s eeeee mm
|
||||
/// 1 sign bit
|
||||
/// 5 exponent bits
|
||||
/// 2 mantissa bits
|
||||
/// bias = 15
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2209.05433.pdf
|
||||
/// and inspired by Half implementation from pytorch/c10/util/Half.h
|
||||
|
||||
#include <torch/headeronly/util/Half.h>
|
||||
|
||||
#include <limits>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
struct alignas(1) Float8_e5m2 {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e5m2() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e5m2(uint8_t bits, from_bits_t) : x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e5m2(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
inline C10_HOST_DEVICE bool isinf() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(std::ostream& out, const Float8_e5m2& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 8-bit floating-point number in fp8 E5M2 format, in bit
|
||||
* representation, to a 32-bit floating-point number in IEEE single-precision
|
||||
* format, in bit representation.
|
||||
*
|
||||
* @note The implementation doesn't use any floating-point operations.
|
||||
*/
|
||||
inline C10_HOST_DEVICE float fp8e5m2_to_fp32_value(uint8_t input) {
|
||||
/*
|
||||
* Extend the fp8 E5M2 number to 32 bits and shift to the
|
||||
* upper part of the 32-bit word:
|
||||
* +---+----+---+-----------------------------+
|
||||
* | S |EEEEE|MM|0000 0000 0000 0000 0000 0000|
|
||||
* +---+----+---+-----------------------------+
|
||||
* Bits 31 26-30 24-25 0-23
|
||||
*
|
||||
* S - sign bit, E - bits of the biased exponent, M - bits of the mantissa, 0
|
||||
* - zero bits.
|
||||
*/
|
||||
uint16_t half_representation = input;
|
||||
half_representation <<= 8;
|
||||
return fp16_ieee_to_fp32_value(half_representation);
|
||||
}
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E5M2 format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e5m2_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of fp32 infinity
|
||||
* 0 11111111 00000000000000000000000
|
||||
*/
|
||||
constexpr uint32_t fp32_inf = UINT32_C(255) << 23;
|
||||
|
||||
/*
|
||||
* Binary representation of 65536.0f, which is the first value
|
||||
* not representable in fp8e5m2 range:
|
||||
* 0 11111 00 - fp8e5m2
|
||||
* 0 10001111 00000000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fp8_max = UINT32_C(143) << 23;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e5m2 normal range
|
||||
* into denorm representation
|
||||
* magic number: ((127 - 15) + (23 - 2) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(134) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
uint8_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fp8_max) {
|
||||
// NaN - all exponent and mantissa bits set to 1
|
||||
result = f_bits > fp32_inf ? UINT8_C(0x7F) : UINT8_C(0x7C);
|
||||
} else {
|
||||
if (f_bits < (UINT32_C(113) << 23)) {
|
||||
// Input number is smaller than 2^(-14), which is the smallest
|
||||
// fp8e5m2 normal number
|
||||
f_bits =
|
||||
fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint32_t mant_odd = (f_bits >> 21) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(15 - 127) << 23) + 0xFFFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 21);
|
||||
}
|
||||
}
|
||||
|
||||
result |= static_cast<uint8_t>(sign >> 24);
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
// -------- below is copied from c10/util/Float8_e5m2-inl.h --------//
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
#define EXP_WIDTH_FP8 5
|
||||
#define MAN_WIDTH_FP8 2
|
||||
#define EXP_BIAS_FP8 15
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2::Float8_e5m2(float value)
|
||||
: x(detail::fp8e5m2_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2::operator float() const {
|
||||
return detail::fp8e5m2_to_fp32_value(x);
|
||||
}
|
||||
|
||||
/// Special values helpers
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2::isnan() const {
|
||||
return (x & 0b01111111) > 0b01111100;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2::isinf() const {
|
||||
return (x & 0b01111111) == 0b01111100;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2
|
||||
operator+(const Float8_e5m2& a, const Float8_e5m2& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2
|
||||
operator-(const Float8_e5m2& a, const Float8_e5m2& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2
|
||||
operator*(const Float8_e5m2& a, const Float8_e5m2& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(
|
||||
const Float8_e5m2& a,
|
||||
const Float8_e5m2& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(const Float8_e5m2& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator+=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator-=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator*=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2& operator/=(
|
||||
Float8_e5m2& a,
|
||||
const Float8_e5m2& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e5m2 a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e5m2 a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e5m2 a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e5m2 a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e5m2 b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e5m2 b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e5m2 b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e5m2 b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e5m2& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e5m2& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e5m2& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e5m2& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e5m2 a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e5m2 a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e5m2 a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e5m2 a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e5m2 b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e5m2 b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e5m2 b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e5m2 b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(Float8_e5m2 a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(Float8_e5m2 a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(Float8_e5m2 a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(Float8_e5m2 a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(int a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(int a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(int a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(int a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(Float8_e5m2 a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(Float8_e5m2 a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(Float8_e5m2 a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(Float8_e5m2 a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e5m2>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator+(int64_t a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator-(int64_t a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator*(int64_t a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2 operator/(int64_t a, Float8_e5m2 b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e5m2 to float.
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
} // namespace c10
|
||||
|
||||
namespace torch::headeronly {
|
||||
using c10::Float8_e5m2;
|
||||
using c10::operator<<;
|
||||
using c10::operator+;
|
||||
using c10::operator-;
|
||||
using c10::operator*;
|
||||
using c10::operator/;
|
||||
using c10::operator+=;
|
||||
using c10::operator-=;
|
||||
using c10::operator*=;
|
||||
using c10::operator/=;
|
||||
|
||||
namespace detail {
|
||||
using c10::detail::fp8e5m2_from_fp32_value;
|
||||
using c10::detail::fp8e5m2_to_fp32_value;
|
||||
} // namespace detail
|
||||
} // namespace torch::headeronly
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e5m2> {
|
||||
public:
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = true;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 3;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 2;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -13;
|
||||
static constexpr int min_exponent10 = -4;
|
||||
static constexpr int max_exponent = 16;
|
||||
static constexpr int max_exponent10 = 4;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before =
|
||||
numeric_limits<float>::tinyness_before;
|
||||
|
||||
static constexpr c10::Float8_e5m2 min() {
|
||||
return c10::Float8_e5m2(0x4, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 max() {
|
||||
return c10::Float8_e5m2(0x7B, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 lowest() {
|
||||
return c10::Float8_e5m2(0xFB, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 epsilon() {
|
||||
return c10::Float8_e5m2(0x34, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 round_error() {
|
||||
return c10::Float8_e5m2(0x38, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 infinity() {
|
||||
return c10::Float8_e5m2(0x7C, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 quiet_NaN() {
|
||||
return c10::Float8_e5m2(0x7F, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2 denorm_min() {
|
||||
return c10::Float8_e5m2(0x01, c10::Float8_e5m2::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
446
torch/headeronly/util/Float8_e5m2fnuz.h
Normal file
446
torch/headeronly/util/Float8_e5m2fnuz.h
Normal file
@ -0,0 +1,446 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e5m2fnuz type (8-bit floating-point) including
|
||||
/// conversions to standard C types and basic arithmetic operations. Note that
|
||||
/// arithmetic operations are implemented by converting to floating point and
|
||||
/// performing the operation in float32.
|
||||
/// Binary configuration remains the same as e5m2:
|
||||
/// s eeeee mm
|
||||
/// 1 sign bit
|
||||
/// 5 exponent bits
|
||||
/// 2 mantissa bits
|
||||
/// The key differences that e5m2fnuz brings are:
|
||||
/// bias = 16
|
||||
/// no infinities or negative zero
|
||||
/// NaN only when sign bit is 1, rest all 0s
|
||||
///
|
||||
/// Implementation based on the paper https://arxiv.org/pdf/2206.02915.pdf and
|
||||
/// the existing Float8_e4m3fn implementation.
|
||||
|
||||
#include <torch/headeronly/macros/Macros.h>
|
||||
#include <torch/headeronly/util/Float8_fnuz_cvt.h>
|
||||
#include <torch/headeronly/util/TypeSafeSignMath.h>
|
||||
#include <torch/headeronly/util/floating_point_utils.h>
|
||||
|
||||
#if defined(__cplusplus)
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#include <iosfwd>
|
||||
#include <ostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
struct alignas(1) Float8_e5m2fnuz {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e5m2fnuz() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e5m2fnuz(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
inline C10_HOST_DEVICE bool isinf() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(
|
||||
std::ostream& out,
|
||||
const Float8_e5m2fnuz& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
namespace detail {
|
||||
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 E5M2 format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e5m2fnuz_from_fp32_value(float f) {
|
||||
/*
|
||||
* Binary representation of 65536.0f, which is the first value not
|
||||
* representable (i.e. the first value which would overflow in to the sign
|
||||
* bit, resulting in a NaN) in fp8e4m3fnuz range:
|
||||
* 1 00000 00 - fp8e5m2fnuz
|
||||
* 0 10001111 00000000000000000000000 - fp32
|
||||
*/
|
||||
constexpr uint32_t fnuz_max = UINT32_C(0x8F) << 23;
|
||||
|
||||
/*
|
||||
* A mask for converting fp32 numbers lower than fp8e5m2fnuz normal range
|
||||
* into denormalized representation.
|
||||
* magic number: ((127 - 16) + (23 - 2) + 1)
|
||||
*/
|
||||
constexpr uint32_t denorm_mask = UINT32_C(0x85) << 23;
|
||||
|
||||
uint32_t f_bits = fp32_to_bits(f);
|
||||
uint32_t result = 0u;
|
||||
|
||||
/*
|
||||
* Extract the sign of the input number into the high bit of the 32-bit word:
|
||||
*
|
||||
* +---+----------------------------------+
|
||||
* | S |0000000 00000000 00000000 00000000|
|
||||
* +---+----------------------------------+
|
||||
* Bits 31 0-31
|
||||
*/
|
||||
const uint32_t sign = f_bits & UINT32_C(0x80000000);
|
||||
|
||||
/*
|
||||
* Set sign bit to 0
|
||||
*/
|
||||
f_bits ^= sign;
|
||||
|
||||
if (f_bits >= fnuz_max) {
|
||||
// NaN -- sign bit set to 1, rest 0s
|
||||
return 0x80;
|
||||
}
|
||||
|
||||
if (f_bits < (UINT32_C(0x70) << 23) /* 2^-15 in float32 */) {
|
||||
// Input exponent is less than -15, the smallest e5m2fnuz exponent, so the
|
||||
// number will become subnormal.
|
||||
f_bits = fp32_to_bits(fp32_from_bits(f_bits) + fp32_from_bits(denorm_mask));
|
||||
result = static_cast<uint8_t>(f_bits - denorm_mask);
|
||||
if (result == 0) {
|
||||
// fnuz types don't have negative zero.
|
||||
return 0;
|
||||
}
|
||||
} else {
|
||||
// resulting mantissa is odd
|
||||
uint8_t mant_odd = (f_bits >> 21) & 1;
|
||||
|
||||
// update exponent, rounding bias part 1
|
||||
f_bits += ((uint32_t)(16 - 127) << 23) + 0xFFFFF;
|
||||
|
||||
// rounding bias part 2
|
||||
f_bits += mant_odd;
|
||||
|
||||
// take the bits!
|
||||
result = static_cast<uint8_t>(f_bits >> 21);
|
||||
}
|
||||
|
||||
result |= sign >> 24;
|
||||
return result;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
//------ below is copied from c10/util/Float8_e5m2fnuz-inl.h ------//
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
/// Constructors
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz::Float8_e5m2fnuz(float value)
|
||||
: x(detail::fp8e5m2fnuz_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz::operator float() const {
|
||||
return torch::headeronly::detail::fp8_fnuz_to_fp32_value<5, 2>(x);
|
||||
}
|
||||
|
||||
/// Special values helpers
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2fnuz::isnan() const {
|
||||
return x == 0b10000000;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e5m2fnuz::isinf() const {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Arithmetic
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz
|
||||
operator+(const Float8_e5m2fnuz& a, const Float8_e5m2fnuz& b) {
|
||||
return static_cast<float>(a) + static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz
|
||||
operator-(const Float8_e5m2fnuz& a, const Float8_e5m2fnuz& b) {
|
||||
return static_cast<float>(a) - static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz
|
||||
operator*(const Float8_e5m2fnuz& a, const Float8_e5m2fnuz& b) {
|
||||
return static_cast<float>(a) * static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(
|
||||
const Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) __ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(const Float8_e5m2fnuz& a) {
|
||||
return -static_cast<float>(a);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator+=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a + b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator-=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a - b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator*=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a * b;
|
||||
return a;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz& operator/=(
|
||||
Float8_e5m2fnuz& a,
|
||||
const Float8_e5m2fnuz& b) {
|
||||
a = a / b;
|
||||
return a;
|
||||
}
|
||||
|
||||
/// Arithmetic with floats
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(Float8_e5m2fnuz a, float b) {
|
||||
return static_cast<float>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(Float8_e5m2fnuz a, float b) {
|
||||
return static_cast<float>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(Float8_e5m2fnuz a, float b) {
|
||||
return static_cast<float>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(Float8_e5m2fnuz a, float b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<float>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float operator+(float a, Float8_e5m2fnuz b) {
|
||||
return a + static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator-(float a, Float8_e5m2fnuz b) {
|
||||
return a - static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator*(float a, Float8_e5m2fnuz b) {
|
||||
return a * static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float operator/(float a, Float8_e5m2fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<float>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE float& operator+=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a += static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator-=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a -= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator*=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a *= static_cast<float>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE float& operator/=(float& a, const Float8_e5m2fnuz& b) {
|
||||
return a /= static_cast<float>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with doubles
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(Float8_e5m2fnuz a, double b) {
|
||||
return static_cast<double>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(Float8_e5m2fnuz a, double b) {
|
||||
return static_cast<double>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(Float8_e5m2fnuz a, double b) {
|
||||
return static_cast<double>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(Float8_e5m2fnuz a, double b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return static_cast<double>(a) / b;
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE double operator+(double a, Float8_e5m2fnuz b) {
|
||||
return a + static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator-(double a, Float8_e5m2fnuz b) {
|
||||
return a - static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator*(double a, Float8_e5m2fnuz b) {
|
||||
return a * static_cast<double>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE double operator/(double a, Float8_e5m2fnuz b)
|
||||
__ubsan_ignore_float_divide_by_zero__ {
|
||||
return a / static_cast<double>(b);
|
||||
}
|
||||
|
||||
/// Arithmetic with ints
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(Float8_e5m2fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(Float8_e5m2fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(Float8_e5m2fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(Float8_e5m2fnuz a, int b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(int a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(int a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(int a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(int a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) / b;
|
||||
}
|
||||
|
||||
//// Arithmetic with int64_t
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(Float8_e5m2fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a + static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(Float8_e5m2fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a - static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(Float8_e5m2fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a * static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(Float8_e5m2fnuz a, int64_t b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return a / static_cast<Float8_e5m2fnuz>(b);
|
||||
}
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator+(int64_t a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) + b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator-(int64_t a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) - b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator*(int64_t a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) * b;
|
||||
}
|
||||
inline C10_HOST_DEVICE Float8_e5m2fnuz operator/(int64_t a, Float8_e5m2fnuz b) {
|
||||
// NOLINTNEXTLINE(bugprone-narrowing-conversions,cppcoreguidelines-narrowing-conversions)
|
||||
return static_cast<Float8_e5m2fnuz>(a) / b;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e5m2fnuz to float.
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace torch::headeronly {
|
||||
using c10::Float8_e5m2fnuz;
|
||||
using c10::operator<<;
|
||||
using c10::operator+;
|
||||
using c10::operator-;
|
||||
using c10::operator*;
|
||||
using c10::operator/;
|
||||
using c10::operator+=;
|
||||
using c10::operator-=;
|
||||
using c10::operator*=;
|
||||
using c10::operator/=;
|
||||
|
||||
namespace detail {
|
||||
using c10::detail::fp8e5m2fnuz_from_fp32_value;
|
||||
}
|
||||
} // namespace torch::headeronly
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e5m2fnuz> {
|
||||
public:
|
||||
static constexpr bool is_signed = true;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = true;
|
||||
static constexpr auto has_denorm_loss = true;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 3;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 2;
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -14;
|
||||
static constexpr int min_exponent10 = -4;
|
||||
static constexpr int max_exponent = 16;
|
||||
static constexpr int max_exponent10 = 4;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before =
|
||||
numeric_limits<float>::tinyness_before;
|
||||
|
||||
static constexpr c10::Float8_e5m2fnuz min() {
|
||||
return c10::Float8_e5m2fnuz(0x04, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz max() {
|
||||
return c10::Float8_e5m2fnuz(0x7F, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz lowest() {
|
||||
return c10::Float8_e5m2fnuz(0xFF, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz epsilon() {
|
||||
return c10::Float8_e5m2fnuz(0x34, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz round_error() {
|
||||
return c10::Float8_e5m2fnuz(0x38, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz infinity() {
|
||||
return c10::Float8_e5m2fnuz(0x80, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
// TODO(future): we are mapping neg_zero to both inf and NaN, this is
|
||||
// surprising and we should figure out what to do about it.
|
||||
static constexpr c10::Float8_e5m2fnuz quiet_NaN() {
|
||||
return c10::Float8_e5m2fnuz(0x80, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e5m2fnuz denorm_min() {
|
||||
return c10::Float8_e5m2fnuz(0x01, c10::Float8_e5m2fnuz::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
226
torch/headeronly/util/Float8_e8m0fnu.h
Normal file
226
torch/headeronly/util/Float8_e8m0fnu.h
Normal file
@ -0,0 +1,226 @@
|
||||
#pragma once
|
||||
|
||||
/// Defines the Float8_e8m0fnu type (8-bit floating-point) including
|
||||
/// conversions to standard C types
|
||||
/// Binary configuration :
|
||||
/// eeeeeeee
|
||||
/// no sign bits
|
||||
/// 8 exponent bits
|
||||
/// no mantissa bits
|
||||
///
|
||||
/// This is the E8M0 dtype from the OCP MX format spec
|
||||
/// (https://www.opencompute.org/documents/ocp-microscaling-formats-mx-v1-0-spec-final-pdf,
|
||||
/// Section 5.4.1)
|
||||
|
||||
#include <torch/headeronly/macros/Macros.h>
|
||||
#include <torch/headeronly/util/floating_point_utils.h>
|
||||
|
||||
// TODO(#146647): do we need to special case OPENCL?
|
||||
#if defined(__cplusplus)
|
||||
#include <cstdint>
|
||||
#elif !defined(__OPENCL_VERSION__)
|
||||
#include <math.h>
|
||||
#include <stdint.h>
|
||||
#endif
|
||||
|
||||
#include <iosfwd>
|
||||
#include <limits>
|
||||
#include <ostream>
|
||||
|
||||
namespace c10 {
|
||||
|
||||
struct alignas(1) Float8_e8m0fnu {
|
||||
uint8_t x;
|
||||
|
||||
struct from_bits_t {};
|
||||
C10_HOST_DEVICE static constexpr from_bits_t from_bits() {
|
||||
return from_bits_t();
|
||||
}
|
||||
|
||||
Float8_e8m0fnu() = default;
|
||||
|
||||
constexpr C10_HOST_DEVICE Float8_e8m0fnu(uint8_t bits, from_bits_t)
|
||||
: x(bits) {}
|
||||
inline C10_HOST_DEVICE Float8_e8m0fnu(float value);
|
||||
inline C10_HOST_DEVICE operator float() const;
|
||||
inline C10_HOST_DEVICE bool isnan() const;
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(
|
||||
std::ostream& out,
|
||||
const Float8_e8m0fnu& value) {
|
||||
out << (float)value;
|
||||
return out;
|
||||
}
|
||||
|
||||
namespace detail {
|
||||
/*
|
||||
* Convert a 32-bit floating-point number in IEEE single-precision format to a
|
||||
* 8-bit floating-point number in fp8 e8m0fnu format, in bit representation.
|
||||
*/
|
||||
inline C10_HOST_DEVICE uint8_t fp8e8m0fnu_from_fp32_value(float f) {
|
||||
// TODO(#146647): maybe rewrite without control flow
|
||||
|
||||
uint32_t f_bits = c10::detail::fp32_to_bits(f);
|
||||
|
||||
// extract the exponent
|
||||
uint32_t exponent = (f_bits >> 23) & 0b11111111;
|
||||
|
||||
// special case float32 NaN and +-inf to map to e8m0 nan
|
||||
if (exponent == 0b11111111) {
|
||||
return exponent;
|
||||
}
|
||||
|
||||
// next, we use guard, round, sticky bits and the LSB to implement round to
|
||||
// nearest, with ties to even
|
||||
|
||||
// guard bit - bit 23, or 22 zero-indexed
|
||||
uint8_t g = (f_bits & 0x400000) > 0;
|
||||
// round bit - bit 22, or 21 zero-indexed
|
||||
uint8_t r = (f_bits & 0x200000) > 0;
|
||||
// sticky bit - bits 21 to 1, or 20 to 0 zero-indexed
|
||||
uint8_t s = (f_bits & 0x1FFFFF) > 0;
|
||||
// in casting to e8m0, LSB is the implied mantissa bit. It equals to 0 if the
|
||||
// original float32 is denormal, and to 1 if the original float32 is normal.
|
||||
uint8_t lsb = exponent > 0;
|
||||
|
||||
// implement the RNE logic
|
||||
bool round_up = false;
|
||||
|
||||
// if g == 0, round down (no-op)
|
||||
if (g == 1) {
|
||||
if ((r == 1) || (s == 1)) {
|
||||
// round up
|
||||
round_up = true;
|
||||
} else {
|
||||
if (lsb == 1) {
|
||||
// round up
|
||||
round_up = true;
|
||||
}
|
||||
// if lsb == 0, round down (no-op)
|
||||
}
|
||||
}
|
||||
|
||||
if (round_up) {
|
||||
// adjust exponent
|
||||
// note that if exponent was 255 we would have already returned earlier, so
|
||||
// we know we can add one safely without running out of bounds
|
||||
exponent++;
|
||||
}
|
||||
|
||||
return exponent;
|
||||
}
|
||||
|
||||
} // namespace detail
|
||||
|
||||
//------- the below is from c10/util/Float8_e8m0fnu-inl.h ------//
|
||||
// TODO(#146647): Can we remove the below warning?
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
/// Constructors
|
||||
inline C10_HOST_DEVICE Float8_e8m0fnu::Float8_e8m0fnu(float value)
|
||||
: x(detail::fp8e8m0fnu_from_fp32_value(value)) {}
|
||||
|
||||
/// Implicit conversions
|
||||
|
||||
inline C10_HOST_DEVICE Float8_e8m0fnu::operator float() const {
|
||||
// TODO(#146647): maybe rewrite without control flow
|
||||
|
||||
// if exponent is zero, need to special case to return 2^-127 instead of zero
|
||||
if (x == 0) {
|
||||
return c10::detail::fp32_from_bits(0x00400000);
|
||||
}
|
||||
|
||||
// if exponent is NaN, need to special case to return properly encoded NaN
|
||||
if (isnan()) {
|
||||
return c10::detail::fp32_from_bits(0x7f800001);
|
||||
}
|
||||
|
||||
// leave sign at 0, set the exponent bits, leave stored mantissa at 0
|
||||
uint32_t res = x << 23;
|
||||
|
||||
return c10::detail::fp32_from_bits(res);
|
||||
}
|
||||
|
||||
/// Special values helper
|
||||
|
||||
inline C10_HOST_DEVICE bool Float8_e8m0fnu::isnan() const {
|
||||
return x == 0b11111111;
|
||||
}
|
||||
|
||||
/// NOTE: we do not define comparisons directly and instead rely on the implicit
|
||||
/// conversion from c10::Float8_e8m0fnu to float.
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
|
||||
} // namespace c10
|
||||
|
||||
namespace torch::headeronly {
|
||||
using c10::Float8_e8m0fnu;
|
||||
using c10::operator<<;
|
||||
|
||||
namespace detail {
|
||||
using c10::detail::fp8e8m0fnu_from_fp32_value;
|
||||
} // namespace detail
|
||||
} // namespace torch::headeronly
|
||||
|
||||
namespace std {
|
||||
|
||||
template <>
|
||||
class numeric_limits<c10::Float8_e8m0fnu> {
|
||||
public:
|
||||
static constexpr bool is_specialized = true;
|
||||
static constexpr bool is_signed = false;
|
||||
static constexpr bool is_integer = false;
|
||||
static constexpr bool is_exact = false;
|
||||
static constexpr bool has_infinity = false;
|
||||
static constexpr bool has_quiet_NaN = true;
|
||||
static constexpr bool has_signaling_NaN = false;
|
||||
static constexpr auto has_denorm = false;
|
||||
static constexpr auto has_denorm_loss = false;
|
||||
static constexpr auto round_style = numeric_limits<float>::round_style;
|
||||
static constexpr bool is_iec559 = false;
|
||||
static constexpr bool is_bounded = true;
|
||||
static constexpr bool is_modulo = false;
|
||||
static constexpr int digits = 1;
|
||||
static constexpr int digits10 = 0;
|
||||
static constexpr int max_digits10 = 1; // just a 2!
|
||||
static constexpr int radix = 2;
|
||||
static constexpr int min_exponent = -126;
|
||||
static constexpr int min_exponent10 = -38;
|
||||
static constexpr int max_exponent = 128;
|
||||
static constexpr int max_exponent10 = 38;
|
||||
static constexpr auto traps = numeric_limits<float>::traps;
|
||||
static constexpr auto tinyness_before = false;
|
||||
|
||||
static constexpr c10::Float8_e8m0fnu min() {
|
||||
// 2^-127
|
||||
return c10::Float8_e8m0fnu(0b00000000, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu lowest() {
|
||||
// 2^-127
|
||||
return c10::Float8_e8m0fnu(0b00000000, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu max() {
|
||||
// 254 biased, which is 127 unbiased, so 2^127
|
||||
return c10::Float8_e8m0fnu(0b11111110, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu epsilon() {
|
||||
// according to https://en.cppreference.com/w/cpp/types/numeric_limits, this
|
||||
// is "the difference between 1.0 and the next representable value of the
|
||||
// given floating-point type". The next representable value is 2.0, so the
|
||||
// difference is 1.0 which is 2^0. 0 unbiased is 127 biased.
|
||||
return c10::Float8_e8m0fnu(0b01111111, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu round_error() {
|
||||
// 0.5 in float, which is 2^-1, and -1 + 127 = 126
|
||||
return c10::Float8_e8m0fnu(0b01111110, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
static constexpr c10::Float8_e8m0fnu quiet_NaN() {
|
||||
return c10::Float8_e8m0fnu(0b11111111, c10::Float8_e8m0fnu::from_bits());
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace std
|
@ -1,6 +1,6 @@
|
||||
#pragma once
|
||||
|
||||
#include <c10/util/floating_point_utils.h>
|
||||
#include <torch/headeronly/util/floating_point_utils.h>
|
||||
|
||||
#include <cstdint>
|
||||
|
||||
@ -8,7 +8,7 @@
|
||||
#include <sycl/sycl.hpp>
|
||||
#endif
|
||||
|
||||
namespace c10::detail {
|
||||
namespace torch::headeronly::detail {
|
||||
|
||||
/*
|
||||
* Convert a 8-bit floating-point number in either f8 E4M3FNUZ or bf8 E5M2FNUZ
|
||||
@ -61,4 +61,8 @@ inline C10_HOST_DEVICE float fp8_fnuz_to_fp32_value(uint8_t x) {
|
||||
return fp32_from_bits(retval);
|
||||
}
|
||||
|
||||
} // namespace c10::detail
|
||||
} // namespace torch::headeronly::detail
|
||||
|
||||
namespace c10::detail {
|
||||
using torch::headeronly::detail::fp8_fnuz_to_fp32_value;
|
||||
}
|
148
torch/headeronly/util/TypeSafeSignMath.h
Normal file
148
torch/headeronly/util/TypeSafeSignMath.h
Normal file
@ -0,0 +1,148 @@
|
||||
#pragma once
|
||||
|
||||
#include <torch/headeronly/macros/Macros.h>
|
||||
#include <limits>
|
||||
#include <type_traits>
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_PUSH()
|
||||
#if C10_CLANG_HAS_WARNING("-Wstring-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wstring-conversion")
|
||||
#endif
|
||||
#if C10_CLANG_HAS_WARNING("-Wimplicit-int-float-conversion")
|
||||
C10_CLANG_DIAGNOSTIC_IGNORE("-Wimplicit-int-float-conversion")
|
||||
#endif
|
||||
|
||||
namespace c10 {
|
||||
|
||||
/// Returns false since we cannot have x < 0 if x is unsigned.
|
||||
template <typename T>
|
||||
inline constexpr bool is_negative(
|
||||
const T& /*x*/,
|
||||
std::true_type /*is_unsigned*/) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Returns true if a signed variable x < 0
|
||||
template <typename T>
|
||||
inline constexpr bool is_negative(const T& x, std::false_type /*is_unsigned*/) {
|
||||
return x < T(0);
|
||||
}
|
||||
|
||||
/// Returns true if x < 0
|
||||
/// NOTE: Will fail on an unsigned custom type
|
||||
/// For the most part it's possible to fix this if
|
||||
/// the custom type has a constexpr constructor.
|
||||
/// However, notably, c10::Half does not :-(
|
||||
template <typename T>
|
||||
inline constexpr bool is_negative(const T& x) {
|
||||
return is_negative(x, std::is_unsigned<T>());
|
||||
}
|
||||
|
||||
/// Returns the sign of an unsigned variable x as 0, 1
|
||||
template <typename T>
|
||||
inline constexpr int signum(const T& x, std::true_type /*is_unsigned*/) {
|
||||
return T(0) < x;
|
||||
}
|
||||
|
||||
/// Returns the sign of a signed variable x as -1, 0, 1
|
||||
template <typename T>
|
||||
inline constexpr int signum(const T& x, std::false_type /*is_unsigned*/) {
|
||||
return (T(0) < x) - (x < T(0));
|
||||
}
|
||||
|
||||
/// Returns the sign of x as -1, 0, 1
|
||||
/// NOTE: Will fail on an unsigned custom type
|
||||
/// For the most part it's possible to fix this if
|
||||
/// the custom type has a constexpr constructor.
|
||||
/// However, notably, c10::Half does not :-(
|
||||
template <typename T>
|
||||
inline constexpr int signum(const T& x) {
|
||||
return signum(x, std::is_unsigned<T>());
|
||||
}
|
||||
|
||||
/// Returns true if a and b are not both negative
|
||||
template <typename T, typename U>
|
||||
inline constexpr bool signs_differ(const T& a, const U& b) {
|
||||
return is_negative(a) != is_negative(b);
|
||||
}
|
||||
|
||||
// Suppress sign compare warning when compiling with GCC
|
||||
// as later does not account for short-circuit rule before
|
||||
// raising the warning, see https://godbolt.org/z/Tr3Msnz99
|
||||
#ifdef __GNUC__
|
||||
#pragma GCC diagnostic push
|
||||
#pragma GCC diagnostic ignored "-Wsign-compare"
|
||||
#endif
|
||||
|
||||
/// Returns true if x is greater than the greatest value of the type Limit
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool greater_than_max(const T& x) {
|
||||
constexpr bool can_overflow =
|
||||
std::numeric_limits<T>::digits > std::numeric_limits<Limit>::digits;
|
||||
return can_overflow && x > (std::numeric_limits<Limit>::max)();
|
||||
}
|
||||
|
||||
#ifdef __GNUC__
|
||||
#pragma GCC diagnostic pop
|
||||
#endif
|
||||
|
||||
/// Returns true if x < lowest(Limit). Standard comparison
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& x,
|
||||
std::false_type /*limit_is_unsigned*/,
|
||||
std::false_type /*x_is_unsigned*/) {
|
||||
return x < std::numeric_limits<Limit>::lowest();
|
||||
}
|
||||
|
||||
/// Returns false since all the limit is signed and therefore includes
|
||||
/// negative values but x cannot be negative because it is unsigned
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& /*x*/,
|
||||
std::false_type /*limit_is_unsigned*/,
|
||||
std::true_type /*x_is_unsigned*/) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Returns true if x < 0, where 0 is constructed from T.
|
||||
/// Limit is not signed, so its lower value is zero
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& x,
|
||||
std::true_type /*limit_is_unsigned*/,
|
||||
std::false_type /*x_is_unsigned*/) {
|
||||
return x < T(0);
|
||||
}
|
||||
|
||||
/// Returns false sign both types are unsigned
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(
|
||||
const T& /*x*/,
|
||||
std::true_type /*limit_is_unsigned*/,
|
||||
std::true_type /*x_is_unsigned*/) {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Returns true if x is less than the lowest value of type T
|
||||
/// NOTE: Will fail on an unsigned custom type
|
||||
/// For the most part it's possible to fix this if
|
||||
/// the custom type has a constexpr constructor.
|
||||
/// However, notably, c10::Half does not :
|
||||
template <typename Limit, typename T>
|
||||
inline constexpr bool less_than_lowest(const T& x) {
|
||||
return less_than_lowest<Limit>(
|
||||
x, std::is_unsigned<Limit>(), std::is_unsigned<T>());
|
||||
}
|
||||
|
||||
} // namespace c10
|
||||
|
||||
C10_CLANG_DIAGNOSTIC_POP()
|
||||
|
||||
namespace torch::headeronly {
|
||||
using c10::greater_than_max;
|
||||
using c10::is_negative;
|
||||
using c10::less_than_lowest;
|
||||
using c10::signs_differ;
|
||||
using c10::signum;
|
||||
} // namespace torch::headeronly
|
Reference in New Issue
Block a user