// -*- C++ -*- //===---------------------------- test_macros.h ---------------------------===// // // 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 // //===----------------------------------------------------------------------===// #ifndef SUPPORT_TEST_MACROS_HPP #define SUPPORT_TEST_MACROS_HPP // Attempt to get STL specific macros like _LIBCPP_VERSION using the most // minimal header possible. If we're testing libc++, we should use `<__config>`. // If <__config> isn't available, fall back to . #ifdef __has_include # if __has_include("<__config>") # include <__config> # define TEST_IMP_INCLUDED_HEADER # endif #endif #ifndef TEST_IMP_INCLUDED_HEADER #include #endif #if defined(__GNUC__) #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wvariadic-macros" #endif #define TEST_STRINGIZE_IMPL(x) #x #define TEST_STRINGIZE(x) TEST_STRINGIZE_IMPL(x) #define TEST_CONCAT1(X, Y) X##Y #define TEST_CONCAT(X, Y) TEST_CONCAT1(X, Y) #ifdef __has_feature #define TEST_HAS_FEATURE(X) __has_feature(X) #else #define TEST_HAS_FEATURE(X) 0 #endif #ifndef __has_include #define __has_include(...) 0 #endif #ifdef __has_extension #define TEST_HAS_EXTENSION(X) __has_extension(X) #else #define TEST_HAS_EXTENSION(X) 0 #endif #ifdef __has_warning #define TEST_HAS_WARNING(X) __has_warning(X) #else #define TEST_HAS_WARNING(X) 0 #endif #ifdef __has_builtin #define TEST_HAS_BUILTIN(X) __has_builtin(X) #else #define TEST_HAS_BUILTIN(X) 0 #endif #ifdef __is_identifier // '__is_identifier' returns '0' if '__x' is a reserved identifier provided by // the compiler and '1' otherwise. #define TEST_HAS_BUILTIN_IDENTIFIER(X) !__is_identifier(X) #else #define TEST_HAS_BUILTIN_IDENTIFIER(X) 0 #endif #if defined(__EDG__) # define TEST_COMPILER_EDG #elif defined(__clang__) # define TEST_COMPILER_CLANG # if defined(__apple_build_version__) # define TEST_COMPILER_APPLE_CLANG # endif #elif defined(_MSC_VER) # define TEST_COMPILER_C1XX #elif defined(__GNUC__) # define TEST_COMPILER_GCC #endif #if defined(__apple_build_version__) #define TEST_APPLE_CLANG_VER (__clang_major__ * 100) + __clang_minor__ #elif defined(__clang_major__) #define TEST_CLANG_VER (__clang_major__ * 100) + __clang_minor__ #elif defined(__GNUC__) #define TEST_GCC_VER (__GNUC__ * 100 + __GNUC_MINOR__) #define TEST_GCC_VER_NEW (TEST_GCC_VER * 10 + __GNUC_PATCHLEVEL__) #endif /* Make a nice name for the standard version */ #ifndef TEST_STD_VER #if __cplusplus <= 199711L # define TEST_STD_VER 3 #elif __cplusplus <= 201103L # define TEST_STD_VER 11 #elif __cplusplus <= 201402L # define TEST_STD_VER 14 #elif __cplusplus <= 201703L # define TEST_STD_VER 17 #else # define TEST_STD_VER 99 // greater than current standard // This is deliberately different than _LIBCPP_STD_VER to discourage matching them up. #endif #endif // Attempt to deduce the GLIBC version #if (defined(__has_include) && __has_include()) || \ defined(__linux__) #include #if defined(__GLIBC_PREREQ) #define TEST_HAS_GLIBC #define TEST_GLIBC_PREREQ(major, minor) __GLIBC_PREREQ(major, minor) #endif #endif #if TEST_STD_VER >= 11 # define TEST_ALIGNOF(...) alignof(__VA_ARGS__) # define TEST_ALIGNAS(...) alignas(__VA_ARGS__) # define TEST_CONSTEXPR constexpr # define TEST_NOEXCEPT noexcept # define TEST_NOEXCEPT_FALSE noexcept(false) # define TEST_NOEXCEPT_COND(...) noexcept(__VA_ARGS__) #else # if defined(TEST_COMPILER_CLANG) # define TEST_ALIGNOF(...) _Alignof(__VA_ARGS__) # else # define TEST_ALIGNOF(...) __alignof(__VA_ARGS__) # endif # define TEST_ALIGNAS(...) __attribute__((__aligned__(__VA_ARGS__))) # define TEST_CONSTEXPR # define TEST_NOEXCEPT throw() # define TEST_NOEXCEPT_FALSE # define TEST_NOEXCEPT_COND(...) #endif #if TEST_STD_VER >= 17 # define TEST_THROW_SPEC(...) #else # define TEST_THROW_SPEC(...) throw(__VA_ARGS__) #endif #if TEST_STD_VER >= 14 # define TEST_CONSTEXPR_CXX14 constexpr #else # define TEST_CONSTEXPR_CXX14 #endif #if TEST_STD_VER >= 17 # define TEST_CONSTEXPR_CXX17 constexpr #else # define TEST_CONSTEXPR_CXX17 #endif #if TEST_STD_VER >= 20 # define TEST_CONSTEXPR_CXX20 constexpr #else # define TEST_CONSTEXPR_CXX20 #endif // Sniff out to see if the underlying C library has C11 features // This is cribbed from __config; but lives here as well because we can't assume libc++ #if __ISO_C_VISIBLE >= 2011 || TEST_STD_VER >= 11 # if defined(__FreeBSD__) // Specifically, FreeBSD does NOT have timespec_get, even though they have all // the rest of C11 - this is PR#38495 # define TEST_HAS_ALIGNED_ALLOC # define TEST_HAS_QUICK_EXIT # elif defined(__BIONIC__) # if __ANDROID_API__ >= 21 # define TEST_HAS_QUICK_EXIT # endif # if __ANDROID_API__ >= 28 # define TEST_HAS_ALIGNED_ALLOC # endif # if __ANDROID_API__ >= 29 # define TEST_HAS_TIMESPEC_GET # endif # elif defined(__Fuchsia__) || defined(__wasi__) || defined(__NetBSD__) # define TEST_HAS_QUICK_EXIT # define TEST_HAS_ALIGNED_ALLOC # define TEST_HAS_TIMESPEC_GET # elif defined(__linux__) // This block preserves the old behavior used by include/__config: // _LIBCPP_GLIBC_PREREQ would be defined to 0 if __GLIBC_PREREQ was not // available. The configuration here may be too vague though, as Bionic, uClibc, // newlib, etc may all support these features but need to be configured. # if defined(TEST_GLIBC_PREREQ) # if TEST_GLIBC_PREREQ(2, 15) # define TEST_HAS_QUICK_EXIT # endif # if TEST_GLIBC_PREREQ(2, 17) # define TEST_HAS_ALIGNED_ALLOC # define TEST_HAS_TIMESPEC_GET # endif # elif defined(_LIBCPP_HAS_MUSL_LIBC) # define TEST_HAS_QUICK_EXIT # define TEST_HAS_ALIGNED_ALLOC # define TEST_HAS_TIMESPEC_GET # endif # elif defined(_WIN32) # if defined(_MSC_VER) && !defined(__MINGW32__) # define TEST_HAS_QUICK_EXIT # define TEST_HAS_ALIGNED_ALLOC # define TEST_HAS_TIMESPEC_GET # endif # elif defined(__APPLE__) // timespec_get and aligned_alloc were introduced in macOS 10.15 and // aligned releases # if (__ENVIRONMENT_MAC_OS_X_VERSION_MIN_REQUIRED__ >= 101500 || \ __ENVIRONMENT_IPHONE_OS_VERSION_MIN_REQUIRED__ >= 130000 || \ __ENVIRONMENT_TV_OS_VERSION_MIN_REQUIRED__ >= 130000 || \ __ENVIRONMENT_WATCH_OS_VERSION_MIN_REQUIRED__ >= 60000) # define TEST_HAS_ALIGNED_ALLOC # define TEST_HAS_TIMESPEC_GET # endif # endif // __APPLE__ #endif /* Features that were introduced in C++14 */ #if TEST_STD_VER >= 14 #define TEST_HAS_EXTENDED_CONSTEXPR #define TEST_HAS_VARIABLE_TEMPLATES #endif /* Features that were introduced in C++17 */ #if TEST_STD_VER >= 17 #endif /* Features that were introduced after C++17 */ #if TEST_STD_VER > 17 #endif #define TEST_ALIGNAS_TYPE(...) TEST_ALIGNAS(TEST_ALIGNOF(__VA_ARGS__)) #if !TEST_HAS_FEATURE(cxx_rtti) && !defined(__cpp_rtti) \ && !defined(__GXX_RTTI) #define TEST_HAS_NO_RTTI #endif #if !defined(TEST_HAS_NO_RTTI) # define RTTI_ASSERT(X) assert(X) #else # define RTTI_ASSERT(X) #endif #if !TEST_HAS_FEATURE(cxx_exceptions) && !defined(__cpp_exceptions) \ && !defined(__EXCEPTIONS) #define TEST_HAS_NO_EXCEPTIONS #endif #if TEST_HAS_FEATURE(address_sanitizer) || TEST_HAS_FEATURE(memory_sanitizer) || \ TEST_HAS_FEATURE(thread_sanitizer) #define TEST_HAS_SANITIZERS #endif #if defined(_LIBCPP_NORETURN) #define TEST_NORETURN _LIBCPP_NORETURN #else #define TEST_NORETURN [[noreturn]] #endif #if defined(_LIBCPP_HAS_NO_ALIGNED_ALLOCATION) || \ (!(TEST_STD_VER > 14 || \ (defined(__cpp_aligned_new) && __cpp_aligned_new >= 201606L))) #define TEST_HAS_NO_ALIGNED_ALLOCATION #endif #if defined(_LIBCPP_SAFE_STATIC) #define TEST_SAFE_STATIC _LIBCPP_SAFE_STATIC #else #define TEST_SAFE_STATIC #endif #if !defined(__cpp_impl_three_way_comparison) \ && (!defined(_MSC_VER) || defined(__clang__) || _MSC_VER < 1920 || _MSVC_LANG <= 201703L) #define TEST_HAS_NO_SPACESHIP_OPERATOR #endif #if TEST_STD_VER < 11 #define ASSERT_NOEXCEPT(...) #define ASSERT_NOT_NOEXCEPT(...) #else #define ASSERT_NOEXCEPT(...) \ static_assert(noexcept(__VA_ARGS__), "Operation must be noexcept") #define ASSERT_NOT_NOEXCEPT(...) \ static_assert(!noexcept(__VA_ARGS__), "Operation must NOT be noexcept") #endif /* Macros for testing libc++ specific behavior and extensions */ #if defined(_LIBCPP_VERSION) #define LIBCPP_ASSERT(...) assert(__VA_ARGS__) #define LIBCPP_STATIC_ASSERT(...) static_assert(__VA_ARGS__) #define LIBCPP_ASSERT_NOEXCEPT(...) ASSERT_NOEXCEPT(__VA_ARGS__) #define LIBCPP_ASSERT_NOT_NOEXCEPT(...) ASSERT_NOT_NOEXCEPT(__VA_ARGS__) #define LIBCPP_ONLY(...) __VA_ARGS__ #else #define LIBCPP_ASSERT(...) ((void)0) #define LIBCPP_STATIC_ASSERT(...) ((void)0) #define LIBCPP_ASSERT_NOEXCEPT(...) ((void)0) #define LIBCPP_ASSERT_NOT_NOEXCEPT(...) ((void)0) #define LIBCPP_ONLY(...) ((void)0) #endif #define TEST_IGNORE_NODISCARD (void) namespace test_macros_detail { template struct is_same { enum { value = 0};} ; template struct is_same { enum {value = 1}; }; } // namespace test_macros_detail #define ASSERT_SAME_TYPE(...) \ static_assert((test_macros_detail::is_same<__VA_ARGS__>::value), \ "Types differ unexpectedly") #ifndef TEST_HAS_NO_EXCEPTIONS #define TEST_THROW(...) throw __VA_ARGS__ #else #if defined(__GNUC__) #define TEST_THROW(...) __builtin_abort() #else #include #define TEST_THROW(...) ::abort() #endif #endif #if defined(__GNUC__) || defined(__clang__) template inline void DoNotOptimize(Tp const& value) { asm volatile("" : : "r,m"(value) : "memory"); } template inline void DoNotOptimize(Tp& value) { #if defined(__clang__) asm volatile("" : "+r,m"(value) : : "memory"); #else asm volatile("" : "+m,r"(value) : : "memory"); #endif } #else #include template inline void DoNotOptimize(Tp const& value) { const volatile void* volatile unused = __builtin_addressof(value); static_cast(unused); _ReadWriteBarrier(); } #endif #if defined(__GNUC__) #define TEST_ALWAYS_INLINE __attribute__((always_inline)) #define TEST_NOINLINE __attribute__((noinline)) #elif defined(_MSC_VER) #define TEST_ALWAYS_INLINE __forceinline #define TEST_NOINLINE __declspec(noinline) #else #define TEST_ALWAYS_INLINE #define TEST_NOINLINE #endif #if defined(__GNUC__) #pragma GCC diagnostic pop #endif #endif // SUPPORT_TEST_MACROS_HPP