Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >动态库依赖关系_查看运行的动态库

动态库依赖关系_查看运行的动态库

作者头像
全栈程序员站长
发布于 2022-09-27 03:38:46
发布于 2022-09-27 03:38:46
2.3K00
代码可运行
举报
运行总次数:0
代码可运行

大家好,又见面了,我是你们的朋友全栈君。

1 前言

这两天在编写一个插件系统Demo的时候,发现了个很奇怪的问题:插件加载器中已经链接了ld库,但是应用程序在链接插件加载器的时候,却还需要显式的来链接ld库。否则就会报:DSO missing from command line。这个报错翻译过来就是没有在命令行中指定该动态库。 这个报错就很搞事了,你说你明明知道需要哪个库,为什么不直接帮我链接呢,非得我显示的在命令行中指定呢?

2 现象描述

问题可以简单描述为:当链接可执行文件时,依赖于libA.so,而libA.so又依赖于libB.so,而且可执行文件中还直接调用了libB.so中的函数,那么此时链接就会出现错误。

2.1 问题发生的前置条件

  • libA.so在编译过程中显式的链接了libB.so
  • 可执行文件中使用了libB.so的函数
  • binuntils版本 ≥ 2.22

2.2 Talk is cheap. Show me the code

话不多说,先看看可以复现改问题的代码吧

libB.so的源码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>

int funB1(){ 
   

    printf("in funB1");

    return 0;
}

int funB2(){ 
   

    printf("in funB2");

    return 0;
}

这里面有两个函数:funB1funB2。 其中funB1函数会被libA调用,而funB2会被可执行文件调用。

编译libB.so:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc libB.cpp -fPIC -shared -o libB.so

libA.so的源码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#include <stdio.h>

int funB1();

int funA1(){ 
   

    printf("in funA1 \n");

    funB1();

    return 0;
}

该库中只有一个函数funA1,该函数在内部调用了libB中的funB1函数。且该函数会被可执行文件调用。

编译libA.so:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc libA.cpp -fPIC -shared -o libA.so -Wl,-rpath=./ -L./ -lB

main.cpp的源码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int funA1();
int funB2();

int main(){ 
   

    funA1();
    funB2();

    return 0;
}

编译main.cpp:(复现错误的编译方法)

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
gcc main.cpp -L./ -lA

当我们按照上面的指令编译main.cpp的时候,便报错了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
/usr/bin/ld: /tmp/ccDQXTKy.o: undefined reference to symbol '_Z5funB2v'
.//libB.so: error adding symbols: DSO missing from command line
collect2: error: ld returned 1 exit status

(问号.jpg)这,这GCC不是搞事吗,你明明知道我需要连接libB.so为啥就不帮我链接上去呢?难道我libA.so没有指明要使用libB.so?我们使用下面的指令来看一下

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ ldd libA.so

得到如下信息:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
	linux-vdso.so.1 =>  (0x00007ffd09def000)
	libB.so => ./libB.so (0x00007fc513d7d000)
	libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fc5139b3000)
	/lib64/ld-linux-x86-64.so.2 (0x00007fc514181000)

明明libA.so已经显式的指明我要依赖libB.so了,那为啥在编译main.cpp的时候链接了libA.so,GCC却还要我们显式的链接libB.so呢?

3 答案

答案很简单,那就是GCC就是想要你显式链接呗。(你是编译器,你牛好吧。)那这是为啥呢? 官方一点的答案就是,自从binutils 2.22版本以后,如果你在程序中使用了你依赖的动态库所依赖的动态库中的函数时,你就必须显式的指定你依赖的动态库所依赖的动态库。

说那么多,我们更想知道的是,通过修改什么参数可以解决这个问题呢?因为你可能不想在编译程序的时候要把动态库所依赖的所有动态库都显示链接一遍。

4 究极答案

