共享库的代码是模块化的,由几个独立的单元组成。每个单元都内置在静态库中。
unit1.c
#include <stdio.h>
void HelloWorld() {
printf("Hello World!\n");
}
unit2.c
#include <stdio.h>
void GoodbyeWorld() {
printf("Goodbye World!\n");
}
CMakeLists.txt
cmake_minimum_required(VERSION 3.0.2)
add_library(unit1 STATIC unit1.c)
target_compile_options(unit1 PRIVATE -fPIC)
add_library(unit2 STATIC unit2.c)
target_compile_options(unit2 PRIVATE -fPIC)
add_library(merged SHARED)
target_link_libraries(merged unit1 unit2)
set_target_properties(merged PROPERTIES LINKER_LANGUAGE C)
构建步骤:
cmake . && cmake --build .
通过libmerged.so导出的符号:
$ nm -D --defined-only libmerged.so
0000000000201020 B __bss_start
0000000000201020 D _edata
0000000000201028 B _end
00000000000005a0 T _fini
0000000000000458 T _init
Q 为什么不导出符号HelloWorld和GoodbyeWorld?如何解决?
我尝试了--version-script
,但没有成功。
CMakeLists.txt中的其他设置
set_target_properties(merged PROPERTIES LINK_FLAGS -Wl,--version-
script=merged.version)
merged.version
merged {
global: HelloWorld; GoodbyeWorld;
local: *;
};
set_target_properties(merged PROPERTIES LINK_FLAGS -Wl,-force_load,libunit1.a)
答案 0 :(得分:2)
您需要将--whole-archive
选项传递给链接器。
在CMake中,您可以执行以下操作。
cmake_minimum_required(VERSION 3.0.2)
add_library(unit1 STATIC unit1.c)
target_compile_options(unit1 PRIVATE -fPIC)
add_library(unit2 STATIC unit2.c)
target_compile_options(unit2 PRIVATE -fPIC)
add_library(merged SHARED)
set_target_properties(merged PROPERTIES LINKER_LANGUAGE C)
target_link_libraries(merged
"-Wl,--whole-archive libunit1.a libunit2.a -Wl,--no-whole-archive"
unit1 unit2
)
注意:target_link_libraries
命令还可以用于指定链接器标志,而不仅仅是库名。引号很重要,否则CMake可能会重新排列标志并删除重复项。
$ nm libmerged.so | grep " T "
000000000000065d T GoodbyeWorld
000000000000064a T HelloWorld
0000000000000670 T _fini
0000000000000520 T _init
为避免该问题,另一种选择是为OBJECT
和{创建 STATIC
而不是 unit1
个库{1}}。
unit2