CMake: Generate and use a wrapper script for stripping binaries
MinGW 11.2.0 comes with a strip.exe that strips the ".gnu_debuglink" section in binaries, a section that is needed for the separate debug information feature. binutils version 2.34 mentions the feature for the first time: https://sourceware.org/binutils/docs-2.34/binutils/strip.html#strip To ensure the debuglink section is preserved, generate a shell wrapper that calls the original strip binary with an extra option to keep the required section. To determine if the option is supported, we build a real shared library on which strip will be called with the --keep-section option. If the option is not supported, a wrapper is not generated and the stock strip binary is used. This logic only applies when targeting Linux and MinGW + a shared library Qt. For other targets, the stock strip binary is used. Developers can opt out of this logic by passing -DQT_NO_STRIP_WRAPPER=TRUE when configuring each Qt repo. Pick-to: 6.2 6.3 Fixes: QTBUG-101653 Change-Id: Idd213d48d087d3c9600c853362aebaba348cde33 Reviewed-by: Cristian Adam <cristian.adam@qt.io> Reviewed-by: Jörg Bornemann <joerg.bornemann@qt.io>
This commit is contained in:
parent
70c3dd4a54
commit
39f657032b
@ -292,9 +292,26 @@ qt_copy_or_install(DIRECTORY cmake/platforms
|
|||||||
# Install public config.tests files.
|
# Install public config.tests files.
|
||||||
qt_copy_or_install(DIRECTORY
|
qt_copy_or_install(DIRECTORY
|
||||||
"config.tests/static_link_order"
|
"config.tests/static_link_order"
|
||||||
|
"config.tests/binary_for_strip"
|
||||||
DESTINATION "${__GlobalConfig_install_dir}/config.tests"
|
DESTINATION "${__GlobalConfig_install_dir}/config.tests"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Install qt-internal-strip files.
|
||||||
|
set(__qt_internal_strip_wrappers
|
||||||
|
libexec/qt-internal-strip.in
|
||||||
|
libexec/qt-internal-strip.bat.in
|
||||||
|
)
|
||||||
|
qt_copy_or_install(PROGRAMS
|
||||||
|
${__qt_internal_strip_wrappers}
|
||||||
|
DESTINATION "${__GlobalConfig_install_dir}/libexec"
|
||||||
|
)
|
||||||
|
if(QT_WILL_INSTALL)
|
||||||
|
foreach(__qt_internal_strip_wrapper ${__qt_internal_strip_wrappers})
|
||||||
|
file(COPY "${__qt_internal_strip_wrapper}"
|
||||||
|
DESTINATION "${__GlobalConfig_build_dir}/libexec")
|
||||||
|
endforeach()
|
||||||
|
endif()
|
||||||
|
|
||||||
# Install public CMake files.
|
# Install public CMake files.
|
||||||
# The functions defined inside can be used in both public projects and while building Qt.
|
# The functions defined inside can be used in both public projects and while building Qt.
|
||||||
# Usually we put such functions into Qt6CoreMacros.cmake, but that's getting bloated.
|
# Usually we put such functions into Qt6CoreMacros.cmake, but that's getting bloated.
|
||||||
|
@ -389,6 +389,8 @@ macro(qt_build_repo_begin)
|
|||||||
|
|
||||||
qt_enable_cmake_languages()
|
qt_enable_cmake_languages()
|
||||||
|
|
||||||
|
qt_internal_generate_binary_strip_wrapper()
|
||||||
|
|
||||||
# Add global docs targets that will work both for per-repo builds, and super builds.
|
# Add global docs targets that will work both for per-repo builds, and super builds.
|
||||||
if(NOT TARGET docs)
|
if(NOT TARGET docs)
|
||||||
add_custom_target(docs)
|
add_custom_target(docs)
|
||||||
|
@ -4,6 +4,166 @@ if(CMAKE_VERSION VERSION_LESS 3.17.0)
|
|||||||
set(CMAKE_CURRENT_FUNCTION_LIST_DIR ${CMAKE_CURRENT_LIST_DIR})
|
set(CMAKE_CURRENT_FUNCTION_LIST_DIR ${CMAKE_CURRENT_LIST_DIR})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
# Builds a shared library which will have strip run on it.
|
||||||
|
function(qt_internal_try_compile_binary_for_strip binary_out_var)
|
||||||
|
# Need to find the config.tests files depending on which repo we are building.
|
||||||
|
if(EXISTS "${QT_CMAKE_DIR}")
|
||||||
|
# building qtbase
|
||||||
|
set(basedir "${QT_CMAKE_DIR}/..")
|
||||||
|
else()
|
||||||
|
# building other repo
|
||||||
|
set(basedir "${_qt_cmake_dir}/${QT_CMAKE_EXPORT_NAMESPACE}")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(config_test_dir "config.tests/binary_for_strip")
|
||||||
|
set(src_dir "${basedir}/${config_test_dir}")
|
||||||
|
|
||||||
|
# Make sure the built project files are not installed when doing an in-source build (like it
|
||||||
|
# happens in Qt's CI) by choosing a build dir that does not coincide with the installed
|
||||||
|
# source dir.
|
||||||
|
set(binary_dir "${CMAKE_CURRENT_BINARY_DIR}/${config_test_dir}_built")
|
||||||
|
|
||||||
|
set(flags "")
|
||||||
|
qt_get_platform_try_compile_vars(platform_try_compile_vars)
|
||||||
|
list(APPEND flags ${platform_try_compile_vars})
|
||||||
|
|
||||||
|
# CI passes the project dir of the Qt repository as absolute path without drive letter:
|
||||||
|
# \Users\qt\work\qt\qtbase
|
||||||
|
# Ensure that arg_PROJECT_PATH is an absolute path with drive letter:
|
||||||
|
# C:/Users/qt/work/qt/qtbase
|
||||||
|
# This works around CMake upstream issue #22534.
|
||||||
|
if(CMAKE_HOST_WIN32)
|
||||||
|
get_filename_component(src_dir "${src_dir}" REALPATH)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Build a real binary that strip can be run on.
|
||||||
|
try_compile(QT_INTERNAL_BUILT_BINARY_FOR_STRIP
|
||||||
|
"${binary_dir}"
|
||||||
|
"${src_dir}"
|
||||||
|
binary_for_strip # project name
|
||||||
|
OUTPUT_VARIABLE build_output
|
||||||
|
CMAKE_FLAGS ${flags}
|
||||||
|
)
|
||||||
|
|
||||||
|
# Retrieve the binary path from the build output.
|
||||||
|
string(REGEX REPLACE ".+###(.+)###.+" "\\1" output_binary_path "${build_output}")
|
||||||
|
|
||||||
|
if(NOT EXISTS "${output_binary_path}")
|
||||||
|
message(FATAL_ERROR "Extracted binary path for strip does not exist: ${output_binary_path}")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(${binary_out_var} "${output_binary_path}" PARENT_SCOPE)
|
||||||
|
endfunction()
|
||||||
|
|
||||||
|
# When using the MinGW 11.2.0 toolchain, cmake --install --strip as used by
|
||||||
|
# qt-cmake-private-intstall.cmake, removes the .gnu_debuglink section in binaries and thus
|
||||||
|
# breaks the separate debug info feature.
|
||||||
|
#
|
||||||
|
# Generate a wrapper shell script that passes an option to keep the debug section.
|
||||||
|
# The wrapper is used when targeting Linux or MinGW with a shared Qt build.
|
||||||
|
# The check to see if the option is supported by 'strip', is done once for every repo configured,
|
||||||
|
# because different machines might have different strip versions installed, without support for
|
||||||
|
# the option we need.
|
||||||
|
#
|
||||||
|
# Once CMake supports custom strip arguments, we can remove the part that creates a shell wrapper.
|
||||||
|
# https://gitlab.kitware.com/cmake/cmake/-/issues/23346
|
||||||
|
function(qt_internal_generate_binary_strip_wrapper)
|
||||||
|
# Return early if check was done already, if explicitly skipped, or when building a static Qt.
|
||||||
|
if(DEFINED CACHE{QT_INTERNAL_STRIP_SUPPORTS_KEEP_SECTION}
|
||||||
|
OR QT_NO_STRIP_WRAPPER
|
||||||
|
OR (NOT QT_BUILD_SHARED_LIBS)
|
||||||
|
)
|
||||||
|
return()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# To make reconfiguration more robust when QT_INTERNAL_STRIP_SUPPORTS_KEEP_SECTION is manually
|
||||||
|
# removed, make sure to always find the original strip first, by first removing the cached var
|
||||||
|
# and then finding the binary again.
|
||||||
|
unset(CMAKE_STRIP CACHE)
|
||||||
|
include(CMakeFindBinUtils)
|
||||||
|
|
||||||
|
# Target Linux and MinGW.
|
||||||
|
if((UNIX OR MINGW)
|
||||||
|
AND NOT APPLE
|
||||||
|
AND CMAKE_STRIP)
|
||||||
|
|
||||||
|
# Getting path to a binary we can run strip on.
|
||||||
|
qt_internal_try_compile_binary_for_strip(valid_binary_path)
|
||||||
|
|
||||||
|
# The strip arguments are used both for the execute_process test and also as content
|
||||||
|
# in the file created by configure_file.
|
||||||
|
set(strip_arguments "--keep-section=.gnu_debuglink")
|
||||||
|
|
||||||
|
# Check if the option is supported.
|
||||||
|
message(STATUS "Performing Test strip --keep-section")
|
||||||
|
execute_process(
|
||||||
|
COMMAND
|
||||||
|
"${CMAKE_STRIP}" ${strip_arguments} "${valid_binary_path}"
|
||||||
|
OUTPUT_VARIABLE strip_probe_output
|
||||||
|
ERROR_VARIABLE strip_probe_output
|
||||||
|
RESULT_VARIABLE strip_result_var
|
||||||
|
)
|
||||||
|
|
||||||
|
# A successful strip of a binary should have a '0' exit code.
|
||||||
|
if(NOT strip_result_var STREQUAL "0")
|
||||||
|
set(keep_section_supported FALSE)
|
||||||
|
else()
|
||||||
|
set(keep_section_supported TRUE)
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# Cache the result.
|
||||||
|
set(QT_INTERNAL_STRIP_SUPPORTS_KEEP_SECTION "${keep_section_supported}" CACHE BOOL
|
||||||
|
"strip supports --keep-section")
|
||||||
|
|
||||||
|
message(DEBUG
|
||||||
|
"qt_internal_generate_binary_strip_wrapper:\n"
|
||||||
|
"original strip: ${CMAKE_STRIP}\n"
|
||||||
|
"strip probe output: ${strip_probe_output}\n"
|
||||||
|
"strip result: ${strip_result_var}\n"
|
||||||
|
"keep section supported: ${keep_section_supported}\n"
|
||||||
|
)
|
||||||
|
message(STATUS "Performing Test strip --keep-section - ${keep_section_supported}")
|
||||||
|
|
||||||
|
# If the option is not supported, don't generate a wrapper and just use the stock binary.
|
||||||
|
if(NOT keep_section_supported)
|
||||||
|
return()
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(wrapper_extension "")
|
||||||
|
|
||||||
|
if(NOT CMAKE_HOST_UNIX)
|
||||||
|
set(wrapper_extension ".bat")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
set(script_name "qt-internal-strip")
|
||||||
|
|
||||||
|
if(EXISTS "${QT_CMAKE_DIR}")
|
||||||
|
# qtbase build-tree case
|
||||||
|
set(wrapper_in_basedir "${QT_CMAKE_DIR}/..")
|
||||||
|
else()
|
||||||
|
# other repo case
|
||||||
|
set(wrapper_in_basedir "${_qt_cmake_dir}/${QT_CMAKE_EXPORT_NAMESPACE}")
|
||||||
|
endif()
|
||||||
|
|
||||||
|
# the libexec literal is used on purpose for the source, so the file is found
|
||||||
|
# on Windows hosts.
|
||||||
|
set(wrapper_in
|
||||||
|
"${wrapper_in_basedir}/libexec/${script_name}${wrapper_extension}.in")
|
||||||
|
|
||||||
|
set(wrapper_out "${QT_BUILD_DIR}/${INSTALL_LIBEXECDIR}/${script_name}${wrapper_extension}")
|
||||||
|
|
||||||
|
set(original_strip "${CMAKE_STRIP}")
|
||||||
|
|
||||||
|
configure_file("${wrapper_in}" "${wrapper_out}" @ONLY)
|
||||||
|
|
||||||
|
# Backup the original strip path for informational purposes.
|
||||||
|
set(QT_INTERNAL_ORIGINAL_STRIP "${original_strip}" CACHE INTERNAL "Original strip binary")
|
||||||
|
|
||||||
|
# Override the strip binary to be used by CMake install target.
|
||||||
|
set(CMAKE_STRIP "${wrapper_out}" CACHE INTERNAL "Custom Qt strip wrapper")
|
||||||
|
endif()
|
||||||
|
endfunction()
|
||||||
|
|
||||||
# Enable separate debug information for the given target
|
# Enable separate debug information for the given target
|
||||||
function(qt_enable_separate_debug_info target installDestination)
|
function(qt_enable_separate_debug_info target installDestination)
|
||||||
set(flags QT_EXECUTABLE)
|
set(flags QT_EXECUTABLE)
|
||||||
|
8
config.tests/binary_for_strip/CMakeLists.txt
Normal file
8
config.tests/binary_for_strip/CMakeLists.txt
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
cmake_minimum_required(VERSION 3.16)
|
||||||
|
project(proj LANGUAGES CXX)
|
||||||
|
add_library(lib1 SHARED lib1.cpp)
|
||||||
|
|
||||||
|
add_custom_target(print_lib_path ALL
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E echo "###$<TARGET_FILE:lib1>###"
|
||||||
|
)
|
||||||
|
add_dependencies(print_lib_path lib1)
|
29
config.tests/binary_for_strip/lib1.cpp
Normal file
29
config.tests/binary_for_strip/lib1.cpp
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
/****************************************************************************
|
||||||
|
**
|
||||||
|
** Copyright (C) 2022 The Qt Company Ltd.
|
||||||
|
** Contact: https://www.qt.io/licensing/
|
||||||
|
**
|
||||||
|
** This file is part of the utils of the Qt Toolkit.
|
||||||
|
**
|
||||||
|
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
|
||||||
|
** Commercial License Usage
|
||||||
|
** Licensees holding valid commercial Qt licenses may use this file in
|
||||||
|
** accordance with the commercial license agreement provided with the
|
||||||
|
** Software or, alternatively, in accordance with the terms contained in
|
||||||
|
** a written agreement between you and The Qt Company. For licensing terms
|
||||||
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
||||||
|
** information use the contact form at https://www.qt.io/contact-us.
|
||||||
|
**
|
||||||
|
** GNU General Public License Usage
|
||||||
|
** Alternatively, this file may be used under the terms of the GNU
|
||||||
|
** General Public License version 3 as published by the Free Software
|
||||||
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
||||||
|
** included in the packaging of this file. Please review the following
|
||||||
|
** information to ensure the GNU General Public License requirements will
|
||||||
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
||||||
|
**
|
||||||
|
** $QT_END_LICENSE$
|
||||||
|
**
|
||||||
|
****************************************************************************/
|
||||||
|
|
||||||
|
int libfunc() { return 0; }
|
1
libexec/qt-internal-strip.bat.in
Normal file
1
libexec/qt-internal-strip.bat.in
Normal file
@ -0,0 +1 @@
|
|||||||
|
@original_strip@ @strip_arguments@ %*
|
4
libexec/qt-internal-strip.in
Executable file
4
libexec/qt-internal-strip.in
Executable file
@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# echo the invoked command
|
||||||
|
set -x
|
||||||
|
@original_strip@ @strip_arguments@ "$@"
|
Loading…
Reference in New Issue
Block a user