ldd用来打印或者查看程序运行所需的共享库(访问共享对象依赖关系),常用来解决程序因缺少某个库文件而不能运行的一些问题。
1、首先ldd不是一个可执行程序,而只是一个shell脚本
2、ldd能够显示可执行模块的dependency
,其原理是通过设置一系列的环境变量。
如:
- LD_TRACE_LOADED_OBJECTS
- LD_WARN
- LD_BIND_NOW
- LD_LIBRARY_VERSION
- LD_VERBOSE
- LD_DEBUG
当LD_TRACE_LOADED_OBJECTS环境变量不为空时,任何可执行程序在运行时,它都会只显示模块的dependency
,而程序并不真正执行。
例如:
1)export LD_TRACE_LOADED_OBJECTS=1
2)再执行任何的程序,如ls等,看看程序的运行结果
ldd默认开启的环境变量是:LD_TRACE_LOADED_OBJECTS=1
其他的变量(和值)分别对应一些选项:
- -d, --data-relocs -> LD_WARN=yes
- -r, --function-relocs ->LD_WARN和LD_BIND_NOW=yes
- -u, --unused -> LD_DEBUG=“unused”
- -v, --verbose -> LD_VERBOSE=yes
LD_TRACE_LOADED_OBJECTS为必要环境变量,其他视具体情况。
撤销该环境变量,ls 即又可以恢复正常使用:
$ unset LD_TRACE_LOADED_OBJECTS
3、ldd显示可执行模块的dependency
的工作原理,其实质是通过ld-linux.so
(elf动态库的装载器)来实现的。我们知道,ld-linux.so
模块会先于executable模块程序工作,并获得控制权,因此当上述的那些环境变量被设置时,ld-linux.so
选择了显示可执行模块的dependency。
ldd命令的本质是执行了:/lib/ld-linux.so.*
刚编译后的文件可能是:/lib/ld.so。如果是libc5则是/lib/ld-linux.so.1, 而glibc2应该是/lib/ld-linux.so.2。
4、实际上可以直接执行ld-linux.so
模块,如:/lib/ld-linux.so.2 --list program
(这相当于ldd program
)
ldd可以获得的共享库文件,其实是通过读取ldconfig命令组建起来的文件(/etc/ld.so.cache
)。
默认的共享库文件搜索/lib优先于/usr/lib,而且也只有这个2个目录。如果想要加入其他路径,则需要通过ldconfig
命令配置相关文件。
一般ld-linux.so
会按照以下顺序搜索共享库:
1、DT_RPATH或DT_RUNPATH段
2、环境变量LD_LIBRARY_PATH
3、/etc/ld.so.cache文件中的路径,但如果可执行程序在连接时候添加了-z nodeflib选项,则跳过。
4、默认路径/lib和/usr/lib,但如果添加了-z nodeflib,则跳过。
5、Linux ldd 参数说明
Usage: ldd [OPTION]... FILE...
--help print this help and exit (获取指令帮助信息)
--version print version information and exit (打印ldd的版本号)
-d, --data-relocs process data relocations (执行重定位和报告任何丢失的对象)
-r, --function-relocs process data and function relocations (执行数据对象和函数的重定位,并且报告任何丢失的对象和函数)
-u, --unused print unused direct dependencies (打印未使用的直接依赖)
-v, --verbose print all information (详细信息模式,打印所有信息,例如包括符号的版本信息)
如果命令行中给定的库名字包含’/',这个程序的libc5版本将使用它作为库名字;否则它将在标准位置搜索库。运行一个当前目录下的共享库,加前缀"./"。
6、ldd 的使用
1)查看ls命令所需的动态库
$ which ls
2)查看libstdc++.so.6动态库依赖的包的详细信息
$ whereis libstdc++.so.6
3) 缺少依赖包时后面会显示 not found
4) 查看ldd命令版本
7、注意
- ldd不能工作在
a.out
格式的共享库上。 - ldd不能工作在一些非常老的
a.out
程序上,这些程序在支持ldd的编译器发行前已经创建。如果你在这种类型的程序上使用ldd,程序将尝试argc = 0
的运行方式,其结果不可预知。