从静态库构建共享库

Building shared library from static libraries

本文关键字:共享 构建 静态      更新时间:2023-10-16

我已经编写了一些代码作为boost::asio的包装器,我正在寻找一种方法将其打包为共享对象(.so(,供我正在处理的一些应用程序使用,但是我想从boost库中删除所有依赖项(主要是因为它们是自定义编译的,并且可能会导致与默认的预构建boost库发生冲突(。我正在将我的代码链接到这些特殊 boost 库的静态版本,但是,链接器提供了以下错误:

g++ -D_GLIBCXX_USE_CXX11_ABI=0 -I/path_to_boost_headers -Wall -fPIC -o build/obj.o -c include/source.cpp 
g++ -D_GLIBCXX_USE_CXX11_ABI=0 -shared -Wl,-soname,libobj.so.1 -o lib/libobj.so.1 build/obj.o -lc /project_path/lib/libboost_serialization.a /project_path/lib/libboost_wserialization.a /project_path/lib/libboost_system.a
/usr/bin/ld: /home/joao/Work/ASBG/code/cpp/extra/socket/lib/libboost_system.a(error_code.o): relocation R_X86_64_PC32 against symbol `_ZN5boost6system16generic_categoryEv' can not be used when making a shared object; recompile with -fPIC
/usr/bin/ld: final link failed: Bad value
collect2: error: ld returned 1 exit status

如果我没记错的话,默认情况下,boost 库是使用-fPIC编译的。即便如此,我在编译 boost 时添加了标志并替换了库:相同的输出。

尝试在静态库周围放置-Wl,--whole-archive-Wl,--no-whole-archive

未经测试,但这样的事情可以工作:

g++ -D_GLIBCXX_USE_CXX11_ABI=0 -shared -Wl,-soname,libobj.so.1 
-o lib/libobj.so.1 build/obj.o -lc 
-Wl,--whole-archive 
/project_path/lib/libboost_serialization.a 
/project_path/lib/libboost_wserialization.a 
/project_path/lib/libboost_system.a 
-Wl,--no-whole-archive

笔记:

  • --whole-archive选项强制链接器查看所有符号,而不仅仅是未解析的符号。在Linux上,有弱符号和强符号的概念。我在这个答案中写了一个解释。
  • 我认为将静态库链接到共享库?与您的问题非常相似。

(另请注意,混合动态库和共享库可能不是一个好主意。您最终可能会遇到综合缺点,这与简化的想法背道而驰。但是,我不知道全貌。只是一个意见。