//===-- Implementation header for erfcf16 -----------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #ifndef LLVM_LIBC_SRC___SUPPORT_MATH_ERFCF16_H #define LLVM_LIBC_SRC___SUPPORT_MATH_ERFCF16_H #include "include/llvm-libc-macros/float16-macros.h" #ifdef LIBC_TYPES_HAS_FLOAT16 #include "src/__support/FPUtil/FEnvImpl.h" #include "src/__support/FPUtil/FPBits.h" #include "src/__support/FPUtil/cast.h" #include "src/__support/FPUtil/except_value_utils.h" #include "src/__support/FPUtil/multiply_add.h" #include "src/__support/macros/config.h" #include "src/__support/macros/optimization.h" // LIBC_UNLIKELY namespace LIBC_NAMESPACE_DECL { namespace math { LIBC_INLINE float16 erfcf16(float16 x) { // Polynomials approximating erfc(|x|) on ( k/2, (k+1)/2 ) generated by // Sollya with: > P = fpminimax(erfc(x), [|0, 1, 2, 3, 4, 5, 6, 7|], [|D...|], // [k/2, (k+1)/2]); // // for k = 0..7. constexpr double COEFFS[8][8] = { {0x1.00000006e5898p0, -0x1.20dd7b4435481p0, 0x1.e2ffc0264c37cp-17, 0x1.80ef5566d3135p-2, 0x1.96ef459387c13p-10, -0x1.e6f847a52d3fep-4, 0x1.9a59af64a5dfap-7, 0x1.f652f3b14963bp-7}, {0x1.001a9f1d3c0b4p0, -0x1.221a42637ae8p0, 0x1.9c3ddc1e5d176p-6, 0x1.347481936316bp-2, 0x1.1db66a2746b8bp-3, -0x1.1d7a264182166p-2, 0x1.ef858095e1609p-4, -0x1.26936198d6b07p-6}, {0x1.f08a38741577bp-1, -0x1.e26ae90822fp-1, -0x1.f300a88478724p-2, 0x1.115a0580ba3f8p0, -0x1.1a1c2bc3ffcdap-1, 0x1.888d0c9a082b8p-4, 0x1.f41c9f1877fdfp-8, -0x1.a72fb878df30bp-9}, {0x1.c1cabe622be64p-1, -0x1.ee19257d1037ap-2, -0x1.7a725229a24b8p0, 0x1.2089bc62b1069p1, -0x1.673f1b9fbced5p0, 0x1.da7db686b0475p-2, -0x1.49a292662ca6ap-4, 0x1.7e2b298da504dp-8}, {0x1.a466e958ca525p0, -0x1.8a850d50339a9p1, 0x1.29b741ccfdd3ap1, -0x1.b250e97982f77p-1, 0x1.ea6896c5fa419p-4, 0x1.b332978509bf1p-7, -0x1.9f1c5d9122108p-8, 0x1.2fb4d83883203p-11}, {0x1.12fb16acc5644p1, -0x1.25003c37e1b24p2, 0x1.0e0dc863b2f12p2, -0x1.16f179d8797a6p1, 0x1.5c8b140bf43f8p-1, -0x1.07473d994c2edp-3, 0x1.bd0c1c2d2a9e3p-7, -0x1.448767255aabbp-11}, {0x1.13a691333e22ap0, -0x1.0e2a642d8d318p1, 0x1.c7d88193df94bp0, -0x1.acf7c7b79498fp-1, 0x1.e626f6202a127p-3, -0x1.4babcc8609859p-5, 0x1.f860060a3658p-9, -0x1.49a4190580d4p-13}, {0x1.cf3a84bf655afp-3, -0x1.968e6988b5cep-2, 0x1.326f1f9499739p-2, -0x1.011785d112c9bp-3, 0x1.0343a0c05e285p-5, -0x1.3a3ae5e48130ep-8, 0x1.a7c5af0484892p-12, -0x1.ea82a062010c3p-17}, }; static constexpr size_t N_ERFCF16_EXCEPTS = 3; static constexpr fputil::ExceptValues ERFCF16_EXCEPTS = {{ // (input, RZ output, RU offset, RD offset, RN offset) // x = 0x0.000216, erfc(x) = 0x1.000 {0x0B17, 0x3BFF, 1, 0, 0}, // x = 0x0.00346, erfc(x) = 0x0.ff8 {0x1B17, 0x3BF7, 1, 0, 1}, // x = -0x0.00346, erfc(x) = 0x1.00c {0x9B17, 0x3C04, 1, 0, 0}, }}; using FPBits = typename fputil::FPBits; FPBits xbits(x); uint16_t x_u = xbits.uintval(); if (auto r = ERFCF16_EXCEPTS.lookup(x_u); LIBC_UNLIKELY(r.has_value())) return r.value(); uint16_t x_abs = xbits.abs().uintval(); // Special cases: NaN and Inf if (LIBC_UNLIKELY(x_abs >= 0x7c00U)) { if (x_abs > 0x7c00U) { if (xbits.is_signaling_nan()) { fputil::raise_except_if_required(FE_INVALID); return FPBits::quiet_nan().get_val(); } return x; } // erfc(+Inf) = 0, erfc(-Inf) = 2 return xbits.is_neg() ? fputil::cast(2.0) : fputil::cast(0.0); } if (LIBC_UNLIKELY(x_abs == 0)) return 1.0f16; // Asymptotic behavior: erfc(x) rounds to 0 or 2 for |x| >= 4.0. if (LIBC_UNLIKELY(x_abs >= 0x4400U)) { // |x| >= 4.0 if (xbits.is_neg()) { // 0x1.0p-12 is ~0.25 ULP of 2.0 in float16, small enough to round // to 2.0 in RN, but large enough to round down in RD/RZ. float xf = fputil::cast(x); return fputil::cast(2.0 - 0x1.0p-12 * (-4.0 / xf)); } fputil::set_errno_if_required(ERANGE); fputil::raise_except_if_required(FE_UNDERFLOW | FE_INEXACT); if (fputil::fenv_is_round_up()) return FPBits::min_subnormal().get_val(); return 0.0f16; } // Polynomial approximation: // erfc(x) ~ erfc(|x|) if x >= 0 // erfc(x) ~ 2 - erfc(|x|) if x < 0 // erfc(|x|) is evaluated using a degree-7 polynomial on each sub-interval. int idx = static_cast(xbits.abs().get_val() * 2.0f); double xd = fputil::cast(xbits.abs().get_val()); double xsq = xd * xd; double x4 = xsq * xsq; double p01 = fputil::multiply_add(xd, COEFFS[idx][1], COEFFS[idx][0]); double p23 = fputil::multiply_add(xd, COEFFS[idx][3], COEFFS[idx][2]); double p45 = fputil::multiply_add(xd, COEFFS[idx][5], COEFFS[idx][4]); double p67 = fputil::multiply_add(xd, COEFFS[idx][7], COEFFS[idx][6]); double p03 = fputil::multiply_add(xsq, p23, p01); double p47 = fputil::multiply_add(xsq, p67, p45); double erfc_abs = fputil::multiply_add(x4, p47, p03); if (xbits.is_neg()) return fputil::cast(2.0 - erfc_abs); return fputil::cast(erfc_abs); } } // namespace math } // namespace LIBC_NAMESPACE_DECL #endif // LIBC_TYPES_HAS_FLOAT16 #endif // LLVM_LIBC_SRC___SUPPORT_MATH_ERFCF16_H