linux下动态库so文件的一些认识
  EtUHSEjq3GlD 2023年11月02日 30 0
牵扯到ELF格式,gcc编译选项待补,简单实用的说明一下,对Linux下的so文件有个实际性的认识。


1.so文件是什么?


2.怎么生成以及使用一个so动态库文件?


3.地址空间,以及线程安全.


4.库的初始化,解析:


5.使用我们自己库里的函数替换系统函数:


//-------------------------------------------------------------------------------


1.so文件是什么?


也是ELF格式文件,共享库(动态库),类似于DLL。节约资源,加快速度,代码升级简化。

知道这么多就够了,实用主义。等有了印象再研究原理。


2.怎么生成以及使用一个so动态库文件?


先写一个C文件:s.c



1. #include <stdio.h>
2. int
3. void out_msg(const char
4. {//2秒钟输出1次信息,并计数
5. for(;;) {printf("%s %d\n", m, ++count); sleep(2);}
6. }


编译:得到输出文件libs.o


gcc -fPIC -g -c s.c -o libs.o




链接:得到输出文件libs.so


gcc -g -shared -Wl,-soname,libs.so -o libs.so libs.o -lc


一个头文件:s.h




1. #ifndef _MY_SO_HEADER_
2. #define _MY_SO_HEADER_
3. void out_msg(const char
4. #endif


再来一个C文件来引用这个库中的函数:ts.c




1. #include <stdio.h>
2. #include "s.h"
3. int main(int argc, char** argv)
4. {
5. "TS Main\n");
6. "TS ");
7. //这句话可以注释掉,在第4节的时候打开就可以。
8. "TS Quit\n");
9. }


编译链接这个文件:得到输出文件ts


gcc -g ts.c -o ts -L. -ls



执行./ts,嗯:成功了。。。还差点


得到了ts:error while loading shared libraries: libs.so: cannot open shared object file: No such file or directory


系统不能找到我们自己定义的libs.so,那么告诉他,修改变量LD_LIBRARY_PATH,为了方便,写个脚本:e(文件名就叫e,懒得弄长了)


#!/bin/sh


export LD_LIBRARY_PATH=${pwd}:${LD_LIBRARY_PATH}


./ts


执行:./e &


屏幕上就开始不停有信息输出了,当然TS Quit你是看不到的,前面是个死循环,后面会用到这句




3.地址空间,以及线程安全:

如果这样:


./e &开始执行后,稍微等待一下然后再 ./e&,


这个时候屏幕信息会怎么样呢?全局变量count会怎么变化?


会是两个进程交叉输出信息,并且各自的count互不干扰,虽然他们引用了同一个so文件。


也就是说只有代码是否线程安全一说,没有代码是否是进程安全这一说法。




4.库的初始化,解析:


windows下的动态库加载,卸载都会有初始化函数以及卸载函数来完成库的初始化以及资源回收,linux当然也可以实现。


ELF文件本身执行时就会执行一个_init()函数以及_fini()函数来完成这个,我们只要把自己的函数能让系统在这个时候执行


就可以了。


修改我们前面的s.c文件:



1. #include <stdio.h>
2. void my_init(void) __attribute__((constructor)); //告诉gcc把这个函数扔到init section
3. void my_fini(void) __attribute__((destructor)); //告诉gcc把这个函数扔到fini section
4. void out_msg(const char
5. {
6. " Ok!\n");
7. }
8. int i; //仍然是个计数器
9. void my_init(void)
10. {
11. "Init ... ... %d\n", ++i);
12. }
13. void my_fini(void)
14. {
15. "Fini ... ... %d\n", ++i);
16. }


重新制作 libs.so,ts本是不用重新编译了,代码维护升级方便很多。


然后执行: ./e &


可以看到屏幕输出:(不完整信息,只是顺序一样)


Init


Main


OK


Quit


Fini


可以看到我们自己定义的初始化函数以及解析函数都被执行了,而且是在最前面以及最后面。


如果s.c中的sleep(5)没有注释掉,那么有机会:


./e&


./e&连续执行两次,那么初始化函数和解析函数也会执行两次,虽然系统只加载了一次libs.so。


如果sleep时候kill 掉后台进程,那么解析函数不会被执行。


5.使用我们自己库里的函数替换系统函数:


创建一个新的文件b.c:我们要替换系统函数malloc以及free(可以自己写个内存泄露检测工具了)




1. #include <stdio.h>
2. void* malloc(int
3. {
4. "My malloc\n");
5. return
6. }
7. void free(void* ad)
8. {
9. "My free\n");
10. }


老规矩,编译链接成一个so文件:得到libb.so


gcc -fPIC -g -c b.c -o libb.o


gcc -g -shared -Wl,-soname,libb.so -o libb.so -lc


修改s.c:重新生成libs.so




1. void
2. {
3. int
4. int*)malloc(100);
5. free(p);
6. "Stop Ok!\n");
7. }


修改脚本文件e:


#!/bin/sh


export LD_PRELOAD=${pwd}libb.so:${LD_PRELOAD}


export LD_LIBRARY_PATH=${pwd}:${LD_LIBRARY_PATH}


./ts


关键就在LD_PRELOAD上了,这个路径指定的so将在所有的so之前加载,并且符号会覆盖后面加载的so文件中的符号。如果可执行文件的权限不合适(SID),这个变量会被忽略。


执行:./e &


嗯,可以看到我们的malloc,free工作了。


暂时就想到这么多了。
【版权声明】本文内容来自摩杜云社区用户原创、第三方投稿、转载,内容版权归原作者所有。本网站的目的在于传递更多信息,不拥有版权,亦不承担相应法律责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@moduyun.com

  1. 分享:
最后一次编辑于 2023年11月08日 0

暂无评论

推荐阅读
  ltERVYe6WHLK   2023年11月02日   21   0   0 数据类型C#初始化
EtUHSEjq3GlD
最新推荐 更多