实际上,这是binutils在2.22版本以后,默认把--no-copy-dt-needed-entries这个选项打开了。当打开了这个选项的时候,编译器在链接的时候是不会递归的去获取依赖动态库的依赖项的,于是就会出现上述的问题。关于该配置项的详细说明如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
   --copy-dt-needed-entries
   --no-copy-dt-needed-entries
       This option affects the treatment of dynamic libraries referred to by DT_NEEDED tags inside ELF dynamic libraries mentioned on the command line.  Normally the linker won't add a DT_NEEDED
       tag to the output binary for each library mentioned in a DT_NEEDED tag in an input dynamic library.  With --copy-dt-needed-entries specified on the command line however any dynamic
       libraries that follow it will have their DT_NEEDED entries added.  The default behaviour can be restored with --no-copy-dt-needed-entries.

       This option also has an effect on the resolution of symbols in dynamic libraries.  With --copy-dt-needed-entries dynamic libraries mentioned on the command line will be recursively
       searched, following their DT_NEEDED tags to other libraries, in order to resolve symbols required by the output binary.  With the default setting however the searching of dynamic
       libraries that follow it will stop with the dynamic library itself.  No DT_NEEDED links will be traversed to resolve symbols.

大概意思就是,跟在--no-copy-dt-needed-entries它后面的库都不会遍历其依赖项,使用--copy-dt-needed-entries则相反。也就是使用下面的指令来编译mian.cpp就可以避免该问题了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ gcc main.cpp -L./ -Wl,--copy-dt-needed-entries -lA

题外话

Linux的ELF文件中,如果依赖于其他的动态库,那么改ELF文件会存在一个.dynamic的段,这个段里面会记录其依赖的动态库信息,其标志位为DT_NEEDED。

5 参考文档

