分享
 
 
 

Linux动态函式库解析(二)

王朝system·作者佚名  2008-05-19
窄屏简体版  字體: |||超大  

动态连结 VS 静态联结

在 Linux 中,执行档我们可以编程成静态联结以及动态连结,以下我们举一个简短的程序作为例子:

#include

int main()

{

printf(" test");

}

若我们执行 :

[root@hlchou /root]# gcc test.c -o test

所产生出来的执行档 test,预设为使用动态函式库,所以我们可以用以下的指令 :

[root@hlchou /root]# ldd test

libc.so.6 = /lib/libc.so.6 (0x40016000)

/lib/ld-linux.so.2 = /lib/ld-linux.so.2 (0x40000000)

来得知目前该执行档共用了哪些动态函式库,以我们所举的 test 执行档来说,共用了两个动态函式库,分别为 libc.so.6 与 ld-linux.so.2。我们还可以透过下面的 file 指令,来得知该执行档的相关属性,如下

[root@hlchou /root]# file test

test: ELF 32-bit LSB executable, Intel 80386, version 1, dynamically

linked (use s shared libs), not stripped

not stripped 表示这个执行档还没有透过 strip 指令来把执行时用不到的符号、以及相关除错的资讯删除,举个例子来说,目前这个test 执行档大小约为 11694 bytes

[root@hlchou /root]# ls -l test

-rwxr-xr-x 1 root root 11694 Oct 24 02:31 test

经过strip後,则变为 3004 bytes

[root@hlchou /root]# strip test

[root@hlchou /root]# ls -l test

-rwxr-xr-x 1 root root 3004 Oct 24 02:48 test

不过读者必须注意到一点,经过 strip 过的执行档,就无法透过其它的除错软件从里面取得函式在编程时所附的相关资讯,这些资讯对我们在除错软件时,可以提供不少的帮助,各位在应用上请自行注意。

相对於编程出来使用动态函式库的执行档 test,我们也可以做出静态联结的执行档 test

[root@hlchou /root]# gcc -static test.c -o test

透过指令 ldd,我们可以确定执行档 test 并没有使用到动态函式库

[root@hlchou /root]# ldd test

not a dynamic executable

再透过指令 file,可以注意到 test 目前为 statically linked,且亦尚未经过 strip

[root@hlchou /root]# file test

test: ELF 32-bit LSB executable, Intel 80386, version 1, statically

linked, not stripped

相信大夥都会好奇,使用静态联结,且又没有经过 strip 删去不必要的符号的执行档的大小会是多少,透过 ls -l来看,我们发现大小变成 932358 bytes 比起静态联结的执行档大了相当多

[root@hlchou /root]# ls -l test

-rwxr-xr-x 1 root root 932258 Oct 24 02:51 test

若再经过 strip,则档案大小变为 215364 bytes

[root@hlchou /root]# strip test

[root@hlchou /root]# ls -l test

-rwxr-xr-x 1 root root 215364 Oct 24 02:55 test

与使用动态函式库的执行档 test 比较起来,大了约 70倍 (215364/3004)。因此,整体来说,在使用的环境中使用动态函式库并且经过 strip 处理的话,可以让整体的空间较为精简。许多执行档都会用到同一组的函式库,像 libc 中的函式是每个执行档都会使用到的,若是使用动态函式库,则可以尽量减少同样的函式库内容重复存在系统中,进而达到节省空间的目的。

笔者一年前曾写过一个可以用来删去动态函式库中不必要函式的工具,针对这个只用到了 printf 的程序来产生新的 libc.so 的话,我们可以得到一个精简过的 libc.so 大小约为 219068 bytes

[root@hlchoua lib]# ls -l libc.so*

-rwxr-xr-x 1 root root 219068 Nov 2 04:47 libc.so

lrwxrwxrwx 1 root root 7 Nov 1 03:40 libc.so.6 - libc.so

与静态联结的执行档大小 215364 bytes 比较起来,若是在这个环境中使用了动态函式库的话成本约为 3004 + 219068 =222072 bytes,不过这是只有一个执行档的情况下,使用动态函式库的环境会小输给使用静态联结的环境,在一个基本的 Linux 环境中,如果大量的使用动态函式库的话,像是有 2 个以上的执行档的话,那用动态函式库的成本就大大的降低了,像如果两个执行档都只用到了 printf,那静态联结的成本为 215364 *2 =430728 bytes,而使用动态函式库的成本为3004 *2 + 219068=225076 bytes,两者相差约一倍。

很明显的,我们可以看到动态函式库在 Linux 环境中所发挥的妙用,它大幅的降低了整体环境的持有成本,提高了环境空间的利用率。

ld-linux.so.2 在 RedHat 6.1 中,我们可以在 /lib 或是 /usr/lib 目录底下找到许多系统上所安装的动态函式库,在文章的这个部分,笔者将把整个函式库大略的架构作一个说明。

