blob: c4694706b7b0e81657c35f4de6afc928c071721f [file] [log] [blame] [edit]
//===-- 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
//
//===---------------------------------------------------------------------===//
#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<const FPState &>(before_fenv);
const FPState &after_state = reinterpret_cast<const FPState &>(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<const char *>(&before_fenv);
const char *after_fenv_ptr = reinterpret_cast<const char *>(&after_fenv);
X87StateDescriptor before_x87_state, after_x87_state;
uint32_t before_mxcsr, after_mxcsr;
inline_copy<sizeof(X87StateDescriptor)>(
before_fenv_ptr, reinterpret_cast<char *>(&before_x87_state));
inline_copy<sizeof(X87StateDescriptor)>(
after_fenv_ptr, reinterpret_cast<char *>(&after_x87_state));
inline_copy<sizeof(uint32_t)>(before_fenv_ptr + sizeof(X87StateDescriptor),
reinterpret_cast<char *>(&before_mxcsr));
inline_copy<sizeof(uint32_t)>(after_fenv_ptr + sizeof(X87StateDescriptor),
reinterpret_cast<char *>(&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<const X87StateDescriptor &>(before_fenv);
const X87StateDescriptor &after_state =
reinterpret_cast<const X87StateDescriptor &>(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<const uint64_t &>(before_fenv);
const uint64_t &after_mxcsr =
reinterpret_cast<const uint64_t &>(after_fenv);
EXPECT_EQ(before_mxcsr, after_mxcsr);
} else if constexpr (sizeof(fenv_t) == sizeof(uint32_t)) {
const uint32_t &before_mxcsr =
reinterpret_cast<const uint32_t &>(before_fenv);
const uint32_t &after_mxcsr =
reinterpret_cast<const uint32_t &>(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<const FEnv &>(before_fenv);
const FEnv &after_state = reinterpret_cast<const FEnv &>(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<const uint32_t &>(before_fenv);
const uint32_t &after_fcsr = reinterpret_cast<const uint32_t &>(after_fenv);
EXPECT_EQ(before_fcsr, after_fcsr);
#else
// No arch-specific `fenv_t` support, so nothing to compare.
#endif
}
} // namespace testing
} // namespace LIBC_NAMESPACE_DECL