mirror of
https://github.com/Kitware/CMake.git
synced 2025-10-14 02:08:27 +08:00

Changes: - Added intro code blocks showing how to include these modules. - Used "command" instead of "macro". - Added some basic examples sections. - Added a rubric title for variables that affect the checks. - Synced and reworded descriptions. - Added "See Also" sections.
192 lines
5.9 KiB
CMake
192 lines
5.9 KiB
CMake
# Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
# file LICENSE.rst or https://cmake.org/licensing for details.
|
|
|
|
#[=======================================================================[.rst:
|
|
CheckFunctionExists
|
|
-------------------
|
|
|
|
This module provides a command to check whether a C function exists.
|
|
|
|
Load this module in a CMake project with:
|
|
|
|
.. code-block:: cmake
|
|
|
|
include(CheckFunctionExists)
|
|
|
|
Commands
|
|
^^^^^^^^
|
|
|
|
This module provides the following command:
|
|
|
|
.. command:: check_function_exists
|
|
|
|
Checks once whether a C function can be linked from system libraries:
|
|
|
|
.. code-block:: cmake
|
|
|
|
check_function_exists(<function> <variable>)
|
|
|
|
This command checks whether the ``<function>`` is provided by libraries
|
|
on the system, and stores the result in an internal cache variable
|
|
``<variable>``.
|
|
|
|
.. note::
|
|
|
|
Prefer using :module:`CheckSymbolExists` or :module:`CheckSourceCompiles`
|
|
instead of this command, for the following reasons:
|
|
|
|
* ``check_function_exists()`` can't detect functions that are inlined
|
|
in headers or defined as preprocessor macros.
|
|
|
|
* ``check_function_exists()`` can't detect anything in the 32-bit
|
|
versions of the Win32 API, because of a mismatch in calling conventions.
|
|
|
|
* ``check_function_exists()`` only verifies linking, it does not verify
|
|
that the function is declared in system headers.
|
|
|
|
.. rubric:: Variables Affecting the Check
|
|
|
|
The following variables may be set before calling this command to modify
|
|
the way the check is run:
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_FLAGS.rst
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_DEFINITIONS.rst
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_INCLUDES.rst
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_LINK_OPTIONS.rst
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_LIBRARIES.rst
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_LINK_DIRECTORIES.rst
|
|
|
|
.. include:: /module/include/CMAKE_REQUIRED_QUIET.rst
|
|
|
|
Examples
|
|
^^^^^^^^
|
|
|
|
Example: Basic Usage
|
|
""""""""""""""""""""
|
|
|
|
In the following example, a check is performed to determine whether the
|
|
linker sees the C function ``fopen()``, and the result is stored in the
|
|
``HAVE_FOPEN`` internal cache variable:
|
|
|
|
.. code-block:: cmake
|
|
|
|
include(CheckFunctionExists)
|
|
|
|
check_function_exists(fopen HAVE_FOPEN)
|
|
|
|
Example: Missing Declaration
|
|
""""""""""""""""""""""""""""
|
|
|
|
As noted above, the :module:`CheckSymbolExists` module is preferred for
|
|
checking C functions, since it also verifies whether the function is
|
|
declared or defined as a macro. In the following example, this module is
|
|
used to check an edge case where a function may not be declared in system
|
|
headers. For instance, on macOS, the ``fdatasync()`` function may be
|
|
available in the C library, but its declaration is not provided in the
|
|
``unistd.h`` system header.
|
|
|
|
.. code-block:: cmake
|
|
:caption: ``CMakeLists.txt``
|
|
|
|
include(CheckFunctionExists)
|
|
include(CheckSymbolExists)
|
|
|
|
check_symbol_exists(fdatasync "unistd.h" HAVE_FDATASYNC)
|
|
|
|
# Check if fdatasync() is available in the C library.
|
|
if(NOT HAVE_FDATASYNC)
|
|
check_function_exists(fdatasync HAVE_FDATASYNC_WITHOUT_DECL)
|
|
endif()
|
|
|
|
In such a case, the project can provide its own declaration if missing:
|
|
|
|
.. code-block:: c
|
|
:caption: ``example.c``
|
|
|
|
#ifdef HAVE_FDATASYNC_WITHOUT_DECL
|
|
extern int fdatasync(int);
|
|
#endif
|
|
|
|
See Also
|
|
^^^^^^^^
|
|
|
|
* The :module:`CheckSymbolExists` module to check whether a C symbol exists.
|
|
* The :module:`CheckSourceCompiles` module to check whether a source code
|
|
can be compiled.
|
|
* The :module:`CheckFortranFunctionExists` module to check whether a
|
|
Fortran function exists.
|
|
#]=======================================================================]
|
|
|
|
include_guard(GLOBAL)
|
|
|
|
macro(CHECK_FUNCTION_EXISTS FUNCTION VARIABLE)
|
|
if(NOT DEFINED "${VARIABLE}" OR "x${${VARIABLE}}" STREQUAL "x${VARIABLE}")
|
|
set(MACRO_CHECK_FUNCTION_DEFINITIONS
|
|
"-DCHECK_FUNCTION_EXISTS=${FUNCTION} ${CMAKE_REQUIRED_FLAGS}")
|
|
if(NOT CMAKE_REQUIRED_QUIET)
|
|
message(CHECK_START "Looking for ${FUNCTION}")
|
|
endif()
|
|
if(CMAKE_REQUIRED_LINK_OPTIONS)
|
|
set(CHECK_FUNCTION_EXISTS_ADD_LINK_OPTIONS
|
|
LINK_OPTIONS ${CMAKE_REQUIRED_LINK_OPTIONS})
|
|
else()
|
|
set(CHECK_FUNCTION_EXISTS_ADD_LINK_OPTIONS)
|
|
endif()
|
|
if(CMAKE_REQUIRED_LIBRARIES)
|
|
set(CHECK_FUNCTION_EXISTS_ADD_LIBRARIES
|
|
LINK_LIBRARIES ${CMAKE_REQUIRED_LIBRARIES})
|
|
else()
|
|
set(CHECK_FUNCTION_EXISTS_ADD_LIBRARIES)
|
|
endif()
|
|
if(CMAKE_REQUIRED_LINK_DIRECTORIES)
|
|
set(_CFE_LINK_DIRECTORIES
|
|
"-DLINK_DIRECTORIES:STRING=${CMAKE_REQUIRED_LINK_DIRECTORIES}")
|
|
else()
|
|
set(_CFE_LINK_DIRECTORIES)
|
|
endif()
|
|
if(CMAKE_REQUIRED_INCLUDES)
|
|
set(CHECK_FUNCTION_EXISTS_ADD_INCLUDES
|
|
"-DINCLUDE_DIRECTORIES:STRING=${CMAKE_REQUIRED_INCLUDES}")
|
|
else()
|
|
set(CHECK_FUNCTION_EXISTS_ADD_INCLUDES)
|
|
endif()
|
|
|
|
if(CMAKE_C_COMPILER_LOADED)
|
|
set(_cfe_source CheckFunctionExists.c)
|
|
elseif(CMAKE_CXX_COMPILER_LOADED)
|
|
set(_cfe_source CheckFunctionExists.cxx)
|
|
else()
|
|
message(FATAL_ERROR "CHECK_FUNCTION_EXISTS needs either C or CXX language enabled")
|
|
endif()
|
|
|
|
try_compile(${VARIABLE}
|
|
SOURCE_FROM_FILE "${_cfe_source}" "${CMAKE_ROOT}/Modules/CheckFunctionExists.c"
|
|
COMPILE_DEFINITIONS ${CMAKE_REQUIRED_DEFINITIONS}
|
|
${CHECK_FUNCTION_EXISTS_ADD_LINK_OPTIONS}
|
|
${CHECK_FUNCTION_EXISTS_ADD_LIBRARIES}
|
|
CMAKE_FLAGS -DCOMPILE_DEFINITIONS:STRING=${MACRO_CHECK_FUNCTION_DEFINITIONS}
|
|
"${CHECK_FUNCTION_EXISTS_ADD_INCLUDES}"
|
|
"${_CFE_LINK_DIRECTORIES}"
|
|
)
|
|
unset(_cfe_source)
|
|
unset(_CFE_LINK_DIRECTORIES)
|
|
|
|
if(${VARIABLE})
|
|
set(${VARIABLE} 1 CACHE INTERNAL "Have function ${FUNCTION}")
|
|
if(NOT CMAKE_REQUIRED_QUIET)
|
|
message(CHECK_PASS "found")
|
|
endif()
|
|
else()
|
|
if(NOT CMAKE_REQUIRED_QUIET)
|
|
message(CHECK_FAIL "not found")
|
|
endif()
|
|
set(${VARIABLE} "" CACHE INTERNAL "Have function ${FUNCTION}")
|
|
endif()
|
|
endif()
|
|
endmacro()
|