其实 Linux 跟 Windows 一样,提供了一组很基本的动态函式库,在 Windows 上面我们知道 kernel32.dll 提供了其它动态函式库基本的函式呼叫,而在 Linux 上面则透过 ld-linux.so.2 提供了其它动态函式库基本的函式,在笔者电脑的 RedHat6.1 上,ld-linux.so.2 是透过 link 到 ld-2.1.2.so(这部分需视各人所使用的 glibc 版本不同而定)

-rwxr-xr-x 1 root root 368878 Jan 20 14:28 ld-2.1.2.so

lrwxrwxrwx 1 root root 11 Jan 20 14:28 ld-linux.so.2 - ld-2.1.2.so

ld-linux.so 是属於 Glibc (GNU C Library) 套件的一部分,只要是使用 Glibc 动态函式库的环境,就可以见到 ld-linux.so 的踪影。

接下来,我们透过指令 ldd 来验证出各个函式库间的阶层关系,首先如下图我们执行了 ”ldd ls”、”ldd pwd” 与 “ldd vi”,可以看出各个执行档呼叫了哪些动态函式库,像执行档 ls 呼叫了 /lib/libc.so.6 (0x40016000)与 /lib/ld-linux.so.2 (0x40000000),而括号内的数字为该函式库载入记忆体的位置,在本文的稍後,会介绍到函式库载入时的细节,到时读者会有更深入的了解。

其实我们不难发现,在 Linux 上使用动态函式库的执行档,几乎都会去呼叫 libc.so.6 与 ld-linux.so.2 这两个动态函式库,笔者过去修改 Glibc 的套件时,也了解到在 Linux 中函式库的关系,ld-linux.so.2 算是最底层的动态函式库,它本身为静态联结,主要的工作是提供基本的函式给其他的函式库,而我们最常会呼叫的 libc.so.6 则是以 ld-linux.so.2 为基础的一个架构完成的动态函式库,它几乎负责了所有我们常用的标准 C 函式库,像是我们在 Linux 下写的 Socket 程序,其中的connect()、bind()、send() .....之类的函式,都是由 libc.so.6 所提供的。

也因此,libc.so.6 的大小也是相当可观的,在 RedHat 6.1 中经过 strip 後,大小约为 1052428 bytes。

[root@hlchoua /root]# ldd /bin/ls

libc.so.6 = /lib/libc.so.6 (0x40016000)

/lib/ld-linux.so.2 = /lib/ld-linux.so.2 (0x40000000)

[root@hlchoua /root]# ldd /bin/pwd

libc.so.6 = /lib/libc.so.6 (0x40016000)

/lib/ld-linux.so.2 = /lib/ld-linux.so.2 (0x40000000)

[root@hlchoua /root]# ldd /bin/vi

libtermcap.so.2 = /lib/libtermcap.so.2 (0x40016000)

libc.so.6 = /lib/libc.so.6 (0x4001b000)

/lib/ld-linux.so.2 = /lib/ld-linux.so.2 (0x40000000)

如下,我们透过 ldd 验证 vi 所用到的动态函式库 /lib/libtermcap.so.2,它本身是呼叫了 libc.so.6 的函式所组成的。

[root@hlchoua /root]# ldd /lib/libtermcap.so.2

libc.so.6 = /lib/libc.so.6 (0x40007000)

/lib/ld-linux.so.2 = /lib/ld-linux.so.2 (0x80000000)

接下来,我们依序测试了 /lib/libc.so.6 与 /lib/ld-linux.so.2

[root@hlchoua /root]# ldd /lib/libc.so.6

/lib/ld-linux.so.2 = /lib/ld-linux.so.2 (0x40000000)

[root@hlchoua /root]# ldd /lib/ld-linux.so.2

statically linked

我们可以整理以上的结论,画成如下的一个架构图

在这个图中,我们可以清楚的明白 ld-linux.so.2 负责了最基础的函式,而 libc.so.6 再根据这些基本的函式架构了完整的 C 函式库,供其它的动态函式库或是应用程序来呼叫。

透过笔者所写的一个 ELF 工具程序(注二),我们也可以清楚的看到 libc.so.6呼叫了 ld-linux.so.2 哪些函式

[root@hlchoua /root]# /I-elf /lib/libc.so.6|more

========================================================

open_target_file:/lib/libc.so.6

==ld-linux.so.2

__register_frame_table

cfsetispeed

xdr_int32_t

utmpname

_dl_global_scope_alloc

__strcasestr

hdestroy_r

rename

__iswctype_l

__sigaddset

xdr_callmsg

pthread_setcancelstate

xdr_union

__wcstoul_internal

setttyent

strrchr

__sysv_signal ...┅(more)

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有