CheckSymbolExists.cmake 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. # Distributed under the OSI-approved BSD 3-Clause License. See accompanying
  2. # file Copyright.txt or https://cmake.org/licensing for details.
  3. #[=======================================================================[.rst:
  4. CheckSymbolExists
  5. -----------------
  6. Provides a macro to check if a symbol exists as a function, variable,
  7. or macro in ``C``.
  8. .. command:: check_symbol_exists
  9. .. code-block:: cmake
  10. check_symbol_exists(<symbol> <files> <variable>)
  11. Check that the ``<symbol>`` is available after including given header
  12. ``<files>`` and store the result in a ``<variable>``. Specify the list
  13. of files in one argument as a semicolon-separated list.
  14. ``<variable>`` will be created as an internal cache variable.
  15. If the header files define the symbol as a macro it is considered
  16. available and assumed to work. If the header files declare the symbol
  17. as a function or variable then the symbol must also be available for
  18. linking (so intrinsics may not be detected).
  19. If the symbol is a type, enum value, or intrinsic it will not be recognized
  20. (consider using :module:`CheckTypeSize` or :module:`CheckSourceCompiles`).
  21. If the check needs to be done in C++, consider using
  22. :module:`CheckCXXSymbolExists` instead.
  23. The following variables may be set before calling this macro to modify
  24. the way the check is run:
  25. ``CMAKE_REQUIRED_FLAGS``
  26. string of compile command line flags.
  27. ``CMAKE_REQUIRED_DEFINITIONS``
  28. a :ref:`;-list <CMake Language Lists>` of macros to define (-DFOO=bar).
  29. ``CMAKE_REQUIRED_INCLUDES``
  30. a :ref:`;-list <CMake Language Lists>` of header search paths to pass to
  31. the compiler.
  32. ``CMAKE_REQUIRED_LINK_OPTIONS``
  33. .. versionadded:: 3.14
  34. a :ref:`;-list <CMake Language Lists>` of options to add to the link command.
  35. ``CMAKE_REQUIRED_LIBRARIES``
  36. a :ref:`;-list <CMake Language Lists>` of libraries to add to the link
  37. command. See policy :policy:`CMP0075`.
  38. ``CMAKE_REQUIRED_QUIET``
  39. .. versionadded:: 3.1
  40. execute quietly without messages.
  41. For example:
  42. .. code-block:: cmake
  43. include(CheckSymbolExists)
  44. # Check for macro SEEK_SET
  45. check_symbol_exists(SEEK_SET "stdio.h" HAVE_SEEK_SET)
  46. # Check for function fopen
  47. check_symbol_exists(fopen "stdio.h" HAVE_FOPEN)
  48. #]=======================================================================]
  49. include_guard(GLOBAL)
  50. cmake_policy(PUSH)
  51. cmake_policy(SET CMP0054 NEW) # if() quoted variables not dereferenced
  52. macro(CHECK_SYMBOL_EXISTS SYMBOL FILES VARIABLE)
  53. if(CMAKE_C_COMPILER_LOADED)
  54. __CHECK_SYMBOL_EXISTS_FILTER_FLAGS(C)
  55. __CHECK_SYMBOL_EXISTS_IMPL(CheckSymbolExists.c "${SYMBOL}" "${FILES}" "${VARIABLE}" )
  56. __CHECK_SYMBOL_EXISTS_RESTORE_FLAGS(C)
  57. elseif(CMAKE_CXX_COMPILER_LOADED)
  58. __CHECK_SYMBOL_EXISTS_FILTER_FLAGS(CXX)
  59. __CHECK_SYMBOL_EXISTS_IMPL(CheckSymbolExists.cxx "${SYMBOL}" "${FILES}" "${VARIABLE}" )
  60. __CHECK_SYMBOL_EXISTS_RESTORE_FLAGS(CXX)
  61. else()
  62. message(FATAL_ERROR "CHECK_SYMBOL_EXISTS needs either C or CXX language enabled")
  63. endif()
  64. endmacro()
  65. macro(__CHECK_SYMBOL_EXISTS_FILTER_FLAGS LANG)
  66. set(__CMAKE_${LANG}_FLAGS_SAVED "${CMAKE_${LANG}_FLAGS}")
  67. string(REGEX REPLACE "(^| )-Werror([= ][^ ]*)?( |$)" " " CMAKE_${LANG}_FLAGS "${CMAKE_${LANG}_FLAGS}")
  68. string(REGEX REPLACE "(^| )-pedantic-errors( |$)" " " CMAKE_${LANG}_FLAGS "${CMAKE_${LANG}_FLAGS}")
  69. endmacro()
  70. macro(__CHECK_SYMBOL_EXISTS_RESTORE_FLAGS LANG)
  71. set(CMAKE_${LANG}_FLAGS "${__CMAKE_${LANG}_FLAGS_SAVED}")
  72. unset(__CMAKE_${LANG}_FLAGS_SAVED)
  73. endmacro()
  74. macro(__CHECK_SYMBOL_EXISTS_IMPL SOURCEFILE SYMBOL FILES VARIABLE)
  75. if(NOT DEFINED "${VARIABLE}" OR "x${${VARIABLE}}" STREQUAL "x${VARIABLE}")
  76. set(_CSE_SOURCE "/* */\n")
  77. set(MACRO_CHECK_SYMBOL_EXISTS_FLAGS ${CMAKE_REQUIRED_FLAGS})
  78. if(CMAKE_REQUIRED_LINK_OPTIONS)
  79. set(CHECK_SYMBOL_EXISTS_LINK_OPTIONS
  80. LINK_OPTIONS ${CMAKE_REQUIRED_LINK_OPTIONS})
  81. else()
  82. set(CHECK_SYMBOL_EXISTS_LINK_OPTIONS)
  83. endif()
  84. if(CMAKE_REQUIRED_LIBRARIES)
  85. set(CHECK_SYMBOL_EXISTS_LIBS
  86. LINK_LIBRARIES ${CMAKE_REQUIRED_LIBRARIES})
  87. else()
  88. set(CHECK_SYMBOL_EXISTS_LIBS)
  89. endif()
  90. if(CMAKE_REQUIRED_INCLUDES)
  91. set(CMAKE_SYMBOL_EXISTS_INCLUDES
  92. "-DINCLUDE_DIRECTORIES:STRING=${CMAKE_REQUIRED_INCLUDES}")
  93. else()
  94. set(CMAKE_SYMBOL_EXISTS_INCLUDES)
  95. endif()
  96. foreach(FILE ${FILES})
  97. string(APPEND _CSE_SOURCE
  98. "#include <${FILE}>\n")
  99. endforeach()
  100. string(APPEND _CSE_SOURCE "
  101. int main(int argc, char** argv)
  102. {
  103. (void)argv;")
  104. set(_CSE_CHECK_NON_MACRO "return ((int*)(&${SYMBOL}))[argc];")
  105. if("${SYMBOL}" MATCHES "^[a-zA-Z_][a-zA-Z0-9_]*$")
  106. # The SYMBOL has a legal macro name. Test whether it exists as a macro.
  107. string(APPEND _CSE_SOURCE "
  108. #ifndef ${SYMBOL}
  109. ${_CSE_CHECK_NON_MACRO}
  110. #else
  111. (void)argc;
  112. return 0;
  113. #endif")
  114. else()
  115. # The SYMBOL cannot be a macro (e.g., a template function).
  116. string(APPEND _CSE_SOURCE "
  117. ${_CSE_CHECK_NON_MACRO}")
  118. endif()
  119. string(APPEND _CSE_SOURCE "
  120. }")
  121. unset(_CSE_CHECK_NON_MACRO)
  122. if(NOT CMAKE_REQUIRED_QUIET)
  123. message(CHECK_START "Looking for ${SYMBOL}")
  124. endif()
  125. try_compile(${VARIABLE}
  126. SOURCE_FROM_VAR "${SOURCEFILE}" _CSE_SOURCE
  127. COMPILE_DEFINITIONS ${CMAKE_REQUIRED_DEFINITIONS}
  128. ${CHECK_SYMBOL_EXISTS_LINK_OPTIONS}
  129. ${CHECK_SYMBOL_EXISTS_LIBS}
  130. CMAKE_FLAGS
  131. -DCOMPILE_DEFINITIONS:STRING=${MACRO_CHECK_SYMBOL_EXISTS_FLAGS}
  132. "${CMAKE_SYMBOL_EXISTS_INCLUDES}"
  133. OUTPUT_VARIABLE OUTPUT)
  134. if(${VARIABLE})
  135. if(NOT CMAKE_REQUIRED_QUIET)
  136. message(CHECK_PASS "found")
  137. endif()
  138. set(${VARIABLE} 1 CACHE INTERNAL "Have symbol ${SYMBOL}")
  139. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeOutput.log
  140. "Determining if the ${SYMBOL} "
  141. "exist passed with the following output:\n"
  142. "${OUTPUT}\nFile ${SOURCEFILE}:\n"
  143. "${_CSE_SOURCE}\n")
  144. else()
  145. if(NOT CMAKE_REQUIRED_QUIET)
  146. message(CHECK_FAIL "not found")
  147. endif()
  148. set(${VARIABLE} "" CACHE INTERNAL "Have symbol ${SYMBOL}")
  149. file(APPEND ${CMAKE_BINARY_DIR}${CMAKE_FILES_DIRECTORY}/CMakeError.log
  150. "Determining if the ${SYMBOL} "
  151. "exist failed with the following output:\n"
  152. "${OUTPUT}\nFile ${SOURCEFILE}:\n"
  153. "${_CSE_SOURCE}\n")
  154. endif()
  155. unset(_CSE_SOURCE)
  156. endif()
  157. endmacro()
  158. cmake_policy(POP)