13a4de6bf6
Remove target specific flags from static_link_order.
Move the check to the common config.tests folder.
Amends 5fb99e3860
Pick-to: 6.2
Task-number: QTBUG-93002
Task-number: QTBUG-94528
Change-Id: I1368075ec6bd1e743b2b89fd93143df38a278ec2
Reviewed-by: Alexandru Croitor <alexandru.croitor@qt.io>
251 lines
9.3 KiB
CMake
251 lines
9.3 KiB
CMake
function(__qt_internal_strip_target_directory_scope_token target out_var)
|
|
# In CMake versions earlier than CMake 3.18, a subdirectory scope id is appended to the
|
|
# target name if the target is referenced in a target_link_libraries command from a
|
|
# different directory scope than where the target was created.
|
|
# Strip it.
|
|
#
|
|
# For informational purposes, in CMake 3.18, the target name looks as follows:
|
|
# ::@(0x5604cb3f6b50);Threads::Threads;::@
|
|
# This case doesn't have to be stripped (at least for now), because when we iterate over
|
|
# link libraries, the tokens appear as separate target names.
|
|
#
|
|
# Example: Threads::Threads::@<0x5604cb3f6b50>
|
|
# Output: Threads::Threads
|
|
string(REGEX REPLACE "::@<.+>$" "" target "${target}")
|
|
set("${out_var}" "${target}" PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
# Tests if linker could resolve circular dependencies between object files and static libraries.
|
|
function(__qt_internal_static_link_order_public_test result)
|
|
# We could trust iOS linker
|
|
if(IOS)
|
|
set(QT_HAVE_LINK_ORDER_MATTERS "FALSE" CACHE BOOL "Link order matters")
|
|
endif()
|
|
|
|
if(DEFINED QT_HAVE_LINK_ORDER_MATTERS)
|
|
set(${result} "${QT_HAVE_LINK_ORDER_MATTERS}" PARENT_SCOPE)
|
|
return()
|
|
endif()
|
|
|
|
if(EXISTS "${QT_CMAKE_DIR}")
|
|
set(test_source_basedir "${QT_CMAKE_DIR}/..")
|
|
else()
|
|
set(test_source_basedir "${_qt_cmake_dir}/${QT_CMAKE_EXPORT_NAMESPACE}")
|
|
endif()
|
|
|
|
try_compile(${result}
|
|
"${CMAKE_CURRENT_BINARY_DIR}/config.tests/static_link_order"
|
|
"${test_source_basedir}/config.tests/static_link_order"
|
|
static_link_order_test
|
|
static_link_order_test
|
|
)
|
|
message(STATUS "Check if linker can resolve circular dependencies - ${${result}}")
|
|
|
|
# Invert the result
|
|
if(${result})
|
|
set(${result} FALSE)
|
|
else()
|
|
set(${result} TRUE)
|
|
endif()
|
|
|
|
set(QT_HAVE_LINK_ORDER_MATTERS "${${result}}" CACHE BOOL "Link order matters")
|
|
|
|
set(${result} "${${result}}" PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
# Sets _qt_link_order_matters flag for the target.
|
|
function(__qt_internal_set_link_order_matters target link_order_matters)
|
|
if(NOT TARGET ${target})
|
|
message(FATAL_ERROR "Unable to set _qt_link_order_matters flag. ${target} is not a target.")
|
|
endif()
|
|
|
|
get_target_property(aliased_target ${target} ALIASED_TARGET)
|
|
if(aliased_target)
|
|
set(target "${aliased_target}")
|
|
endif()
|
|
|
|
if(link_order_matters)
|
|
set(link_order_matters TRUE)
|
|
else()
|
|
set(link_order_matters FALSE)
|
|
endif()
|
|
set_target_properties(${target} PROPERTIES _qt_link_order_matters "${link_order_matters}")
|
|
endfunction()
|
|
|
|
# Function combines __qt_internal_static_link_order_public_test and
|
|
# __qt_internal_set_link_order_matters calls on Qt::Platform target.
|
|
function(__qt_internal_check_link_order_matters)
|
|
__qt_internal_static_link_order_public_test(
|
|
link_order_matters
|
|
)
|
|
__qt_internal_set_link_order_matters(
|
|
${QT_CMAKE_EXPORT_NAMESPACE}::Platform "${link_order_matters}"
|
|
)
|
|
|
|
if("${ARGC}" GREATER "0" AND NOT ARGV0 STREQUAL "")
|
|
set(${ARGV0} ${link_order_matters} PARENT_SCOPE)
|
|
endif()
|
|
endfunction()
|
|
|
|
function(__qt_internal_process_dependency_resource_objects target)
|
|
# The CMake versions greater than 3.21 take care about the resource object files order in a
|
|
# linker line, it's expected that all object files are located at the beginning of the linker
|
|
# line.
|
|
# So circular dependencies between static libraries and object files are resolved and no need
|
|
# to call the finalizer code.
|
|
# TODO: This check is added before the actual release of CMake 3.21. So need to confirm that the
|
|
# target version meets the expectations.
|
|
if(CMAKE_VERSION VERSION_GREATER_EQUAL 3.21)
|
|
return()
|
|
endif()
|
|
get_target_property(processed ${target} _qt_resource_object_finalizers_processed)
|
|
if(processed)
|
|
return()
|
|
endif()
|
|
set_target_properties(${target} PROPERTIES _qt_resource_object_finalizers_processed TRUE)
|
|
|
|
__qt_internal_check_finalizer_mode(${target}
|
|
use_finalizer_mode
|
|
resource_objects
|
|
)
|
|
|
|
if(NOT use_finalizer_mode)
|
|
return()
|
|
endif()
|
|
|
|
__qt_internal_collect_dependency_resource_objects(${target} resource_objects)
|
|
target_sources(${target} PRIVATE "${resource_objects}")
|
|
endfunction()
|
|
|
|
function(__qt_internal_collect_dependency_resource_objects target out_var)
|
|
set_property(GLOBAL PROPERTY _qt_resource_processed_targets "")
|
|
|
|
__qt_internal_collect_resource_objects_recursively(resource_targets ${target} ${target})
|
|
|
|
# Collect resource objects of plugins and plugin dependencies.
|
|
__qt_internal_collect_plugin_targets_from_dependencies(${target} plugin_targets)
|
|
__qt_internal_collect_dependency_plugin_resource_objects(${target}
|
|
"${plugin_targets}"
|
|
plugin_resource_objects
|
|
)
|
|
|
|
set_property(GLOBAL PROPERTY _qt_resource_processed_targets "")
|
|
|
|
list(REMOVE_DUPLICATES resource_targets)
|
|
set(resource_objects "")
|
|
foreach(dep IN LISTS resource_targets)
|
|
list(PREPEND resource_objects "$<TARGET_OBJECTS:${dep}>")
|
|
endforeach()
|
|
|
|
set(${out_var} "${plugin_resource_objects};${resource_objects}" PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
function(__qt_internal_collect_dependency_plugin_resource_objects target plugin_targets out_var)
|
|
set(plugin_resource_objects "")
|
|
foreach(plugin_target IN LISTS plugin_targets)
|
|
__qt_internal_collect_resource_objects_recursively(plugin_resource_targets
|
|
"${QT_CMAKE_EXPORT_NAMESPACE}::${plugin_target}"
|
|
${target}
|
|
)
|
|
__qt_internal_get_static_plugin_condition_genex("${plugin_target}" plugin_condition)
|
|
|
|
foreach(plugin_resource_target IN LISTS plugin_resource_targets)
|
|
list(APPEND plugin_resource_objects
|
|
"$<${plugin_condition}:$<TARGET_OBJECTS:${plugin_resource_target}>>"
|
|
)
|
|
endforeach()
|
|
endforeach()
|
|
set(${out_var} "${plugin_resource_objects}" PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
function(__qt_internal_collect_resource_objects_recursively out_var target initial_target)
|
|
get_property(resource_processed_targets GLOBAL PROPERTY _qt_resource_processed_targets)
|
|
|
|
set(interface_libs "")
|
|
set(libs "")
|
|
if(NOT "${target}" STREQUAL "${initial_target}")
|
|
get_target_property(interface_libs ${target} INTERFACE_LINK_LIBRARIES)
|
|
endif()
|
|
get_target_property(type ${target} TYPE)
|
|
if(NOT type STREQUAL "INTERFACE_LIBRARY")
|
|
get_target_property(libs ${target} LINK_LIBRARIES)
|
|
endif()
|
|
|
|
set(resource_targets "")
|
|
foreach(lib IN LISTS libs interface_libs)
|
|
if(TARGET ${lib})
|
|
get_target_property(aliased_target ${lib} ALIASED_TARGET)
|
|
if(aliased_target)
|
|
set(lib ${aliased_target})
|
|
endif()
|
|
|
|
if(${lib} IN_LIST resource_processed_targets)
|
|
continue()
|
|
else()
|
|
list(APPEND resource_processed_targets ${lib})
|
|
set_property(GLOBAL APPEND PROPERTY _qt_resource_processed_targets ${lib})
|
|
endif()
|
|
|
|
get_target_property(is_qt_resource ${lib} _is_qt_resource_target)
|
|
if(is_qt_resource)
|
|
list(APPEND resource_targets ${lib})
|
|
else()
|
|
__qt_internal_collect_resource_objects_recursively(next_level_resources
|
|
${lib}
|
|
${initial_target}
|
|
)
|
|
list(APPEND resource_targets ${next_level_resources})
|
|
endif()
|
|
endif()
|
|
endforeach()
|
|
set(${out_var} "${resource_targets}" PARENT_SCOPE)
|
|
endfunction()
|
|
|
|
function(__qt_internal_promote_target_to_global target)
|
|
get_property(is_global TARGET ${target} PROPERTY IMPORTED_GLOBAL)
|
|
if(NOT is_global)
|
|
message(DEBUG "Promoting target to global: '${target}'")
|
|
set_property(TARGET ${target} PROPERTY IMPORTED_GLOBAL TRUE)
|
|
endif()
|
|
endfunction()
|
|
|
|
function(__qt_internal_promote_target_to_global_checked target)
|
|
# With CMake version 3.21 we use a different mechanism that allows us to promote all targets
|
|
# within a scope.
|
|
if(QT_PROMOTE_TO_GLOBAL_TARGETS AND CMAKE_VERSION VERSION_LESS 3.21)
|
|
__qt_internal_promote_target_to_global(${target})
|
|
endif()
|
|
endfunction()
|
|
|
|
function(__qt_internal_promote_targets_in_dir_scope_to_global)
|
|
# IMPORTED_TARGETS got added in 3.21.
|
|
if(CMAKE_VERSION VERSION_LESS 3.21)
|
|
return()
|
|
endif()
|
|
|
|
get_directory_property(targets IMPORTED_TARGETS)
|
|
foreach(target IN LISTS targets)
|
|
__qt_internal_promote_target_to_global(${target})
|
|
endforeach()
|
|
endfunction()
|
|
|
|
function(__qt_internal_promote_targets_in_dir_scope_to_global_checked)
|
|
if(QT_PROMOTE_TO_GLOBAL_TARGETS)
|
|
__qt_internal_promote_targets_in_dir_scope_to_global()
|
|
endif()
|
|
endfunction()
|
|
|
|
# This function ends up being called multiple times as part of a find_package(Qt6Foo) call,
|
|
# due sub-packages depending on the Qt6 package. Ensure the finalizer is ran only once per
|
|
# directory scope.
|
|
function(__qt_internal_defer_promote_targets_in_dir_scope_to_global)
|
|
get_directory_property(is_deferred _qt_promote_targets_is_deferred)
|
|
if(NOT is_deferred)
|
|
set_property(DIRECTORY PROPERTY _qt_promote_targets_is_deferred TRUE)
|
|
|
|
if(CMAKE_VERSION VERSION_GREATER_EQUAL 3.19)
|
|
cmake_language(DEFER CALL __qt_internal_promote_targets_in_dir_scope_to_global_checked)
|
|
endif()
|
|
endif()
|
|
endfunction()
|