1,DSO missing from command line原因及解决办法:https://segmentfault.com/a/1190000002462705 2,折腾gcc/g++链接时.o文件及库的顺序问题: https://www.cnblogs.com/OCaml/archive/2012/06/18/2554086.html#sec-1-4-1

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/188583.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
静态链接库和动态链接库的区别
Linux下得库有动态与静态两种,动态通常用.so为后缀,静态用.a为后缀。面对比一下两者:
狼啸风云
2019/11/03
8.7K0
CMake和静态库顺序
C/C++程序的许多同学被静态库的依赖折腾,因为默认情况下要求被依赖的库放在依赖它的库后面,当一个程序或共享库依赖的静态库较多时,可能会陷入解决链接问题的坑中。如果对静态库不熟悉,需要结构nm等工具来解决顺序问题。
一见
2018/12/25
6.9K0
【图片+代码】:Linux 动态链接过程中的【重定位】底层原理
在上一篇文章中,我们一起学习了Linux系统中 GCC编译器在编译可执行程序时,静态链接过程中是如何进行符号重定位的。
IOT物联网小镇
2022/04/06
2.8K0
【图片+代码】:Linux 动态链接过程中的【重定位】底层原理
链接选项rpath,容易被忽略的一个功能
最近在做一个项目的客户端的SDK,实现为一个动态链接库,其依赖于其他的若干个动态库文件。那么在利用SDK开发应用程序的时候,编译的时候除了链接SDK库本身,还得链接SDK库所依赖的库,这样编译命令里面得一一的添加这些库。
河边一枝柳
2021/08/06
9240
[Bazel]自定义规则实现将多个静态库合并为一个动态库或静态库
为了实现如标题所述的将多个静态库合并为一个动态库,内置的 Bazel 规则是没有这个功能的,Bazel C/C++ 相关的内置规则有:
别打名名
2020/07/28
5.5K0
Linux下so动态库一些不为人知的秘密
Linux 下有动态库和静态库,动态库以.so为扩展名,静态库以.a为扩展名。二者都使用广泛。本文主要讲动态库方面知识。
刘盼
2018/07/26
4.2K0
Linux下so动态库一些不为人知的秘密
Linux 动态库 undefined symbol 原因定位与解决方法
在使用动态库开发部署时,遇到最多的问题可能就是 undefined symbol 了,导致这个出现这个问题的原因有多种多样,快速找到原因,采用对应的方法解决是本文写作的目的。
自学气象人
2023/06/20
8.1K0
Linux 动态库 undefined symbol 原因定位与解决方法
Tool之ld
ld - GNU的Linker,通常是编译程序的最后一步 记录一下手册,以备不时之需 -> ldpentium -v GNU ld (Wind River VxWorks G++ 4.3-386) 2.19.51.20090709 -> ldpentium --help Usage: ldpentium [options] file... Options: -a KEYWORD Shared library control for HP/UX compatibili
Taishan3721
2023/02/20
1K0
Tool之ld
详解共享库的动态加载
这边文章不是一个如何引导,尽管它确实展示了如何编译和调试共享库和可执行文件。为了解动态加载的内部工作方式进行了优化。写这篇文章是为了消除我在该主题上的知识欠缺,以便成为一名更好的程序员。我希望它也能帮助您变得更好。
后场技术
2021/05/18
3.4K0
详解共享库的动态加载
咱不知道的动态链接库小细节
动态链接库(又简称动态库)是很多工程项目中不可缺少的一部分。俗称.so文件(姑且就以linux系统为例,在windows中称为dll,在mac中为的dylib),在平时的使用中我们对其察觉可能并不是很深,但其实我们玩电脑的时候无时不刻在使用动态链接库。
老潘
2023/10/19
1K0
咱不知道的动态链接库小细节
CMake使用总结
CMake意为cross-platform make,可用于管理c/c++工程。CMake解析配置文件CMakeLists.txt生成Makefile,相比直接用Makefile管理工程,CMake更灵活和简单。
bear_fish
2018/09/19
1.9K0
linux下的so、o、lo、a、la文件的区别
o: 编译的目标文件 a: 静态库,其实就是把若干o文件打了个包 so: 动态链接库(共享库)
全栈程序员站长
2022/07/20
9.3K0
Linux下动态库(.so)和静态库(.a) 的区别
动态库(共享库)的代码在可执行程序运行时才载入内存,在编译过程中仅简单的引用,因此代码体积比较小。
bear_fish
2018/09/20
17.6K0
Linux gcc编译生成静态库和共享动态库的过程
这篇文章主要通过实例演示在Linux下如何使用gcc分别编译生成静态库和动态库文件以及其它程序如何使用这个生成的静态库和动态库。
typecodes
2024/03/29
9150
Linux gcc编译生成静态库和共享动态库的过程
c++动态库和静态库的区别_静态库里面包含动态库
C++静态库与动态库
全栈程序员站长
2022/11/11
2.1K0
[082]破局Cmake中的PRIVATE,PUBLIC,INTERFACE
最近看了很多项目的代码,代码是用cmake编译的,由于各种库之间链接关系错综复杂,加上PRIVATE,PUBLIC,INTERFACE属性值,我在添加代码的时候总会遇到稀奇古怪的编译的问题,网上看了很多文章,写的都不是很靠谱,正好看到一个b站视频讲的不错,解决了我很多疑惑,我又有了新的疑惑,折腾了一晚上终于把这个搞明白了,分享给大家。
王小二
2023/08/16
1.4K0
[082]破局Cmake中的PRIVATE,PUBLIC,INTERFACE
再议GCC编译时的静态库依赖顺序问题
使用上面的Makefile编译,将会遇到如下所示的“undefined reference”问题:
一见
2018/08/06
3.8K0
linux 动态库 静态库_静态库里面包含动态库
那么这三个时间有什么作用呢? 我们在使用自动化构建工具Makefile时,如果连续make会发现:
全栈程序员站长
2022/11/10
7.7K0
linux 动态库 静态库_静态库里面包含动态库
linux 什么是SO文件
so其实就是shared object的意思。今天看了上面的博客,感觉好吃力。赶紧做个笔记记录一下。下面的内容大多都是连接中的,穿插我自己的笔记
全栈程序员站长
2022/07/22
6.1K0
货拉拉 Android 动态资源管理系统原理与实践(下)
看到移除 so文件可能有些同学会问,这不是只要在as中删除libs目录就搞定了么?这样会有几个问题
用户1907613
2022/12/12
1.2K0
货拉拉 Android 动态资源管理系统原理与实践(下)
相关推荐
静态链接库和动态链接库的区别
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验