xref: /aosp_15_r20/external/llvm-libc/test/src/math/smoke/SetPayloadTest.h (revision 71db0c75aadcf003ffe3238005f61d7618a3fead)
1*71db0c75SAndroid Build Coastguard Worker //===-- Utility class to test different flavors of setpayload ---*- 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 #ifndef LIBC_TEST_SRC_MATH_SMOKE_SETPAYLOADTEST_H
10*71db0c75SAndroid Build Coastguard Worker #define LIBC_TEST_SRC_MATH_SMOKE_SETPAYLOADTEST_H
11*71db0c75SAndroid Build Coastguard Worker 
12*71db0c75SAndroid Build Coastguard Worker #include "test/UnitTest/FEnvSafeTest.h"
13*71db0c75SAndroid Build Coastguard Worker #include "test/UnitTest/FPMatcher.h"
14*71db0c75SAndroid Build Coastguard Worker #include "test/UnitTest/Test.h"
15*71db0c75SAndroid Build Coastguard Worker 
16*71db0c75SAndroid Build Coastguard Worker using LIBC_NAMESPACE::Sign;
17*71db0c75SAndroid Build Coastguard Worker 
18*71db0c75SAndroid Build Coastguard Worker template <typename T>
19*71db0c75SAndroid Build Coastguard Worker class SetPayloadTestTemplate : public LIBC_NAMESPACE::testing::FEnvSafeTest {
20*71db0c75SAndroid Build Coastguard Worker 
21*71db0c75SAndroid Build Coastguard Worker   DECLARE_SPECIAL_CONSTANTS(T)
22*71db0c75SAndroid Build Coastguard Worker 
23*71db0c75SAndroid Build Coastguard Worker public:
24*71db0c75SAndroid Build Coastguard Worker   typedef int (*SetPayloadFunc)(T *, T);
25*71db0c75SAndroid Build Coastguard Worker 
testInvalidPayloads(SetPayloadFunc func)26*71db0c75SAndroid Build Coastguard Worker   void testInvalidPayloads(SetPayloadFunc func) {
27*71db0c75SAndroid Build Coastguard Worker     T res;
28*71db0c75SAndroid Build Coastguard Worker 
29*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(aNaN)));
30*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(neg_aNaN)));
31*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(inf)));
32*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(neg_inf)));
33*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(0.1)));
34*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(-0.1)));
35*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(-1.0)));
36*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(0x42.1p+0)));
37*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, T(-0x42.1p+0)));
38*71db0c75SAndroid Build Coastguard Worker 
39*71db0c75SAndroid Build Coastguard Worker     FPBits nan_payload_bits = FPBits::one();
40*71db0c75SAndroid Build Coastguard Worker     nan_payload_bits.set_biased_exponent(FPBits::FRACTION_LEN - 1 +
41*71db0c75SAndroid Build Coastguard Worker                                          FPBits::EXP_BIAS);
42*71db0c75SAndroid Build Coastguard Worker     T nan_payload = nan_payload_bits.get_val();
43*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(1, func(&res, nan_payload));
44*71db0c75SAndroid Build Coastguard Worker   }
45*71db0c75SAndroid Build Coastguard Worker 
testValidPayloads(SetPayloadFunc func)46*71db0c75SAndroid Build Coastguard Worker   void testValidPayloads(SetPayloadFunc func) {
47*71db0c75SAndroid Build Coastguard Worker     T res;
48*71db0c75SAndroid Build Coastguard Worker 
49*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(0, func(&res, T(0.0)));
50*71db0c75SAndroid Build Coastguard Worker     EXPECT_TRUE(FPBits(res).is_quiet_nan());
51*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(FPBits(aNaN).uintval(), FPBits(res).uintval());
52*71db0c75SAndroid Build Coastguard Worker 
53*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(0, func(&res, T(1.0)));
54*71db0c75SAndroid Build Coastguard Worker     EXPECT_TRUE(FPBits(res).is_quiet_nan());
55*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(FPBits::quiet_nan(Sign::POS, 1).uintval(), FPBits(res).uintval());
56*71db0c75SAndroid Build Coastguard Worker 
57*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(0, func(&res, T(0x42.0p+0)));
58*71db0c75SAndroid Build Coastguard Worker     EXPECT_TRUE(FPBits(res).is_quiet_nan());
59*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(FPBits::quiet_nan(Sign::POS, 0x42).uintval(),
60*71db0c75SAndroid Build Coastguard Worker               FPBits(res).uintval());
61*71db0c75SAndroid Build Coastguard Worker 
62*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(0, func(&res, T(0x123.0p+0)));
63*71db0c75SAndroid Build Coastguard Worker     EXPECT_TRUE(FPBits(res).is_quiet_nan());
64*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(FPBits::quiet_nan(Sign::POS, 0x123).uintval(),
65*71db0c75SAndroid Build Coastguard Worker               FPBits(res).uintval());
66*71db0c75SAndroid Build Coastguard Worker 
67*71db0c75SAndroid Build Coastguard Worker     // The following code is creating a NaN payload manually to prevent a
68*71db0c75SAndroid Build Coastguard Worker     // conversion from BigInt to float128.
69*71db0c75SAndroid Build Coastguard Worker     FPBits nan_payload_bits = FPBits::one();
70*71db0c75SAndroid Build Coastguard Worker     nan_payload_bits.set_biased_exponent(FPBits::FRACTION_LEN - 2 +
71*71db0c75SAndroid Build Coastguard Worker                                          FPBits::EXP_BIAS);
72*71db0c75SAndroid Build Coastguard Worker     nan_payload_bits.set_mantissa(FPBits::SIG_MASK - 3);
73*71db0c75SAndroid Build Coastguard Worker     T nan_payload = nan_payload_bits.get_val();
74*71db0c75SAndroid Build Coastguard Worker 
75*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(0, func(&res, nan_payload));
76*71db0c75SAndroid Build Coastguard Worker     EXPECT_TRUE(FPBits(res).is_quiet_nan());
77*71db0c75SAndroid Build Coastguard Worker     EXPECT_EQ(
78*71db0c75SAndroid Build Coastguard Worker         FPBits::quiet_nan(Sign::POS, FPBits::FRACTION_MASK >> 1).uintval(),
79*71db0c75SAndroid Build Coastguard Worker         FPBits(res).uintval());
80*71db0c75SAndroid Build Coastguard Worker   }
81*71db0c75SAndroid Build Coastguard Worker };
82*71db0c75SAndroid Build Coastguard Worker 
83*71db0c75SAndroid Build Coastguard Worker #define LIST_SETPAYLOAD_TESTS(T, func)                                         \
84*71db0c75SAndroid Build Coastguard Worker   using LlvmLibcSetPayloadTest = SetPayloadTestTemplate<T>;                    \
85*71db0c75SAndroid Build Coastguard Worker   TEST_F(LlvmLibcSetPayloadTest, InvalidPayloads) {                            \
86*71db0c75SAndroid Build Coastguard Worker     testInvalidPayloads(&func);                                                \
87*71db0c75SAndroid Build Coastguard Worker   }                                                                            \
88*71db0c75SAndroid Build Coastguard Worker   TEST_F(LlvmLibcSetPayloadTest, ValidPayloads) { testValidPayloads(&func); }
89*71db0c75SAndroid Build Coastguard Worker 
90*71db0c75SAndroid Build Coastguard Worker #endif // LIBC_TEST_SRC_MATH_SMOKE_SETPAYLOADTEST_H
91