| // Copyright 2018 The BoringSSL Authors | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //     https://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 |  | 
 | #ifndef OPENSSL_HEADER_CRYPTO_TEST_ABI_TEST_H | 
 | #define OPENSSL_HEADER_CRYPTO_TEST_ABI_TEST_H | 
 |  | 
 | #include <gtest/gtest.h> | 
 |  | 
 | #include <string> | 
 | #include <type_traits> | 
 | #include <vector> | 
 |  | 
 | #include <openssl/base.h> | 
 |  | 
 | #include "../internal.h" | 
 |  | 
 |  | 
 | // abi_test provides routines for verifying that functions satisfy platform ABI | 
 | // requirements. | 
 | namespace abi_test { | 
 |  | 
 | // Result stores the result of an ABI test. | 
 | struct Result { | 
 |   bool ok() const { return errors.empty(); } | 
 |  | 
 |   std::vector<std::string> errors; | 
 | }; | 
 |  | 
 | namespace internal { | 
 |  | 
 | // DeductionGuard wraps |T| in a template, so that template argument deduction | 
 | // does not apply to it. This may be used to force C++ to deduce template | 
 | // arguments from another parameter. | 
 | template <typename T> | 
 | struct DeductionGuard { | 
 |   using Type = T; | 
 | }; | 
 |  | 
 | // Reg128 contains storage space for a 128-bit register. | 
 | struct alignas(16) Reg128 { | 
 |   bool operator==(const Reg128 &x) const { return x.lo == lo && x.hi == hi; } | 
 |   bool operator!=(const Reg128 &x) const { return !((*this) == x); } | 
 |   uint64_t lo, hi; | 
 | }; | 
 |  | 
 | // LOOP_CALLER_STATE_REGISTERS is a macro that iterates over all registers the | 
 | // callee is expected to save for the caller, with the exception of the stack | 
 | // pointer. The stack pointer is tested implicitly by the function successfully | 
 | // returning at all. | 
 | #if defined(OPENSSL_X86_64) | 
 |  | 
 | // References: | 
 | // SysV64: https://github.com/hjl-tools/x86-psABI/wiki/x86-64-psABI-1.0.pdf | 
 | // Win64: https://docs.microsoft.com/en-us/cpp/build/x64-software-conventions?view=vs-2017#register-usage | 
 | #if defined(OPENSSL_WINDOWS) | 
 | #define LOOP_CALLER_STATE_REGISTERS()  \ | 
 |   CALLER_STATE_REGISTER(uint64_t, rbx) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, rbp) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, rdi) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, rsi) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r12) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r13) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r14) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r15) \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm6)  \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm7)  \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm8)  \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm9)  \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm10) \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm11) \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm12) \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm13) \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm14) \ | 
 |   CALLER_STATE_REGISTER(Reg128, xmm15) | 
 | #else | 
 | #define LOOP_CALLER_STATE_REGISTERS()  \ | 
 |   CALLER_STATE_REGISTER(uint64_t, rbx) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, rbp) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r12) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r13) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r14) \ | 
 |   CALLER_STATE_REGISTER(uint64_t, r15) | 
 | #endif  // OPENSSL_WINDOWS | 
 |  | 
 | #elif defined(OPENSSL_X86) | 
 |  | 
 | // References: | 
 | // SysV32: https://uclibc.org/docs/psABI-i386.pdf and | 
 | // Win32: https://docs.microsoft.com/en-us/cpp/cpp/argument-passing-and-naming-conventions?view=vs-2017 | 
 | #define LOOP_CALLER_STATE_REGISTERS()  \ | 
 |   CALLER_STATE_REGISTER(uint32_t, esi) \ | 
 |   CALLER_STATE_REGISTER(uint32_t, edi) \ | 
 |   CALLER_STATE_REGISTER(uint32_t, ebx) \ | 
 |   CALLER_STATE_REGISTER(uint32_t, ebp) | 
 |  | 
 | #elif defined(OPENSSL_ARM) | 
 |  | 
 | // References: | 
 | // AAPCS: https://developer.arm.com/docs/ihi0042/latest | 
 | // iOS32: https://developer.apple.com/library/archive/documentation/Xcode/Conceptual/iPhoneOSABIReference/Articles/ARMv6FunctionCallingConventions.html | 
 | // Linux: http://sourcery.mentor.com/sgpp/lite/arm/portal/kbattach142/arm_gnu_linux_%20abi.pdf | 
 | // | 
 | // ARM specifies a common calling convention, except r9 is left to the platform. | 
 | // Linux treats r9 as callee-saved, while iOS 3+ treats it as caller-saved. Most | 
 | // of our assembly treats it as callee-saved to be uniform, but we match the | 
 | // platform to avoid false positives when testing compiler-generated output. | 
 | #define LOOP_CALLER_STATE_REGISTERS_PRE_R9() \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d8)        \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d9)        \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d10)       \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d11)       \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d12)       \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d13)       \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d14)       \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d15)       \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r4)        \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r5)        \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r6)        \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r7)        \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r8) | 
 | #define LOOP_CALLER_STATE_REGISTERS_POST_R9() \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r10)        \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r11) | 
 | #if defined(OPENSSL_APPLE) | 
 | #define LOOP_CALLER_STATE_REGISTERS()  \ | 
 |   LOOP_CALLER_STATE_REGISTERS_PRE_R9() \ | 
 |   LOOP_CALLER_STATE_REGISTERS_POST_R9() | 
 | #else  // !OPENSSL_APPLE | 
 | #define LOOP_CALLER_STATE_REGISTERS()  \ | 
 |   LOOP_CALLER_STATE_REGISTERS_PRE_R9() \ | 
 |   CALLER_STATE_REGISTER(uint32_t, r9)  \ | 
 |   LOOP_CALLER_STATE_REGISTERS_POST_R9() | 
 | #endif  // OPENSSL_APPLE | 
 |  | 
 | #elif defined(OPENSSL_AARCH64) | 
 |  | 
 | // References: | 
 | // AAPCS64: https://developer.arm.com/docs/ihi0055/latest | 
 | // iOS64: https://developer.apple.com/library/archive/documentation/Xcode/Conceptual/iPhoneOSABIReference/Articles/ARM64FunctionCallingConventions.html | 
 | // | 
 | // In aarch64, r18 (accessed as w18 or x18 in a 64-bit context) is the platform | 
 | // register. iOS says user code may not touch it. We found no clear reference | 
 | // for Linux. The iOS behavior implies portable assembly cannot use it, and | 
 | // aarch64 has many registers. Thus this framework ignores register's existence. | 
 | // We test r18 violations in arm-xlate.pl. | 
 | #define LOOP_CALLER_STATE_REGISTERS()                                \ | 
 |   /* Per AAPCS64, section 5.1.2, only the bottom 64 bits of v8-v15 */ \ | 
 |   /* are preserved. These are accessed as dN. */                     \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d8)                                \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d9)                                \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d10)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d11)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d12)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d13)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d14)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, d15)                               \ | 
 |   /* For consistency with dN, use the 64-bit name xN, rather than */ \ | 
 |   /* the generic rN. */                                              \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x19)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x20)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x21)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x22)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x23)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x24)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x25)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x26)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x27)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x28)                               \ | 
 |   CALLER_STATE_REGISTER(uint64_t, x29) | 
 |  | 
 | #endif  // X86_64 || X86 || ARM || AARCH64 | 
 |  | 
 | // Enable ABI testing if all of the following are true. | 
 | // | 
 | // - We have CallerState and trampoline support for the architecture. | 
 | // | 
 | // - Assembly is enabled. | 
 | // | 
 | // - This is not a shared library build. Assembly functions are not reachable | 
 | //   from tests in shared library builds. | 
 | #if defined(LOOP_CALLER_STATE_REGISTERS) && !defined(OPENSSL_NO_ASM) && \ | 
 |     !defined(BORINGSSL_SHARED_LIBRARY) | 
 | #define SUPPORTS_ABI_TEST | 
 |  | 
 | // CallerState contains all caller state that the callee is expected to | 
 | // preserve. | 
 | struct CallerState { | 
 | #define CALLER_STATE_REGISTER(type, name) type name; | 
 |   LOOP_CALLER_STATE_REGISTERS() | 
 | #undef CALLER_STATE_REGISTER | 
 | }; | 
 |  | 
 | // RunTrampoline runs |func| on |argv|, recording ABI errors in |out|. It does | 
 | // not perform any type-checking. If |unwind| is true and unwind tests have been | 
 | // enabled, |func| is single-stepped under an unwind test. | 
 | crypto_word_t RunTrampoline(Result *out, crypto_word_t func, | 
 |                             const crypto_word_t *argv, size_t argc, | 
 |                             bool unwind); | 
 |  | 
 | template <typename T> | 
 | inline crypto_word_t ToWord(T t) { | 
 |   // ABIs typically pass floats and structs differently from integers and | 
 |   // pointers. We only need to support the latter. | 
 |   static_assert(std::is_integral<T>::value || std::is_pointer<T>::value, | 
 |                 "parameter types must be integral or pointer types"); | 
 |   // We only support types which fit in registers. | 
 |   static_assert(sizeof(T) <= sizeof(crypto_word_t), | 
 |                 "parameter types must be at most word-sized"); | 
 |  | 
 |   // ABIs are complex around arguments that are smaller than native words. | 
 |   // Parameters passed in memory are sometimes packed and sometimes padded to a | 
 |   // word. When parameters are padded in memory or passed in a larger register, | 
 |   // the unused bits may be undefined or sign- or zero-extended. | 
 |   // | 
 |   // We could simply cast to |crypto_word_t| everywhere but, on platforms where | 
 |   // padding is undefined, we perturb the bits to test the function accounts for | 
 |   // for this. | 
 | #if defined(OPENSSL_32_BIT) | 
 |   // We never pass parameters smaller than int, so require word-sized parameters | 
 |   // on 32-bit architectures for simplicity. | 
 |   static_assert(sizeof(T) == 4, "parameter types must be word-sized"); | 
 |   return (crypto_word_t)t; | 
 | #elif defined(OPENSSL_X86_64) || defined(OPENSSL_AARCH64) | 
 |   // AAPCS64, section 5.4.2, clauses C.7 and C.14 says any remaining bits in | 
 |   // aarch are unspecified. iOS64 contradicts this and says the callee extends | 
 |   // arguments up to 32 bits, and only the upper 32 bits are unspecified. | 
 |   // | 
 |   // On x86_64, Win64 leaves all unused bits unspecified. SysV also leaves | 
 |   // unused bits in stack parameters unspecified, but it behaves like iOS64 for | 
 |   // register parameters. This was determined via experimentation. | 
 |   // | 
 |   // We limit to 32-bit and 64-bit parameters, the subset where the above all | 
 |   // align, and then test that functions tolerate arbitrary unused bits. | 
 |   // | 
 |   // TODO(davidben): Find authoritative citations for x86_64. For x86_64, I | 
 |   // observed the behavior of Clang, GCC, and MSVC. ABI rules here may be | 
 |   // inferred from two kinds of experiments: | 
 |   // | 
 |   // 1. When passing a value to a small-argument-taking function, does the | 
 |   //    compiler ensure unused bits are cleared, sign-extended, etc.? Tests for | 
 |   //    register parameters are confounded by x86_64's implicit clearing of | 
 |   //    registers' upper halves, but passing some_u64 >> 1 usually clears this. | 
 |   // | 
 |   // 2. When compiling a small-argument-taking function, does the compiler make | 
 |   //    assumptions about unused bits of arguments? | 
 |   // | 
 |   // MSVC was observed to tolerate and produce arbitrary values for unused bits, | 
 |   // which is conclusive. GCC and Clang, targeting Linux, were similarly | 
 |   // conclusive on stack parameters. Clang was also conclusive for register | 
 |   // parameters. Callers only extended parameters up to 32 bits, and callees | 
 |   // took advantage of the 32-bit extension. GCC only exhibited the callee | 
 |   // behavior. | 
 |   static_assert(sizeof(T) >= 4, "parameters must be at least 32 bits wide"); | 
 |   crypto_word_t ret; | 
 |   // Filling extra bits with 0xaa will be vastly out of bounds for code | 
 |   // expecting either sign- or zero-extension. (0xaa is 0b10101010.) | 
 |   OPENSSL_memset(&ret, 0xaa, sizeof(ret)); | 
 |   OPENSSL_memcpy(&ret, &t, sizeof(t)); | 
 |   return ret; | 
 | #else | 
 | #error "unknown architecture" | 
 | #endif | 
 | } | 
 |  | 
 | // CheckImpl runs |func| on |args|, recording ABI errors in |out|. If |unwind| | 
 | // is true and unwind tests have been enabled, |func| is single-stepped under an | 
 | // unwind test. | 
 | // | 
 | // It returns the value as a |crypto_word_t| to work around problems when |R| is | 
 | // void. |args| is wrapped in a |DeductionGuard| so |func| determines the | 
 | // template arguments. Otherwise, |args| may deduce |Args| incorrectly. For | 
 | // instance, if |func| takes const int *, and the caller passes an int *, the | 
 | // compiler will complain the deduced types do not match. | 
 | template <typename R, typename... Args> | 
 | inline crypto_word_t CheckImpl(Result *out, bool unwind, R (*func)(Args...), | 
 |                                typename DeductionGuard<Args>::Type... args) { | 
 |   // We only support up to 8 arguments, so all arguments on aarch64 are passed | 
 |   // in registers. This is simpler and avoids the iOS discrepancy around packing | 
 |   // small arguments on the stack. (See the iOS64 reference.) | 
 |   static_assert(sizeof...(args) <= 8, | 
 |                 "too many arguments for abi_test_trampoline"); | 
 |  | 
 |   // Allocate one extra entry so MSVC does not complain about zero-size arrays. | 
 |   crypto_word_t argv[sizeof...(args) + 1] = { | 
 |       ToWord(args)..., | 
 |   }; | 
 |   return RunTrampoline(out, reinterpret_cast<crypto_word_t>(func), argv, | 
 |                        sizeof...(args), unwind); | 
 | } | 
 | #else | 
 | // To simplify callers when ABI testing support is unavoidable, provide a backup | 
 | // CheckImpl implementation. It must be specialized for void returns because we | 
 | // call |func| directly. | 
 | template <typename R, typename... Args> | 
 | inline std::enable_if_t<!std::is_void<R>::value, crypto_word_t> CheckImpl( | 
 |     Result *out, bool /* unwind */, R (*func)(Args...), | 
 |     typename DeductionGuard<Args>::Type... args) { | 
 |   *out = Result(); | 
 |   return func(args...); | 
 | } | 
 |  | 
 | template <typename... Args> | 
 | inline crypto_word_t CheckImpl(Result *out, bool /* unwind */, | 
 |                                void (*func)(Args...), | 
 |                                typename DeductionGuard<Args>::Type... args) { | 
 |   *out = Result(); | 
 |   func(args...); | 
 |   return 0; | 
 | } | 
 | #endif  // SUPPORTS_ABI_TEST | 
 |  | 
 | // FixVAArgsString takes a string like "f, 1, 2" and returns a string like | 
 | // "f(1, 2)". | 
 | // | 
 | // This is needed because the |CHECK_ABI| macro below cannot be defined as | 
 | // CHECK_ABI(func, ...). The C specification requires that variadic macros bind | 
 | // at least one variadic argument. Clang, GCC, and MSVC all ignore this, but | 
 | // there are issues with trailing commas and different behaviors across | 
 | // compilers. | 
 | std::string FixVAArgsString(const char *str); | 
 |  | 
 | // CheckGTest behaves like |CheckImpl|, but it returns the correct type and | 
 | // raises GTest assertions on failure. If |unwind| is true and unwind tests are | 
 | // enabled, |func| is single-stepped under an unwind test. | 
 | template <typename R, typename... Args> | 
 | inline R CheckGTest(const char *va_args_str, const char *file, int line, | 
 |                     bool unwind, R (*func)(Args...), | 
 |                     typename DeductionGuard<Args>::Type... args) { | 
 |   Result result; | 
 |   crypto_word_t ret = CheckImpl(&result, unwind, func, args...); | 
 |   if (!result.ok()) { | 
 |     testing::Message msg; | 
 |     msg << "ABI failures in " << FixVAArgsString(va_args_str) << ":\n"; | 
 |     for (const auto &error : result.errors) { | 
 |       msg << "    " << error << "\n"; | 
 |     } | 
 |     ADD_FAILURE_AT(file, line) << msg; | 
 |   } | 
 |   return (R)ret; | 
 | } | 
 |  | 
 | }  // namespace internal | 
 |  | 
 | // Check runs |func| on |args| and returns the result. If ABI-testing is | 
 | // supported in this build configuration, it writes any ABI failures to |out|. | 
 | // Otherwise, it runs the function transparently. | 
 | template <typename R, typename... Args> | 
 | inline R Check(Result *out, R (*func)(Args...), | 
 |                typename internal::DeductionGuard<Args>::Type... args) { | 
 |   return (R)internal::CheckImpl(out, false, func, args...); | 
 | } | 
 |  | 
 | // EnableUnwindTests enables unwind tests, if supported. If not supported, it | 
 | // does nothing. | 
 | void EnableUnwindTests(); | 
 |  | 
 | // UnwindTestsEnabled returns true if unwind tests are enabled and false | 
 | // otherwise. | 
 | bool UnwindTestsEnabled(); | 
 |  | 
 | }  // namespace abi_test | 
 |  | 
 | // CHECK_ABI calls the first argument on the remaining arguments and returns the | 
 | // result. If ABI-testing is supported in this build configuration, it adds a | 
 | // non-fatal GTest failure if the call did not satisfy ABI requirements. | 
 | // | 
 | // |CHECK_ABI| does return the value and thus may replace any function call, | 
 | // provided it takes only simple parameters. However, it is recommended to test | 
 | // ABI separately from functional tests of assembly. Fully instrumenting a | 
 | // function for ABI checking requires single-stepping the function, which is | 
 | // inefficient. | 
 | // | 
 | // Functional testing requires coverage of input values, while ABI testing only | 
 | // requires branch coverage. Most of our assembly is constant-time, so usually | 
 | // only a few instrumented calls are necessary. | 
 | // | 
 | // TODO(https://crbug.com/boringssl/259): Most of Windows assembly currently | 
 | // fails SEH testing. For now, |CHECK_ABI| behaves like |CHECK_ABI_NO_UNWIND| | 
 | // on Windows. Functions which work with unwind testing on Windows should use | 
 | // |CHECK_ABI_SEH|. | 
 | #if defined(OPENSSL_WINDOWS) | 
 | #define CHECK_ABI(...) CHECK_ABI_NO_UNWIND(__VA_ARGS__) | 
 | #else | 
 | #define CHECK_ABI(...) CHECK_ABI_SEH(__VA_ARGS__) | 
 | #endif | 
 |  | 
 | // CHECK_ABI_SEH behaves like |CHECK_ABI| but enables unwind testing on Windows. | 
 | #define CHECK_ABI_SEH(...)                                               \ | 
 |   abi_test::internal::CheckGTest(#__VA_ARGS__, __FILE__, __LINE__, true, \ | 
 |                                  __VA_ARGS__) | 
 |  | 
 | // CHECK_ABI_NO_UNWIND behaves like |CHECK_ABI| but disables unwind testing. | 
 | #define CHECK_ABI_NO_UNWIND(...)                                          \ | 
 |   abi_test::internal::CheckGTest(#__VA_ARGS__, __FILE__, __LINE__, false, \ | 
 |                                  __VA_ARGS__) | 
 |  | 
 |  | 
 | // Internal functions. | 
 |  | 
 | #if defined(SUPPORTS_ABI_TEST) | 
 | struct Uncallable { | 
 |   Uncallable() = delete; | 
 | }; | 
 |  | 
 | extern "C" { | 
 |  | 
 | // abi_test_trampoline loads callee-saved registers from |state|, calls |func| | 
 | // with |argv|, then saves the callee-saved registers into |state|. It returns | 
 | // the result of |func|. If |unwind| is non-zero, this function triggers unwind | 
 | // instrumentation. | 
 | // | 
 | // We give |func| type |crypto_word_t| to avoid tripping MSVC's warning 4191. | 
 | crypto_word_t abi_test_trampoline(crypto_word_t func, | 
 |                                   abi_test::internal::CallerState *state, | 
 |                                   const crypto_word_t *argv, size_t argc, | 
 |                                   crypto_word_t unwind); | 
 |  | 
 | #if defined(OPENSSL_X86_64) | 
 | // abi_test_unwind_start points at the instruction that starts unwind testing in | 
 | // |abi_test_trampoline|. This is the value of the instruction pointer at the | 
 | // first |SIGTRAP| during unwind testing. | 
 | // | 
 | // This symbol is not a function and should not be called. | 
 | void abi_test_unwind_start(Uncallable); | 
 |  | 
 | // abi_test_unwind_return points at the instruction immediately after the call in | 
 | // |abi_test_trampoline|. When unwinding the function under test, this is the | 
 | // expected address in the |abi_test_trampoline| frame. After this address, the | 
 | // unwind tester should ignore |SIGTRAP| until |abi_test_unwind_stop|. | 
 | // | 
 | // This symbol is not a function and should not be called. | 
 | void abi_test_unwind_return(Uncallable); | 
 |  | 
 | // abi_test_unwind_stop is the value of the instruction pointer at the final | 
 | // |SIGTRAP| during unwind testing. | 
 | // | 
 | // This symbol is not a function and should not be called. | 
 | void abi_test_unwind_stop(Uncallable); | 
 |  | 
 | // abi_test_bad_unwind_wrong_register preserves the ABI, but annotates the wrong | 
 | // register in unwind metadata. | 
 | void abi_test_bad_unwind_wrong_register(void); | 
 |  | 
 | // abi_test_bad_unwind_temporary preserves the ABI, but temporarily corrupts the | 
 | // storage space for a saved register, breaking unwind. | 
 | void abi_test_bad_unwind_temporary(void); | 
 |  | 
 | #if defined(OPENSSL_WINDOWS) | 
 | // abi_test_bad_unwind_epilog preserves the ABI, and correctly annotates the | 
 | // prolog, but the epilog does not match Win64's rules, breaking unwind during | 
 | // the epilog. | 
 | void abi_test_bad_unwind_epilog(void); | 
 | #endif | 
 | #endif  // OPENSSL_X86_64 | 
 |  | 
 | #if defined(OPENSSL_X86_64) || defined(OPENSSL_X86) | 
 | // abi_test_get_and_clear_direction_flag clears the direction flag. If the flag | 
 | // was previously set, it returns one. Otherwise, it returns zero. | 
 | int abi_test_get_and_clear_direction_flag(void); | 
 |  | 
 | // abi_test_set_direction_flag sets the direction flag. This does not conform to | 
 | // ABI requirements and must only be called within a |CHECK_ABI| guard to avoid | 
 | // errors later in the program. | 
 | int abi_test_set_direction_flag(void); | 
 | #endif  // OPENSSL_X86_64 || OPENSSL_X86 | 
 |  | 
 | }  // extern "C" | 
 | #endif  // SUPPORTS_ABI_TEST | 
 |  | 
 |  | 
 | #endif  // OPENSSL_HEADER_CRYPTO_TEST_ABI_TEST_H |