107 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			CMake
		
	
	
	
			
		
		
	
	
			107 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			CMake
		
	
	
	
# 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()
 |