# Distributed under the OSI-approved BSD 3-Clause License. See accompanying # file LICENSE.rst or https://cmake.org/licensing for details. #[=======================================================================[.rst: CheckStructHasMember -------------------- This module provides a command to check whether a struct or class has a specified member variable. Load this module in a CMake project with: .. code-block:: cmake include(CheckStructHasMember) Commands ^^^^^^^^ This module provides the following command: .. command:: check_struct_has_member Checks once if the given struct or class has the specified member variable: .. code-block:: cmake check_struct_has_member( [LANGUAGE ] ) This command checks once whether the struct or class ```` contains the specified ```` after including the given header(s) ```` where the prototype should be declared. Multiple header files can be specified in one argument as a string using a :ref:`semicolon-separated list `. The result is stored in an internal cache variable ````. The options are: ``LANGUAGE `` Use the ```` compiler to perform the check. Acceptable values are ``C`` and ``CXX``. If not specified, it defaults to ``C``. .. 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: Checking C Struct Member """"""""""""""""""""""""""""""""" In the following example, this module checks if the C struct ``timeval`` has a member variable ``tv_sec`` after including the ```` header. The result of the check is stored in the internal cache variable ``HAVE_TIMEVAL_TV_SEC``. .. code-block:: cmake include(CheckStructHasMember) check_struct_has_member( "struct timeval" tv_sec sys/select.h HAVE_TIMEVAL_TV_SEC ) Example: Checking C++ Struct Member """"""""""""""""""""""""""""""""""" In the following example, this module checks if the C++ struct ``std::tm`` has a member variable ``tm_gmtoff`` after including the ```` header. The result of the check is stored in the internal cache variable ``HAVE_TM_GMTOFF``. .. code-block:: cmake include(CheckStructHasMember) check_struct_has_member( std::tm tm_gmtoff ctime HAVE_TM_GMTOFF LANGUAGE CXX ) Example: Isolated Check With Compile Definitions """""""""""""""""""""""""""""""""""""""""""""""" In the following example, the check is performed with temporarily modified compile definitions using the :module:`CMakePushCheckState` module: .. code-block:: cmake include(CheckStructHasMember) include(CMakePushCheckState) cmake_push_check_state(RESET) set(CMAKE_REQUIRED_DEFINITIONS -D_GNU_SOURCE) check_struct_has_member( "struct utsname" domainname sys/utsname.h HAVE_UTSNAME_DOMAINNAME ) cmake_pop_check_state() #]=======================================================================] include_guard(GLOBAL) include(CheckSourceCompiles) macro (CHECK_STRUCT_HAS_MEMBER _STRUCT _MEMBER _HEADER _RESULT) set(_INCLUDE_FILES) foreach (it ${_HEADER}) string(APPEND _INCLUDE_FILES "#include <${it}>\n") endforeach () if("x${ARGN}" STREQUAL "x") set(_lang C) elseif("x${ARGN}" MATCHES "^xLANGUAGE;([a-zA-Z]+)$") set(_lang "${CMAKE_MATCH_1}") else() message(FATAL_ERROR "Unknown arguments:\n ${ARGN}\n") endif() set(_CHECK_STRUCT_MEMBER_SOURCE_CODE " ${_INCLUDE_FILES} int main(void) { (void)sizeof(((${_STRUCT} *)0)->${_MEMBER}); return 0; } ") if("${_lang}" STREQUAL "C") check_source_compiles(C "${_CHECK_STRUCT_MEMBER_SOURCE_CODE}" ${_RESULT}) elseif("${_lang}" STREQUAL "CXX") check_source_compiles(CXX "${_CHECK_STRUCT_MEMBER_SOURCE_CODE}" ${_RESULT}) else() message(FATAL_ERROR "Unknown language:\n ${_lang}\nSupported languages: C, CXX.\n") endif() endmacro () # FIXME(#24994): The following modules are included only for compatibility # with projects that accidentally relied on them with CMake 3.26 and below. include(CheckCSourceCompiles) include(CheckCXXSourceCompiles)