|
|
|
# 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")
|
|
|
|
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()
|