c - 使用 CMake 生成静态可执行文件
问题描述
请耐心等待这个问题,它相当长。
TLDR:具有子目录库的 CMake 项目链接成功,但创建了一个动态可执行文件。
代码位于:https ://github.com/georcon/cmake-issue
另请注意:我已阅读所有相关问题/答案,但没有人回答这个问题。
我创建了以下最小示例:
创建一个静态链接的可执行文件(正常工作)
(Git 标签:SimpleExecutable)
主程序
#include <uuid/uuid.h>
#include <stdio.h>
int main(){
uuid_t some_uuid;
char uuid_str[40];
uuid_generate(some_uuid);
uuid_unparse(some_uuid, uuid_str);
printf("UUID: %s\n", uuid_str);
return 0;
}
CMakeLists.txt
project(cmake-issue VERSION 1.0 DESCRIPTION "Static target issue" LANGUAGES C)
add_executable(main-dynamic main.c)
target_link_libraries(main-dynamic uuid)
add_executable(main-static main.c)
target_link_libraries(main-static uuid -static)
结果
ldd main-dynamic
linux-vdso.so.1 (0x00007ffc406bb000)
libuuid.so.1 => /lib/x86_64-linux-gnu/libuuid.so.1 (0x00007f76781cd000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f7677fdb000)
/lib64/ld-linux-x86-64.so.2 (0x00007f76781eb000)
ldd main-static
not a dynamic executable
使用库创建静态可执行文件
(Git 标记:ExecutableWithLibrary)
库/lib.h
#ifndef LIB_H
#define LIB_H
void PrintUUID();
#endif //LIB_H
库/lib.c
#include <uuid/uuid.h>
#include <stdio.h>
void PrintUUID(){
uuid_t some_uuid;
char uuid_str[40];
uuid_generate(some_uuid);
uuid_unparse(some_uuid, uuid_str);
printf("UUID: %s\n", uuid_str);
}
lib/CMakeLists.txt
cmake_minimum_required(VERSION 3.13)
project(testlibrary VERSION 1.0 DESCRIPTION "Static target issue - Library" LANGUAGES C)
add_library(testlib lib.c)
target_link_libraries(testlib uuid)
主程序
#include "lib/lib.h"
int main(){
PrintUUID();
return 0;
}
CMakeLists.txt
cmake_minimum_required(VERSION 3.13)
project(cmake-issue VERSION 1.0 DESCRIPTION "Static target issue" LANGUAGES C)
add_subdirectory(lib ./bin)
link_directories(./lib/ ./bin)
add_executable(main-dynamic main.c)
add_dependencies(main-dynamic testlib)
target_link_libraries(main-dynamic libtestlib.a uuid -static)
link_libraries("-static")
add_executable(main-static main.c)
target_link_libraries(main-static PUBLIC "-static" libtestlib.a uuid)
add_dependencies(main-static testlib)
#target_link_libraries(main-static libtestlib.a uuid -static)
结果
ldd main-static
linux-vdso.so.1 (0x00007ffe6b485000)
libuuid.so.1 => /lib/x86_64-linux-gnu/libuuid.so.1 (0x00007fc67edd3000)
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fc67ebe1000)
/lib/ld64.so.1 => /lib64/ld-linux-x86-64.so.2 (0x00007fc67edec000)
查看链接器命令:
~/cmake_issue$ cat CMakeFiles/main-static.dir/link.txt /usr/bin/cc
CMakeFiles/main-static.dir/main.co -o main-static
-L/home/georcon/cmake_issue/./lib -L/home/georcon/cmake_issue/./bin -Wl,-rpath,/home/georcon/cmake_issue/./lib:/home/georcon/cmake_issue/./bin -static -static -Wl,-Bstatic -ltestlib -Wl , -Bdynamic -luuid
在这种情况下,为什么 CMake 不生成静态链接的可执行文件?
解决方案
长话短说:您需要告诉 CMake 您更喜欢与库进行静态链接。这是通过设置属性来完成的LINK_SEARCH_START_STATIC
。您还需要告诉 CMake 不要在库列表末尾重置静态链接。这是通过设置属性来完成的LINK_SEARCH_END_STATIC
:
set_target_properties(main-static PROPERTIES
LINK_SEARCH_START_STATIC ON
LINK_SEARCH_END_STATIC ON
)
另请参阅该问题:CMake 和静态链接。
到底是怎么回事
实际上,链接器选项-static
不仅会禁用 PIE,还会影响命令中列出的其他库...除非-dynamic
指定。
CMake 有一个关于“默认链接类型”的概念,它适用于uuid
CMake 无法推断其类型的每个库(在您的情况下)。此外,CMake在添加到链接器命令行的每个库之后都会维护该默认链接类型。CMake 期望用户有同样的行为,他们手动添加链接器标志。
你的第一个例子是错误的,但突然起作用了:
您添加-static
which 将当前链接类型变为static。因此,您打破了 CMake 对当前链接类型的期望。
当使用 uuid 生成链接器选项时,CMake 期望当前链接是动态的。因此,CMake 不添加-dynamic
链接器开关。
那个时候CMake的期望与现实不对应,而现实又与你的期望相对应:uuid
是静态链接的。
但第二个例子揭示了问题:
当与libtestlib.a
库链接时,CMake 完全知道这是一个静态库,因此Wl,-Bstatic
在该库之前添加了选项。但是CMake需要在每个选项之后保持默认链接,所以它-Wl,-Bdynamic
在库之后添加:
-Wl,-Bstatic -ltestlib -Wl,-Bdynamic
有了这样的选项,CMake对默认动态链接的期望与现实相对应:uuid
是动态链接的。但现在这个现实并不符合你的期望。
推荐阅读
- go - Google Dataflow - 无效的区域端点 - Go 客户端的模板上不可能设置区域
- css - 如何在 less fade() 函数中使用 css var()
- python - 如果子对象为空,如何在python3中删除父json元素
- cypress - 检查“q-input”元素是否处于错误状态的简单方法是什么?
- c - 在 Windows 上使用 CMake 构建 open62541 失败
- python - 下拉菜单 KivyMD 剂量显示
- c# - 如何修复文件创建日期
- go - tls: DialWithDialer 超时
- python - 在 django 中运行 compilemessages 后为项目中的 html 和 txt 文件创建的重复 .py 文件
- c - 找到两个连续的素数,使得它们之间的间隔大于或等于 N