GCC 共享库在链接时的搜索位置和优选次序

时间:2022-06-23 10:52:01

假设有两个相同的共享库,一个在标准的共享库搜索目录(/lib/i386-linux-gnu), 一个在非标准目录(/home/charles/tmp):

p$ ls /home/charles/tmp/libshared.so  /lib/i386-linux-gnu/libshared.so  -l
-rwxr-xr-x 1 root    root    7706 Jan 24 14:56 /home/charles/tmp/libshared.so
-rwxrwxr-x 1 charles charles 7706 Jan 24 14:45 /lib/i386-linux-gnu/libshared.so

在/home/charles/tmp下有个测试程序main.c, 调用共享库里的函数。

用如下的命令编译:

gcc main.c  -L /home/charles/tmp   -g -o main -lshared 
用ldd看一下link的共享库:

$ ldd main
	linux-gate.so.1 =>  (0xb7776000)
	libshared.so => /lib/i386-linux-gnu/libshared.so (0xb774f000)
	libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb75a0000)
	/lib/ld-linux.so.2 (0xb7777000)
可以看出,虽然我们指定了 要使用 /home/charles/tmp下的库,但实际上用的还是标准搜索路径下的库。

换句话说,GNU ld 优选搜索标准路径下的文件,然后才是 -L指定的

现在执行 

export LD_LIBRARY_PATH=/home/charles/tmp
$ gcc main.c  -L /home/charles/tmp   -g -o main -lshared 
$ ldd main
	linux-gate.so.1 =>  (0xb7798000)
	libshared.so => /home/charles/tmp/libshared.so (0xb7792000)
	libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb75c2000)
	/lib/ld-linux.so.2 (0xb7799000)
可见, LD_LIBRARY_PATH指定的路径优选于系统标准目录。

最后尝试一下如下的方法:

gcc main.c  -wl,-rpath,/home/charles/tmp  -g -o main -lshared

或者:

gcc main.c  -Wl,-rpath /home/charles/tmp  -g -o main -lshared
$ ldd main
	linux-gate.so.1 =>  (0xb77e0000)
	libshared.so => /home/charles/tmp/libshared.so (0xb77da000)
	libc.so.6 => /lib/i386-linux-gnu/libc.so.6 (0xb760a000)
	/lib/ld-linux.so.2 (0xb77e1000)

最后,总结一下,共享库的搜索顺序(按照优先次序):

 1) LD_LIBRARY_PATH指定的目录

2) rpath指定的路径。

rpath指定的路径同时被写进了二进制文件里面。

p$ readelf  -d main | grep rpath
 0x0000000f (RPATH)                      Library rpath: [/home/charles/tmp]

3)系统标准路径 

这包括 /lib, /usr/lib, /usr/local/lib以及在/etc/ld.conf 里面指定的路径。