//===-- FEnvSafeTest.cpp ---------------------------------------*- 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 // //===---------------------------------------------------------------------===// #ifdef LIBC_MATH_USE_SYSTEM_FENV #undef LIBC_MATH_USE_SYSTEM_FENV #endif // LIBC_MATH_USE_SYSTEM_FENV #include "FEnvSafeTest.h" #include "src/__support/FPUtil/FEnvImpl.h" #include "src/__support/libc_errno.h" #include "src/__support/macros/config.h" #include "src/__support/macros/properties/architectures.h" #include "test/UnitTest/ErrnoCheckingTest.h" namespace LIBC_NAMESPACE_DECL { namespace testing { void FEnvSafeTest::PreserveFEnv::check() { fenv_t after; test.get_fenv(after); test.expect_fenv_eq(before, after); } void FEnvSafeTest::TearDown() { if (!should_be_unchanged) { restore_fenv(); } // TODO (PR 135320): Remove this override once all FEnvSafeTest instances are // updated to validate or ignore errno. libc_errno = 0; ErrnoCheckingTest::TearDown(); } void FEnvSafeTest::get_fenv(fenv_t &fenv) { ASSERT_EQ(LIBC_NAMESPACE::fputil::get_env(&fenv), 0); } void FEnvSafeTest::set_fenv(const fenv_t &fenv) { ASSERT_EQ(LIBC_NAMESPACE::fputil::set_env(&fenv), 0); } void FEnvSafeTest::expect_fenv_eq(const fenv_t &before_fenv, const fenv_t &after_fenv) { #if defined(LIBC_TARGET_ARCH_IS_AARCH64) && !defined(LIBC_COMPILER_IS_MSVC) && \ defined(__ARM_FP) using FPState = LIBC_NAMESPACE::fputil::FEnv::FPState; const FPState &before_state = reinterpret_cast(before_fenv); const FPState &after_state = reinterpret_cast(after_fenv); EXPECT_EQ(before_state.ControlWord, after_state.ControlWord); EXPECT_EQ(before_state.StatusWord, after_state.StatusWord); #elif defined(LIBC_TARGET_ARCH_IS_X86) using LIBC_NAMESPACE::cpp::inline_copy; using LIBC_NAMESPACE::fputil::internal::X87StateDescriptor; if constexpr (sizeof(fenv_t) >= sizeof(X87StateDescriptor) + sizeof(uint32_t)) { const char *before_fenv_ptr = reinterpret_cast(&before_fenv); const char *after_fenv_ptr = reinterpret_cast(&after_fenv); X87StateDescriptor before_x87_state, after_x87_state; uint32_t before_mxcsr, after_mxcsr; inline_copy( before_fenv_ptr, reinterpret_cast(&before_x87_state)); inline_copy( after_fenv_ptr, reinterpret_cast(&after_x87_state)); inline_copy(before_fenv_ptr + sizeof(X87StateDescriptor), reinterpret_cast(&before_mxcsr)); inline_copy(after_fenv_ptr + sizeof(X87StateDescriptor), reinterpret_cast(&after_mxcsr)); EXPECT_EQ(before_x87_state.control_word, after_x87_state.control_word); EXPECT_EQ(before_x87_state.status_word, after_x87_state.status_word); EXPECT_EQ(before_mxcsr, after_mxcsr); } else if constexpr (sizeof(fenv_t) == sizeof(X87StateDescriptor)) { const X87StateDescriptor &before_state = reinterpret_cast(before_fenv); const X87StateDescriptor &after_state = reinterpret_cast(after_fenv); EXPECT_EQ(before_state.control_word, after_state.control_word); EXPECT_EQ(before_state.status_word, after_state.status_word); } else if constexpr (sizeof(fenv_t) == sizeof(uint64_t)) { const uint64_t &before_mxcsr = reinterpret_cast(before_fenv); const uint64_t &after_mxcsr = reinterpret_cast(after_fenv); EXPECT_EQ(before_mxcsr, after_mxcsr); } else if constexpr (sizeof(fenv_t) == sizeof(uint32_t)) { const uint32_t &before_mxcsr = reinterpret_cast(before_fenv); const uint32_t &after_mxcsr = reinterpret_cast(after_fenv); EXPECT_EQ(before_mxcsr, after_mxcsr); } #elif defined(LIBC_TARGET_ARCH_IS_ARM) && defined(__ARM_FP) using LIBC_NAMESPACE::fputil::FEnv; const FEnv &before_state = reinterpret_cast(before_fenv); const FEnv &after_state = reinterpret_cast(after_fenv); EXPECT_EQ(before_state.fpscr, after_state.fpscr); #elif defined(LIBC_TARGET_ARCH_IS_ANY_RISCV) const uint32_t &before_fcsr = reinterpret_cast(before_fenv); const uint32_t &after_fcsr = reinterpret_cast(after_fenv); EXPECT_EQ(before_fcsr, after_fcsr); #else // No arch-specific `fenv_t` support, so nothing to compare. (void)before_fenv; (void)after_fenv; #endif } } // namespace testing } // namespace LIBC_NAMESPACE_DECL