同名符号在linux下发生冲突的解决方案


本文摘自PHP中文网,作者黄舟,侵删。

这篇文章主要介绍了 linux 下同名符号冲突问题解决方案的相关资料,需要的朋友可以参考下

linux 下同名符号冲突问题解决方案

最近的工作中遇到如下令人蛋疼的问题:

Linux 下有三个模块aa、bb、cc,基本情况如下:

cc 编译连接得到 cc.so 动态库,cc 中有如下接口:


1

2

3

4

5

6

cc_fun

{

……

do();//调用名为do的cc模块内部函数

……

}

bb 编译连接得到 bb.a 静态库,bb 中有如下接口:


1

2

3

4

5

6

7

8

9

bb_fun

{

……

handle = dlopen(cc.so, RTLD_LAZY);//加载cc.so

pccfun = dlsym(handle, “cc_fun”);//获取cc_fun函数指针

(*pccfun)();//调用cc_fun函数,此时应该会调用cc模块中的do()函数

do();//调用名为do的bb模块内部函数(与cc模块中的do()函数同名,实现却不相同)

……

}

aa 编译后通过 -lbb 链接选项的方式连接 bb.a 得到 aa 可执行程序,并调用 bb.a 的接口函数 bb_fun():


1

2

3

4

5

6

main

{

……

bb_fun();//调用bb_fun函数

……

}

工作中发现 aa 在运行时行为异常,总是有内存泄露和功能异常,通过定位发现问题集中在同名的 do() 函数上。通过输出打印发现程序中两次调用 do() 函数都调用到了 bb 模块中的 do() 函数,而 cc 模块中的 do() 函数从未被调用到,导致程序行为异常和内存泄露。

后经多方查证了解到因为 linux 程序中各个库中的符号表最终都会加载到程序所在的全局符号表中,此时如果有同名符号就只能调用到第一个加载进来的符号,也就是说后边加载的同名符号都会被之前的覆盖。cc 模块中的 do() 函数被 bb 模块中的 do() 函数覆盖了,所以无法被调用到。

废话不多说。。。

在试验过很多不满意的方法之后,最终的解决方法如下:

1.在 cc 的 makefile 中加入 -Wl,-Bsymbolic -Wl,--version-script,version 的连接选项,意思是用 version 文件中的脚本指定其导出哪些函数。

2.version 文件的实现如下:


1

2

3

4

5

VERS{

global:

cc_fun;

local: *;

};

意思是指定 cc 模块只导出接口函数 cc_fun,其余函数都设为 local 不做导出。

将该文件保存在 makefile 所在目录即可。

3.重新编译连接三个模块,问题解决。

以上就是同名符号在linux下发生冲突的解决方案的详细内容,更多文章请关注木庄网络博客

相关阅读 >>

如何使用deletecommand删除一条云助手命令

通过更换系统盘实现更换系统盘(非公共镜像)

Linux上不了网

如何查询云服务器 ecs 提供的实例规格族资源

常用vim操作的实例详解

Linux怎么查看文件大小

useradd命令详解

Linux怎么重置管理员密码

Linux中ssh远程登录失败怎么办

Linux如何查看软件的安装位置

更多相关阅读请进入《Linux》频道 >>



打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...