| # Distributed under the OSI-approved BSD 3-Clause License. See accompanying |
| # file Copyright.txt or https://cmake.org/licensing for details. |
| |
| #[=======================================================================[.rst: |
| FindThreads |
| ----------- |
| |
| This module determines the thread library of the system. |
| |
| Imported Targets |
| ^^^^^^^^^^^^^^^^ |
| |
| .. versionadded:: 3.1 |
| |
| This module defines the following :prop_tgt:`IMPORTED` target: |
| |
| ``Threads::Threads`` |
| The thread library, if found. |
| |
| Result Variables |
| ^^^^^^^^^^^^^^^^ |
| |
| The following variables are set: |
| |
| ``Threads_FOUND`` |
| If a supported thread library was found. |
| ``CMAKE_THREAD_LIBS_INIT`` |
| The thread library to use. This may be empty if the thread functions |
| are provided by the system libraries and no special flags are needed |
| to use them. |
| ``CMAKE_USE_WIN32_THREADS_INIT`` |
| If the found thread library is the win32 one. |
| ``CMAKE_USE_PTHREADS_INIT`` |
| If the found thread library is pthread compatible. |
| ``CMAKE_HP_PTHREADS_INIT`` |
| If the found thread library is the HP thread library. |
| |
| Variables Affecting Behavior |
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ |
| |
| .. variable:: THREADS_PREFER_PTHREAD_FLAG |
| |
| .. versionadded:: 3.1 |
| |
| If the use of the -pthread compiler and linker flag is preferred then |
| the caller can set this variable to TRUE. The compiler flag can only be |
| used with the imported target. Use of both the imported target as well |
| as this switch is highly recommended for new code. |
| |
| This variable has no effect if the system libraries provide the |
| thread functions, i.e. when ``CMAKE_THREAD_LIBS_INIT`` will be empty. |
| #]=======================================================================] |
| |
| include (CheckLibraryExists) |
| set(Threads_FOUND FALSE) |
| set(CMAKE_REQUIRED_QUIET_SAVE ${CMAKE_REQUIRED_QUIET}) |
| set(CMAKE_REQUIRED_QUIET ${Threads_FIND_QUIETLY}) |
| |
| if(CMAKE_C_COMPILER_LOADED) |
| include (CheckIncludeFile) |
| include (CheckCSourceCompiles) |
| elseif(CMAKE_CXX_COMPILER_LOADED) |
| include (CheckIncludeFileCXX) |
| include (CheckCXXSourceCompiles) |
| else() |
| message(FATAL_ERROR "FindThreads only works if either C or CXX language is enabled") |
| endif() |
| |
| # simple pthread test code |
| set(PTHREAD_C_CXX_TEST_SOURCE [====[ |
| #include <pthread.h> |
| |
| static void* test_func(void* data) |
| { |
| return data; |
| } |
| |
| int main(void) |
| { |
| pthread_t thread; |
| pthread_create(&thread, NULL, test_func, NULL); |
| pthread_detach(thread); |
| pthread_cancel(thread); |
| pthread_join(thread, NULL); |
| pthread_atfork(NULL, NULL, NULL); |
| pthread_exit(NULL); |
| |
| return 0; |
| } |
| ]====]) |
| |
| # Internal helper macro. |
| # Do NOT even think about using it outside of this file! |
| macro(_threads_check_libc) |
| if(NOT Threads_FOUND) |
| if(CMAKE_C_COMPILER_LOADED) |
| CHECK_C_SOURCE_COMPILES("${PTHREAD_C_CXX_TEST_SOURCE}" CMAKE_HAVE_LIBC_PTHREAD) |
| elseif(CMAKE_CXX_COMPILER_LOADED) |
| CHECK_CXX_SOURCE_COMPILES("${PTHREAD_C_CXX_TEST_SOURCE}" CMAKE_HAVE_LIBC_PTHREAD) |
| endif() |
| if(CMAKE_HAVE_LIBC_PTHREAD) |
| set(CMAKE_THREAD_LIBS_INIT "") |
| set(Threads_FOUND TRUE) |
| endif() |
| endif () |
| endmacro() |
| |
| # Internal helper macro. |
| # Do NOT even think about using it outside of this file! |
| macro(_threads_check_lib LIBNAME FUNCNAME VARNAME) |
| if(NOT Threads_FOUND) |
| CHECK_LIBRARY_EXISTS(${LIBNAME} ${FUNCNAME} "" ${VARNAME}) |
| if(${VARNAME}) |
| set(CMAKE_THREAD_LIBS_INIT "-l${LIBNAME}") |
| set(Threads_FOUND TRUE) |
| endif() |
| endif () |
| endmacro() |
| |
| # Internal helper macro. |
| # Do NOT even think about using it outside of this file! |
| macro(_threads_check_flag_pthread) |
| if(NOT Threads_FOUND) |
| # If we did not find -lpthreads, -lpthread, or -lthread, look for -pthread |
| # except on compilers known to not have it. |
| if(MSVC) |
| # Compilers targeting the MSVC ABI do not have a -pthread flag. |
| set(THREADS_HAVE_PTHREAD_ARG FALSE) |
| elseif(NOT DEFINED THREADS_HAVE_PTHREAD_ARG) |
| message(CHECK_START "Check if compiler accepts -pthread") |
| if(CMAKE_C_COMPILER_LOADED) |
| set(_threads_src CheckForPthreads.c) |
| elseif(CMAKE_CXX_COMPILER_LOADED) |
| set(_threads_src CheckForPthreads.cxx) |
| endif() |
| try_compile(THREADS_HAVE_PTHREAD_ARG |
| SOURCE_FROM_FILE "${_threads_src}" "${CMAKE_CURRENT_LIST_DIR}/CheckForPthreads.c" |
| CMAKE_FLAGS -DLINK_LIBRARIES:STRING=-pthread |
| OUTPUT_VARIABLE _cmake_check_pthreads_output) |
| |
| string(APPEND _cmake_find_threads_output "${_cmake_check_pthreads_output}") |
| unset(_cmake_check_pthreads_output) |
| unset(_threads_src) |
| |
| if(THREADS_HAVE_PTHREAD_ARG) |
| set(Threads_FOUND TRUE) |
| message(CHECK_PASS "yes") |
| else() |
| message(CHECK_FAIL "no") |
| endif() |
| |
| endif() |
| |
| if(THREADS_HAVE_PTHREAD_ARG) |
| set(Threads_FOUND TRUE) |
| set(CMAKE_THREAD_LIBS_INIT "-pthread") |
| endif() |
| endif() |
| endmacro() |
| |
| # Check if pthread functions are in normal C library. |
| # We list some pthread functions in PTHREAD_C_CXX_TEST_SOURCE test code. |
| # If the pthread functions already exist in C library, we could just use |
| # them instead of linking to the additional pthread library. |
| _threads_check_libc() |
| |
| # Check for -pthread first if enabled. This is the recommended |
| # way, but not backwards compatible as one must also pass -pthread |
| # as compiler flag then. |
| if (THREADS_PREFER_PTHREAD_FLAG) |
| _threads_check_flag_pthread() |
| endif () |
| |
| if(CMAKE_SYSTEM MATCHES "GHS-MULTI") |
| _threads_check_lib(posix pthread_create CMAKE_HAVE_PTHREADS_CREATE) |
| endif() |
| _threads_check_lib(pthreads pthread_create CMAKE_HAVE_PTHREADS_CREATE) |
| _threads_check_lib(pthread pthread_create CMAKE_HAVE_PTHREAD_CREATE) |
| |
| if (NOT THREADS_PREFER_PTHREAD_FLAG) |
| _threads_check_flag_pthread() |
| endif() |
| |
| if(CMAKE_THREAD_LIBS_INIT OR CMAKE_HAVE_LIBC_PTHREAD) |
| set(CMAKE_USE_PTHREADS_INIT 1) |
| set(Threads_FOUND TRUE) |
| endif() |
| |
| if(CMAKE_SYSTEM_NAME MATCHES "Windows") |
| set(CMAKE_USE_WIN32_THREADS_INIT 1) |
| set(Threads_FOUND TRUE) |
| endif() |
| |
| if(CMAKE_USE_PTHREADS_INIT) |
| if(CMAKE_SYSTEM_NAME MATCHES "HP-UX") |
| # Use libcma if it exists and can be used. It provides more |
| # symbols than the plain pthread library. CMA threads |
| # have actually been deprecated: |
| # http://docs.hp.com/en/B3920-90091/ch12s03.html#d0e11395 |
| # http://docs.hp.com/en/947/d8.html |
| # but we need to maintain compatibility here. |
| # The CMAKE_HP_PTHREADS setting actually indicates whether CMA threads |
| # are available. |
| CHECK_LIBRARY_EXISTS(cma pthread_attr_create "" CMAKE_HAVE_HP_CMA) |
| if(CMAKE_HAVE_HP_CMA) |
| set(CMAKE_THREAD_LIBS_INIT "-lcma") |
| set(CMAKE_HP_PTHREADS_INIT 1) |
| set(Threads_FOUND TRUE) |
| endif() |
| set(CMAKE_USE_PTHREADS_INIT 1) |
| endif() |
| |
| if(CMAKE_SYSTEM MATCHES "OSF1-V") |
| set(CMAKE_USE_PTHREADS_INIT 0) |
| set(CMAKE_THREAD_LIBS_INIT ) |
| endif() |
| |
| if(CMAKE_SYSTEM MATCHES "CYGWIN_NT" OR CMAKE_SYSTEM MATCHES "MSYS_NT") |
| set(CMAKE_USE_PTHREADS_INIT 1) |
| set(Threads_FOUND TRUE) |
| set(CMAKE_THREAD_LIBS_INIT ) |
| set(CMAKE_USE_WIN32_THREADS_INIT 0) |
| endif() |
| endif() |
| |
| set(CMAKE_REQUIRED_QUIET ${CMAKE_REQUIRED_QUIET_SAVE}) |
| include(${CMAKE_CURRENT_LIST_DIR}/FindPackageHandleStandardArgs.cmake) |
| FIND_PACKAGE_HANDLE_STANDARD_ARGS(Threads DEFAULT_MSG Threads_FOUND) |
| |
| if(THREADS_FOUND AND NOT TARGET Threads::Threads) |
| add_library(Threads::Threads INTERFACE IMPORTED) |
| |
| if(THREADS_HAVE_PTHREAD_ARG) |
| set_property(TARGET Threads::Threads |
| PROPERTY INTERFACE_COMPILE_OPTIONS "$<$<COMPILE_LANG_AND_ID:CUDA,NVIDIA>:SHELL:-Xcompiler -pthread>" |
| "$<$<NOT:$<COMPILE_LANG_AND_ID:CUDA,NVIDIA>>:-pthread>") |
| endif() |
| |
| if(CMAKE_THREAD_LIBS_INIT) |
| set_property(TARGET Threads::Threads PROPERTY INTERFACE_LINK_LIBRARIES "${CMAKE_THREAD_LIBS_INIT}") |
| endif() |
| elseif(NOT THREADS_FOUND AND _cmake_find_threads_output) |
| file(APPEND |
| ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log |
| "Determining if compiler accepts -pthread failed with the following output:\n${_cmake_find_threads_output}\n\n") |
| endif() |
| |
| unset(_cmake_find_threads_output) |