| /* |
| * Copyright (c) 2016, Alliance for Open Media. All rights reserved |
| * |
| * This source code is subject to the terms of the BSD 2 Clause License and |
| * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License |
| * was not distributed with this source code in the LICENSE file, you can |
| * obtain it at www.aomedia.org/license/software. If the Alliance for Open |
| * Media Patent License 1.0 was not distributed with this source code in the |
| * PATENTS file, you can obtain it at www.aomedia.org/license/patent. |
| */ |
| |
| #ifndef TEST_REGISTER_STATE_CHECK_H_ |
| #define TEST_REGISTER_STATE_CHECK_H_ |
| |
| #include "third_party/googletest/src/googletest/include/gtest/gtest.h" |
| #include "./aom_config.h" |
| #include "aom/aom_integer.h" |
| |
| // ASM_REGISTER_STATE_CHECK(asm_function) |
| // Minimally validates the environment pre & post function execution. This |
| // variant should be used with assembly functions which are not expected to |
| // fully restore the system state. See platform implementations of |
| // RegisterStateCheck for details. |
| // |
| // API_REGISTER_STATE_CHECK(api_function) |
| // Performs all the checks done by ASM_REGISTER_STATE_CHECK() and any |
| // additional checks to ensure the environment is in a consistent state pre & |
| // post function execution. This variant should be used with API functions. |
| // See platform implementations of RegisterStateCheckXXX for details. |
| // |
| |
| #if defined(_WIN64) |
| |
| #undef NOMINMAX |
| #define NOMINMAX |
| #define WIN32_LEAN_AND_MEAN |
| #include <windows.h> |
| #include <winnt.h> |
| |
| inline bool operator==(const M128A &lhs, const M128A &rhs) { |
| return (lhs.Low == rhs.Low && lhs.High == rhs.High); |
| } |
| |
| namespace libaom_test { |
| |
| // Compares the state of xmm[6-15] at construction with their state at |
| // destruction. These registers should be preserved by the callee on |
| // Windows x64. |
| class RegisterStateCheck { |
| public: |
| RegisterStateCheck() { initialized_ = StoreRegisters(&pre_context_); } |
| ~RegisterStateCheck() { EXPECT_TRUE(Check()); } |
| |
| private: |
| static bool StoreRegisters(CONTEXT *const context) { |
| const HANDLE this_thread = GetCurrentThread(); |
| EXPECT_TRUE(this_thread != NULL); |
| context->ContextFlags = CONTEXT_FLOATING_POINT; |
| const bool context_saved = GetThreadContext(this_thread, context) == TRUE; |
| EXPECT_TRUE(context_saved) << "GetLastError: " << GetLastError(); |
| return context_saved; |
| } |
| |
| // Compares the register state. Returns true if the states match. |
| bool Check() const { |
| if (!initialized_) return false; |
| CONTEXT post_context; |
| if (!StoreRegisters(&post_context)) return false; |
| |
| const M128A *xmm_pre = &pre_context_.Xmm6; |
| const M128A *xmm_post = &post_context.Xmm6; |
| for (int i = 6; i <= 15; ++i) { |
| EXPECT_EQ(*xmm_pre, *xmm_post) << "xmm" << i << " has been modified!"; |
| ++xmm_pre; |
| ++xmm_post; |
| } |
| return !testing::Test::HasNonfatalFailure(); |
| } |
| |
| bool initialized_; |
| CONTEXT pre_context_; |
| }; |
| |
| #define ASM_REGISTER_STATE_CHECK(statement) \ |
| do { \ |
| libaom_test::RegisterStateCheck reg_check; \ |
| statement; \ |
| } while (false) |
| |
| } // namespace libaom_test |
| |
| #elif defined(CONFIG_SHARED) && defined(HAVE_NEON_ASM) && !CONFIG_SHARED && \ |
| HAVE_NEON_ASM && CONFIG_AV1 |
| |
| extern "C" { |
| // Save the d8-d15 registers into store. |
| void aom_push_neon(int64_t *store); |
| } |
| |
| namespace libaom_test { |
| |
| // Compares the state of d8-d15 at construction with their state at |
| // destruction. These registers should be preserved by the callee on |
| // arm platform. |
| class RegisterStateCheck { |
| public: |
| RegisterStateCheck() { initialized_ = StoreRegisters(pre_store_); } |
| ~RegisterStateCheck() { EXPECT_TRUE(Check()); } |
| |
| private: |
| static bool StoreRegisters(int64_t store[8]) { |
| aom_push_neon(store); |
| return true; |
| } |
| |
| // Compares the register state. Returns true if the states match. |
| bool Check() const { |
| if (!initialized_) return false; |
| int64_t post_store[8]; |
| aom_push_neon(post_store); |
| for (int i = 0; i < 8; ++i) { |
| EXPECT_EQ(pre_store_[i], post_store[i]) << "d" << i + 8 |
| << " has been modified"; |
| } |
| return !testing::Test::HasNonfatalFailure(); |
| } |
| |
| bool initialized_; |
| int64_t pre_store_[8]; |
| }; |
| |
| #define ASM_REGISTER_STATE_CHECK(statement) \ |
| do { \ |
| libaom_test::RegisterStateCheck reg_check; \ |
| statement; \ |
| } while (false) |
| |
| } // namespace libaom_test |
| |
| #else |
| |
| namespace libaom_test { |
| |
| class RegisterStateCheck {}; |
| #define ASM_REGISTER_STATE_CHECK(statement) statement |
| |
| } // namespace libaom_test |
| |
| #endif // _WIN64 |
| |
| #if ARCH_X86 || ARCH_X86_64 |
| #if defined(__GNUC__) |
| |
| namespace libaom_test { |
| |
| // Checks the FPU tag word pre/post execution to ensure emms has been called. |
| class RegisterStateCheckMMX { |
| public: |
| RegisterStateCheckMMX() { |
| __asm__ volatile("fstenv %0" : "=rm"(pre_fpu_env_)); |
| } |
| ~RegisterStateCheckMMX() { EXPECT_TRUE(Check()); } |
| |
| private: |
| // Checks the FPU tag word pre/post execution, returning false if not cleared |
| // to 0xffff. |
| bool Check() const { |
| EXPECT_EQ(0xffff, pre_fpu_env_[4]) |
| << "FPU was in an inconsistent state prior to call"; |
| |
| uint16_t post_fpu_env[14]; |
| __asm__ volatile("fstenv %0" : "=rm"(post_fpu_env)); |
| EXPECT_EQ(0xffff, post_fpu_env[4]) |
| << "FPU was left in an inconsistent state after call"; |
| return !testing::Test::HasNonfatalFailure(); |
| } |
| |
| uint16_t pre_fpu_env_[14]; |
| }; |
| |
| #define API_REGISTER_STATE_CHECK(statement) \ |
| do { \ |
| libaom_test::RegisterStateCheckMMX reg_check; \ |
| ASM_REGISTER_STATE_CHECK(statement); \ |
| } while (false) |
| |
| } // namespace libaom_test |
| |
| #endif // __GNUC__ |
| #endif // ARCH_X86 || ARCH_X86_64 |
| |
| #ifndef API_REGISTER_STATE_CHECK |
| #define API_REGISTER_STATE_CHECK ASM_REGISTER_STATE_CHECK |
| #endif |
| |
| #endif // TEST_REGISTER_STATE_CHECK_H_ |