| # Distributed under the OSI-approved BSD 3-Clause License. See accompanying |
| # file Copyright.txt or https://cmake.org/licensing for details. |
| |
| |
| # Function to compile a source file to identify the compiler ABI. |
| # This is used internally by CMake and should not be included by user |
| # code. |
| |
| include(${CMAKE_ROOT}/Modules/Internal/CMakeDetermineLinkerId.cmake) |
| include(${CMAKE_ROOT}/Modules/CMakeParseImplicitIncludeInfo.cmake) |
| include(${CMAKE_ROOT}/Modules/CMakeParseImplicitLinkInfo.cmake) |
| include(${CMAKE_ROOT}/Modules/CMakeParseLibraryArchitecture.cmake) |
| include(CMakeTestCompilerCommon) |
| |
| function(CMAKE_DETERMINE_COMPILER_ABI lang src) |
| if(NOT DEFINED CMAKE_${lang}_ABI_COMPILED) |
| message(CHECK_START "Detecting ${lang} compiler ABI info") |
| |
| # Compile the ABI identification source. |
| set(BIN "${CMAKE_PLATFORM_INFO_DIR}/CMakeDetermineCompilerABI_${lang}.bin") |
| set(CMAKE_FLAGS ) |
| set(COMPILE_DEFINITIONS ) |
| set(LINK_OPTIONS ) |
| if(DEFINED CMAKE_${lang}_VERBOSE_FLAG) |
| set(LINK_OPTIONS "${CMAKE_${lang}_VERBOSE_FLAG}") |
| set(COMPILE_DEFINITIONS "${CMAKE_${lang}_VERBOSE_FLAG}") |
| endif() |
| if(DEFINED CMAKE_${lang}_VERBOSE_COMPILE_FLAG) |
| set(COMPILE_DEFINITIONS "${CMAKE_${lang}_VERBOSE_COMPILE_FLAG}") |
| endif() |
| if(DEFINED CMAKE_${lang}_VERBOSE_LINK_FLAG) |
| list(APPEND LINK_OPTIONS "${CMAKE_${lang}_VERBOSE_LINK_FLAG}") |
| endif() |
| if(lang MATCHES "^(CUDA|HIP)$") |
| if(CMAKE_CUDA_ARCHITECTURES STREQUAL "native") |
| # We are about to detect the native architectures, so we do |
| # not yet know them. Use all architectures during detection. |
| set(CMAKE_${lang}_ARCHITECTURES "all") |
| endif() |
| set(CMAKE_${lang}_RUNTIME_LIBRARY "Static") |
| endif() |
| if(lang STREQUAL "CXX") |
| set(CMAKE_${lang}_SCAN_FOR_MODULES OFF) |
| endif() |
| if(NOT "x${CMAKE_${lang}_COMPILER_ID}" STREQUAL "xMSVC") |
| # Avoid adding our own platform standard libraries for compilers |
| # from which we might detect implicit link libraries. |
| list(APPEND CMAKE_FLAGS "-DCMAKE_${lang}_STANDARD_LIBRARIES=") |
| endif() |
| list(JOIN LINK_OPTIONS " " LINK_OPTIONS) |
| list(APPEND CMAKE_FLAGS "-DEXE_LINKER_FLAGS=${LINK_OPTIONS}") |
| |
| __TestCompiler_setTryCompileTargetType() |
| |
| # Avoid failing ABI detection on warnings. |
| if(CMAKE_TRY_COMPILE_CONFIGURATION) |
| string(TOUPPER "${CMAKE_TRY_COMPILE_CONFIGURATION}" _tc_config) |
| else() |
| set(_tc_config "DEBUG") |
| endif() |
| foreach(v CMAKE_${lang}_FLAGS CMAKE_${lang}_FLAGS_${_tc_config}) |
| string(REGEX REPLACE "(^| )-Werror([= ][^-][^ ]*)?( |$)" " " ${v} "${${v}}") |
| endforeach() |
| |
| # Save the current LC_ALL, LC_MESSAGES, and LANG environment variables |
| # and set them to "C" that way GCC's "search starts here" text is in |
| # English and we can grok it. |
| set(_orig_lc_all $ENV{LC_ALL}) |
| set(_orig_lc_messages $ENV{LC_MESSAGES}) |
| set(_orig_lang $ENV{LANG}) |
| set(ENV{LC_ALL} C) |
| set(ENV{LC_MESSAGES} C) |
| set(ENV{LANG} C) |
| try_compile(CMAKE_${lang}_ABI_COMPILED |
| SOURCES ${src} |
| CMAKE_FLAGS ${CMAKE_FLAGS} |
| # Ignore unused flags when we are just determining the ABI. |
| "--no-warn-unused-cli" |
| COMPILE_DEFINITIONS ${COMPILE_DEFINITIONS} |
| OUTPUT_VARIABLE OUTPUT |
| COPY_FILE "${BIN}" |
| COPY_FILE_ERROR _copy_error |
| __CMAKE_INTERNAL ABI |
| ) |
| |
| # Restore original LC_ALL, LC_MESSAGES, and LANG |
| set(ENV{LC_ALL} ${_orig_lc_all}) |
| set(ENV{LC_MESSAGES} ${_orig_lc_messages}) |
| set(ENV{LANG} ${_orig_lang}) |
| |
| # Move result from cache to normal variable. |
| set(CMAKE_${lang}_ABI_COMPILED ${CMAKE_${lang}_ABI_COMPILED}) |
| unset(CMAKE_${lang}_ABI_COMPILED CACHE) |
| if(CMAKE_${lang}_ABI_COMPILED AND _copy_error) |
| set(CMAKE_${lang}_ABI_COMPILED 0) |
| endif() |
| set(CMAKE_${lang}_ABI_COMPILED ${CMAKE_${lang}_ABI_COMPILED} PARENT_SCOPE) |
| |
| # Load the resulting information strings. |
| if(CMAKE_${lang}_ABI_COMPILED) |
| message(CHECK_PASS "done") |
| if(CMAKE_HOST_APPLE AND CMAKE_SYSTEM_NAME STREQUAL "Darwin" AND NOT CMAKE_OSX_ARCHITECTURES MATCHES "\\$") |
| file(READ_MACHO "${BIN}" ARCHITECTURES archs CAPTURE_ERROR macho_error) # undocumented file() subcommand |
| if (NOT macho_error) |
| # sort and prune the list of found architectures |
| set(arch_list_sorted ${archs}) |
| list(SORT arch_list_sorted) |
| list(REMOVE_DUPLICATES arch_list_sorted) |
| # sort and prune the list of requested architectures |
| set(requested_arch_list ${CMAKE_OSX_ARCHITECTURES}) |
| list(SORT requested_arch_list) |
| list(REMOVE_DUPLICATES requested_arch_list) |
| message(CONFIGURE_LOG |
| "Effective list of requested architectures (possibly empty) : \"${requested_arch_list}\"\n" |
| "Effective list of architectures found in the ABI info binary: \"${arch_list_sorted}\"\n") |
| # If all generated architectures were known to READ_MACHO (i.e. libmacho): |
| # Compare requested and found: |
| # - if no architecture(s) were requested explicitly, just check if READ_MACHO returned |
| # an architecture for the ABI info binary. |
| # - otherwise, check if the requested and found lists are equal |
| if(arch_list_sorted MATCHES "unknown") |
| # Rare but not impossible: a host with a toolchain capable of generating binaries with |
| # architectures that the system libmacho is too old to know. Report the found archs as |
| # usual, warn about the unknowns and skip the comparison with CMAKE_OSX_ARCHITECTURES. |
| message(WARNING "The ${lang} compiler generates universal binaries with at least 1 architecture not known to the host") |
| elseif(requested_arch_list AND arch_list_sorted |
| AND NOT "${requested_arch_list}" STREQUAL "${arch_list_sorted}") |
| # inform the user of the mismatch but show the raw input and output lists |
| message(FATAL_ERROR |
| "The ${lang} compiler targets architectures:\n" |
| " \"${archs}\"\n" |
| "but CMAKE_OSX_ARCHITECTURES is\n" |
| " \"${CMAKE_OSX_ARCHITECTURES}\"\n") |
| endif() |
| endif() |
| endif() |
| cmake_policy(PUSH) |
| cmake_policy(SET CMP0159 NEW) # file(STRINGS) with REGEX updates CMAKE_MATCH_<n> |
| file(STRINGS "${BIN}" ABI_STRINGS LIMIT_COUNT 32 REGEX "INFO:[A-Za-z0-9_]+\\[[^]]*\\]") |
| cmake_policy(POP) |
| set(ABI_SIZEOF_DPTR "NOTFOUND") |
| set(ABI_BYTE_ORDER "NOTFOUND") |
| set(ABI_NAME "NOTFOUND") |
| foreach(info ${ABI_STRINGS}) |
| if("${info}" MATCHES "INFO:sizeof_dptr\\[0*([^]]*)\\]" AND NOT ABI_SIZEOF_DPTR) |
| set(ABI_SIZEOF_DPTR "${CMAKE_MATCH_1}") |
| endif() |
| if("${info}" MATCHES "INFO:byte_order\\[(BIG_ENDIAN|LITTLE_ENDIAN)\\]") |
| set(byte_order "${CMAKE_MATCH_1}") |
| if(ABI_BYTE_ORDER STREQUAL "NOTFOUND") |
| # Tentatively use the value because this is the first occurrence. |
| set(ABI_BYTE_ORDER "${byte_order}") |
| elseif(NOT ABI_BYTE_ORDER STREQUAL "${byte_order}") |
| # Drop value because multiple occurrences do not match. |
| set(ABI_BYTE_ORDER "") |
| endif() |
| endif() |
| if("${info}" MATCHES "INFO:abi\\[([^]]*)\\]" AND NOT ABI_NAME) |
| set(ABI_NAME "${CMAKE_MATCH_1}") |
| endif() |
| endforeach() |
| |
| if(ABI_SIZEOF_DPTR) |
| set(CMAKE_${lang}_SIZEOF_DATA_PTR "${ABI_SIZEOF_DPTR}" PARENT_SCOPE) |
| elseif(CMAKE_${lang}_SIZEOF_DATA_PTR_DEFAULT) |
| set(CMAKE_${lang}_SIZEOF_DATA_PTR "${CMAKE_${lang}_SIZEOF_DATA_PTR_DEFAULT}" PARENT_SCOPE) |
| endif() |
| |
| if(ABI_BYTE_ORDER) |
| set(CMAKE_${lang}_BYTE_ORDER "${ABI_BYTE_ORDER}" PARENT_SCOPE) |
| endif() |
| |
| if(ABI_NAME) |
| set(CMAKE_${lang}_COMPILER_ABI "${ABI_NAME}" PARENT_SCOPE) |
| endif() |
| |
| # Parse implicit include directory for this language, if available. |
| if(CMAKE_${lang}_VERBOSE_FLAG) |
| set (implicit_incdirs "") |
| cmake_parse_implicit_include_info("${OUTPUT}" "${lang}" |
| implicit_incdirs log rv) |
| message(CONFIGURE_LOG |
| "Parsed ${lang} implicit include dir info: rv=${rv}\n${log}\n\n") |
| if("${rv}" STREQUAL "done") |
| # Entries that we have been told to explicitly pass as standard include |
| # directories will not be implicitly added by the compiler. |
| if(CMAKE_${lang}_STANDARD_INCLUDE_DIRECTORIES) |
| list(REMOVE_ITEM implicit_incdirs ${CMAKE_${lang}_STANDARD_INCLUDE_DIRECTORIES}) |
| endif() |
| |
| # We parsed implicit include directories, so override the default initializer. |
| set(_CMAKE_${lang}_IMPLICIT_INCLUDE_DIRECTORIES_INIT "${implicit_incdirs}") |
| endif() |
| endif() |
| set(CMAKE_${lang}_IMPLICIT_INCLUDE_DIRECTORIES "${_CMAKE_${lang}_IMPLICIT_INCLUDE_DIRECTORIES_INIT}" PARENT_SCOPE) |
| |
| if(_CMAKE_${lang}_IMPLICIT_LINK_INFORMATION_DETERMINED_EARLY) |
| # Use implicit linker information detected during compiler id step. |
| set(implicit_dirs "${CMAKE_${lang}_IMPLICIT_LINK_DIRECTORIES}") |
| set(implicit_objs "") |
| set(implicit_libs "${CMAKE_${lang}_IMPLICIT_LINK_LIBRARIES}") |
| set(implicit_fwks "${CMAKE_${lang}_IMPLICIT_LINK_FRAMEWORK_DIRECTORIES}") |
| else() |
| # Parse implicit linker information for this language, if available. |
| set(implicit_dirs "") |
| set(implicit_objs "") |
| set(implicit_libs "") |
| set(implicit_fwks "") |
| set(compute_artifacts COMPUTE_LINKER linker_tool) |
| if(CMAKE_${lang}_VERBOSE_FLAG) |
| list(APPEND compute_artifacts COMPUTE_IMPLICIT_LIBS implicit_libs |
| COMPUTE_IMPLICIT_DIRS implicit_dirs |
| COMPUTE_IMPLICIT_FWKS implicit_fwks |
| COMPUTE_IMPLICIT_OBJECTS implicit_objs) |
| endif() |
| cmake_parse_implicit_link_info2("${OUTPUT}" log "${CMAKE_${lang}_IMPLICIT_OBJECT_REGEX}" |
| ${compute_artifacts} LANGUAGE ${lang}) |
| message(CONFIGURE_LOG |
| "Parsed ${lang} implicit link information:\n${log}\n\n") |
| # for VS IDE Intel Fortran we have to figure out the |
| # implicit link path for the fortran run time using |
| # a try-compile |
| if("${lang}" MATCHES "Fortran" |
| AND "${CMAKE_GENERATOR}" MATCHES "Visual Studio") |
| message(CHECK_START "Determine Intel Fortran Compiler Implicit Link Path") |
| # Build a sample project which reports symbols. |
| try_compile(IFORT_LIB_PATH_COMPILED |
| PROJECT IntelFortranImplicit |
| SOURCE_DIR ${CMAKE_ROOT}/Modules/IntelVSImplicitPath |
| BINARY_DIR ${CMAKE_BINARY_DIR}/CMakeFiles/IntelVSImplicitPath |
| CMAKE_FLAGS |
| "-DCMAKE_Fortran_FLAGS:STRING=${CMAKE_Fortran_FLAGS}" |
| OUTPUT_VARIABLE _output) |
| file(WRITE |
| "${CMAKE_BINARY_DIR}/CMakeFiles/IntelVSImplicitPath/output.txt" |
| "${_output}") |
| include(${CMAKE_BINARY_DIR}/CMakeFiles/IntelVSImplicitPath/output.cmake OPTIONAL) |
| message(CHECK_PASS "done") |
| endif() |
| endif() |
| |
| # Implicit link libraries cannot be used explicitly for multiple |
| # OS X architectures, so we skip it. |
| if(DEFINED CMAKE_OSX_ARCHITECTURES) |
| if("${CMAKE_OSX_ARCHITECTURES}" MATCHES ";") |
| set(implicit_libs "") |
| endif() |
| endif() |
| |
| if(DEFINED ENV{CMAKE_${lang}_IMPLICIT_LINK_DIRECTORIES_EXCLUDE}) |
| list(REMOVE_ITEM implicit_dirs $ENV{CMAKE_${lang}_IMPLICIT_LINK_DIRECTORIES_EXCLUDE}) |
| endif() |
| |
| set(CMAKE_${lang}_COMPILER_LINKER "${linker_tool}" PARENT_SCOPE) |
| cmake_determine_linker_id(${lang} "${linker_tool}") |
| set(CMAKE_${lang}_COMPILER_LINKER_ID "${CMAKE_${lang}_COMPILER_LINKER_ID}" PARENT_SCOPE) |
| set(CMAKE_${lang}_COMPILER_LINKER_VERSION ${CMAKE_${lang}_COMPILER_LINKER_VERSION} PARENT_SCOPE) |
| set(CMAKE_${lang}_COMPILER_LINKER_FRONTEND_VARIANT ${CMAKE_${lang}_COMPILER_LINKER_FRONTEND_VARIANT} PARENT_SCOPE) |
| |
| set(CMAKE_${lang}_IMPLICIT_LINK_LIBRARIES "${implicit_libs}" PARENT_SCOPE) |
| set(CMAKE_${lang}_IMPLICIT_LINK_DIRECTORIES "${implicit_dirs}" PARENT_SCOPE) |
| set(CMAKE_${lang}_IMPLICIT_LINK_FRAMEWORK_DIRECTORIES "${implicit_fwks}" PARENT_SCOPE) |
| |
| cmake_parse_library_architecture(${lang} "${implicit_dirs}" "${implicit_objs}" architecture_flag) |
| if(architecture_flag) |
| set(CMAKE_${lang}_LIBRARY_ARCHITECTURE "${architecture_flag}" PARENT_SCOPE) |
| endif() |
| |
| else() |
| message(CHECK_FAIL "failed") |
| endif() |
| endif() |
| endfunction() |