| # atomic builtins are required for threading support. |
| |
| INCLUDE(CheckCXXSourceCompiles) |
| INCLUDE(CheckLibraryExists) |
| |
| # Sometimes linking against libatomic is required for atomic ops, if |
| # the platform doesn't support lock-free atomics. |
| |
| function(check_working_cxx_atomics varname) |
| set(OLD_CMAKE_REQUIRED_FLAGS ${CMAKE_REQUIRED_FLAGS}) |
| set(CMAKE_REQUIRED_FLAGS "${CMAKE_REQUIRED_FLAGS} -std=c++11") |
| CHECK_CXX_SOURCE_COMPILES(" |
| #include <atomic> |
| std::atomic<int> x; |
| int main() { |
| return x; |
| } |
| " ${varname}) |
| set(CMAKE_REQUIRED_FLAGS ${OLD_CMAKE_REQUIRED_FLAGS}) |
| endfunction(check_working_cxx_atomics) |
| |
| function(check_working_cxx_atomics64 varname) |
| set(OLD_CMAKE_REQUIRED_FLAGS ${CMAKE_REQUIRED_FLAGS}) |
| set(CMAKE_REQUIRED_FLAGS "-std=c++11 ${CMAKE_REQUIRED_FLAGS}") |
| CHECK_CXX_SOURCE_COMPILES(" |
| #include <atomic> |
| #include <cstdint> |
| std::atomic<uint64_t> x (0); |
| int main() { |
| uint64_t i = x.load(std::memory_order_relaxed); |
| return 0; |
| } |
| " ${varname}) |
| set(CMAKE_REQUIRED_FLAGS ${OLD_CMAKE_REQUIRED_FLAGS}) |
| endfunction(check_working_cxx_atomics64) |
| |
| |
| # This isn't necessary on MSVC, so avoid command-line switch annoyance |
| # by only running on GCC-like hosts. |
| if (LLVM_COMPILER_IS_GCC_COMPATIBLE) |
| # First check if atomics work without the library. |
| check_working_cxx_atomics(HAVE_CXX_ATOMICS_WITHOUT_LIB) |
| # If not, check if the library exists, and atomics work with it. |
| if(NOT HAVE_CXX_ATOMICS_WITHOUT_LIB) |
| check_library_exists(atomic __atomic_fetch_add_4 "" HAVE_LIBATOMIC) |
| if( HAVE_LIBATOMIC ) |
| list(APPEND CMAKE_REQUIRED_LIBRARIES "atomic") |
| check_working_cxx_atomics(HAVE_CXX_ATOMICS_WITH_LIB) |
| if (NOT HAVE_CXX_ATOMICS_WITH_LIB) |
| message(FATAL_ERROR "Host compiler must support std::atomic!") |
| endif() |
| else() |
| message(FATAL_ERROR "Host compiler appears to require libatomic, but cannot find it.") |
| endif() |
| endif() |
| endif() |
| |
| # Check for 64 bit atomic operations. |
| if(MSVC) |
| set(HAVE_CXX_ATOMICS64_WITHOUT_LIB True) |
| else() |
| check_working_cxx_atomics64(HAVE_CXX_ATOMICS64_WITHOUT_LIB) |
| endif() |
| |
| # If not, check if the library exists, and atomics work with it. |
| if(NOT HAVE_CXX_ATOMICS64_WITHOUT_LIB) |
| check_library_exists(atomic __atomic_load_8 "" HAVE_CXX_LIBATOMICS64) |
| if(HAVE_CXX_LIBATOMICS64) |
| list(APPEND CMAKE_REQUIRED_LIBRARIES "atomic") |
| check_working_cxx_atomics64(HAVE_CXX_ATOMICS64_WITH_LIB) |
| if (NOT HAVE_CXX_ATOMICS64_WITH_LIB) |
| message(FATAL_ERROR "Host compiler must support std::atomic!") |
| endif() |
| else() |
| message(FATAL_ERROR "Host compiler appears to require libatomic, but cannot find it.") |
| endif() |
| endif() |
| |
| ## TODO: This define is only used for the legacy atomic operations in |
| ## llvm's Atomic.h, which should be replaced. Other code simply |
| ## assumes C++11 <atomic> works. |
| CHECK_CXX_SOURCE_COMPILES(" |
| #ifdef _MSC_VER |
| #include <windows.h> |
| #endif |
| int main() { |
| #ifdef _MSC_VER |
| volatile LONG val = 1; |
| MemoryBarrier(); |
| InterlockedCompareExchange(&val, 0, 1); |
| InterlockedIncrement(&val); |
| InterlockedDecrement(&val); |
| #else |
| volatile unsigned long val = 1; |
| __sync_synchronize(); |
| __sync_val_compare_and_swap(&val, 1, 0); |
| __sync_add_and_fetch(&val, 1); |
| __sync_sub_and_fetch(&val, 1); |
| #endif |
| return 0; |
| } |
| " LLVM_HAS_ATOMICS) |
| |
| if( NOT LLVM_HAS_ATOMICS ) |
| message(STATUS "Warning: LLVM will be built thread-unsafe because atomic builtins are missing") |
| endif() |