1*71db0c75SAndroid Build Coastguard Worker //===-- Utility class to test fixed-point sqrt ------------------*- C++ -*-===// 2*71db0c75SAndroid Build Coastguard Worker // 3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information. 5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*71db0c75SAndroid Build Coastguard Worker // 7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===// 8*71db0c75SAndroid Build Coastguard Worker 9*71db0c75SAndroid Build Coastguard Worker #include "test/UnitTest/Test.h" 10*71db0c75SAndroid Build Coastguard Worker 11*71db0c75SAndroid Build Coastguard Worker #include "src/__support/CPP/bit.h" 12*71db0c75SAndroid Build Coastguard Worker #include "src/__support/FPUtil/BasicOperations.h" 13*71db0c75SAndroid Build Coastguard Worker #include "src/__support/FPUtil/sqrt.h" 14*71db0c75SAndroid Build Coastguard Worker #include "src/__support/fixed_point/fx_rep.h" 15*71db0c75SAndroid Build Coastguard Worker #include "src/__support/fixed_point/sqrt.h" 16*71db0c75SAndroid Build Coastguard Worker 17*71db0c75SAndroid Build Coastguard Worker template <typename T> class SqrtTest : public LIBC_NAMESPACE::testing::Test { 18*71db0c75SAndroid Build Coastguard Worker 19*71db0c75SAndroid Build Coastguard Worker using FXRep = LIBC_NAMESPACE::fixed_point::FXRep<T>; 20*71db0c75SAndroid Build Coastguard Worker static constexpr T zero = FXRep::ZERO(); 21*71db0c75SAndroid Build Coastguard Worker static constexpr T min = FXRep::MIN(); 22*71db0c75SAndroid Build Coastguard Worker static constexpr T max = FXRep::MAX(); 23*71db0c75SAndroid Build Coastguard Worker static constexpr T half = static_cast<T>(0.5); 24*71db0c75SAndroid Build Coastguard Worker static constexpr T quarter = static_cast<T>(0.25); 25*71db0c75SAndroid Build Coastguard Worker static constexpr T one = 26*71db0c75SAndroid Build Coastguard Worker (FXRep::INTEGRAL_LEN > 0) ? static_cast<T>(1) : FXRep::MAX(); 27*71db0c75SAndroid Build Coastguard Worker static constexpr T eps = FXRep::EPS(); 28*71db0c75SAndroid Build Coastguard Worker 29*71db0c75SAndroid Build Coastguard Worker public: 30*71db0c75SAndroid Build Coastguard Worker typedef T (*SqrtFunc)(T); 31*71db0c75SAndroid Build Coastguard Worker testSpecialNumbers(SqrtFunc func)32*71db0c75SAndroid Build Coastguard Worker void testSpecialNumbers(SqrtFunc func) { 33*71db0c75SAndroid Build Coastguard Worker EXPECT_EQ(zero, func(zero)); 34*71db0c75SAndroid Build Coastguard Worker EXPECT_EQ(half, func(quarter)); 35*71db0c75SAndroid Build Coastguard Worker 36*71db0c75SAndroid Build Coastguard Worker if constexpr (FXRep::INTEGRAL_LEN) { 37*71db0c75SAndroid Build Coastguard Worker EXPECT_EQ(one, func(one)); 38*71db0c75SAndroid Build Coastguard Worker EXPECT_EQ(static_cast<T>(2.0), func(static_cast<T>(4.0))); 39*71db0c75SAndroid Build Coastguard Worker } 40*71db0c75SAndroid Build Coastguard Worker 41*71db0c75SAndroid Build Coastguard Worker using StorageType = typename FXRep::StorageType; 42*71db0c75SAndroid Build Coastguard Worker 43*71db0c75SAndroid Build Coastguard Worker constexpr size_t COUNT = 255; 44*71db0c75SAndroid Build Coastguard Worker constexpr StorageType STEP = 45*71db0c75SAndroid Build Coastguard Worker StorageType(~StorageType(0)) / static_cast<StorageType>(COUNT); 46*71db0c75SAndroid Build Coastguard Worker constexpr double ERR = 3.0 * static_cast<double>(eps); 47*71db0c75SAndroid Build Coastguard Worker StorageType x = 0; 48*71db0c75SAndroid Build Coastguard Worker for (size_t i = 0; i < COUNT; ++i, x += STEP) { 49*71db0c75SAndroid Build Coastguard Worker T v = LIBC_NAMESPACE::cpp::bit_cast<T>(x); 50*71db0c75SAndroid Build Coastguard Worker double v_d = static_cast<double>(v); 51*71db0c75SAndroid Build Coastguard Worker double errors = LIBC_NAMESPACE::fputil::abs( 52*71db0c75SAndroid Build Coastguard Worker static_cast<double>(func(v)) - 53*71db0c75SAndroid Build Coastguard Worker LIBC_NAMESPACE::fputil::sqrt<double>(v_d)); 54*71db0c75SAndroid Build Coastguard Worker if (errors > ERR) { 55*71db0c75SAndroid Build Coastguard Worker // Print out the failure input and output. 56*71db0c75SAndroid Build Coastguard Worker EXPECT_EQ(v, zero); 57*71db0c75SAndroid Build Coastguard Worker EXPECT_EQ(func(v), zero); 58*71db0c75SAndroid Build Coastguard Worker } 59*71db0c75SAndroid Build Coastguard Worker ASSERT_TRUE(errors <= ERR); 60*71db0c75SAndroid Build Coastguard Worker } 61*71db0c75SAndroid Build Coastguard Worker } 62*71db0c75SAndroid Build Coastguard Worker }; 63*71db0c75SAndroid Build Coastguard Worker 64*71db0c75SAndroid Build Coastguard Worker #define LIST_SQRT_TESTS(T, func) \ 65*71db0c75SAndroid Build Coastguard Worker using LlvmLibcSqrtTest = SqrtTest<T>; \ 66*71db0c75SAndroid Build Coastguard Worker TEST_F(LlvmLibcSqrtTest, SpecialNumbers) { testSpecialNumbers(&func); } \ 67*71db0c75SAndroid Build Coastguard Worker static_assert(true, "Require semicolon.") 68