链接到共享库时,不会导出静态库中的符号

时间:2020-04-16 22:01:02

标签: c cmake clang

共享库的代码是模块化的,由几个独立的单元组成。每个单元都内置在静态库中。


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)
    

1 个答案:

答案 0 :(得分:2)

您需要将--whole-archive选项传递给链接器。 在CMake中,您可以执行以下操作。

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)
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}}。

CMakeLists.txt

unit2