linux动态链接共享库(如何解决应用程序跑不起来not found等问题)

        所谓linux中的Share Libraries和Dynamic linking扮演的角色类似Windows 中的dll文件一样。现在的OS操作系统中,大部分的程序都是动态链接的,就是说,很多程序在编译成可执行程序时,会共享一些库。这样会带来至少一个好处:应用程序可以减小自己的体积,对于各种各样的应用程序中的OS来说,可以大大减少很多存储空间了。
        Linux中存在两种库类型。 
1. 一种称之为静态库。以.a 后缀。这种库本身在编译成目标文件的时候是和应用程序链接在一起的,所以编译出来的应用程序相对较大。
2. 第二种称之为动态链接共享库。 以.so为后缀。虽然这种库形式只有一种,但是在使用中可以有两种形式:
a. 运行(runtime)时动态链接,共享库在编译器编译时被链接,但并没有包括在应用程序中(目标程序中),只在运行时链接。而且是必要的,如果没有共享库,程序将跑不起来。
b. 运行中使用动态链接装载器中的函数来进行动态的加载、卸载或者链接。如浏览器使用中装载的各种插件。


        我们知道Linux中的执行文件使用的一种叫做ELF格式来表示。这是一种特殊的二进制格式式。 ELF = Executable and Linkable Format,可执行链接格式,是UNIX系统实验室(USL)作为应用程序二进制接口(Application Binary Interface,ABI)而开发和发布的。扩展名为elf。工具接口标准委员会(TIS)选择了正在发展中的ELF标准作为工作在32位INTEL体系上不同操作系统之间可移植的二进制文件格式。
在移植的linux中经常碰到自己移植的文件系统上经常会碰到,应用程序跑不起来,通常出现类似这样的错误:
-/bin/sh: xxx not found. 
或者
xxx: error while loading shared libraries: libc.so.6: cannot open shared object file: No such file or directory
如果文件系统已经移植好,能正常使用其他应用程序。那显然是应用程序缺少了相关的库了。一句话就是共享库或动态链接出现了问题。


        我们知道一个程序要想在内存中运行,除了编译之外还要经过链接和装入这两个步骤。当然linux中动态链接也是经过这三个过程。Linux 使用这个ld-linux.so*(我的平台是使用ld-linux.so6)中的来装载(其实这只是一个链接)其他库。所以这个库必须放在linux中/lib下。对于其他,通常我们共享库放在/lib这个路径下,而且也是系统默认的搜索路径。
Linux共享库的搜索路径先后顺序:
1、编译目标代码时指定的动态库搜索路径:在编译的时候指定-Wl,-rpath=路径
2、环境变量LD_LIBRARY_PATH指定的动态库搜索路径
3、配置文件/etc/ld.so.conf中指定的动态库搜索路径
4、默认的动态库搜索路径/lib
5、默认的动态库搜索路径 /usr/lib


解决的思路,首先要却定自己编译的程序是否编译于目标平台的以及相关的信息然后设置好动链接共享库环境。结合例子helloarm程序。
wsn@wsn-Inspiron-1427:~/mylinux$ file helloarm
helloarm: ELF 32-bit LSB executable, ARM, version 1 (SYSV), dynamically linked (uses shared libs), for GNU/Linux 2.6.14, not stripped
wsn@wsn-Inspiron-1427:~/mylinux$ readelf -a helloarm
。。。。
Version needs section '.gnu.version_r' contains 2 entries:
 Addr: 0x00000000000082c8  Offset: 0x0002c8  Link: 5 (.dynstr)
  000000: Version: 1  File: libgcc_s.so.1  Cnt: 1
  0x0010:   Name: GCC_3.5  Flags: none  Version: 3
  0x0020: Version: 1  File: libc.so.6  Cnt: 1
  0x0030:   Name: GLIBC_2.4  Flags: none  Version: 2


Notes at offset 0x00000148 with length 0x00000020:
  Owner Data sizeDescription
  GNU 0x00000010NT_GNU_ABI_TAG (ABI version tag)
Attribute Section: aeabi
File Attributes
  Tag_CPU_name: "4T"
  Tag_CPU_arch: v4T
  Tag_ARM_ISA_use: Yes
  Tag_THUMB_ISA_use: Thumb-1
  Tag_ABI_PCS_wchar_t: 4
  Tag_ABI_FP_denormal: Needed
  Tag_ABI_FP_exceptions: Needed
  Tag_ABI_FP_number_model: IEEE 754
  Tag_ABI_align_needed: 8-byte
  Tag_ABI_enum_size: int

        从以上可以看出helloarm程序的目标平台是ARMv4T,而且已经使用动态链接共享库。
查看应用程序所需要的库:
wsn@wsn-Inspiron-1427:~/mylinux$ readelf -d helloarm |grep NEEDED


Dynamic section at offset 0x5a8 contains 25 entries:
  Tag        Type                         Name/Value
 0x00000001 (NEEDED)                     Shared library: [libgcc_s.so.1]
 0x00000001 (NEEDED)                     Shared library: [libc.so.6]
  …

        所需要的库文件分别是: libgcc_s.so.1和libc.so.6
        所以,最后我们将ld-linux-so6、libgcc_s.so.1、libc.so.6直接放到默认中/lib下面即可。当然也可以可以在/etc下新建ld.so.conf类似这样:
#/etc/ld.so.conf
/usr/X11R6/lib
/usr/lib
...
..
/usr/lib/sane
/usr/lib/mysql
/opt/lib



        或者通过export LD_LIBRARY_PATH=路径:$LD_LIBRARY_PATHLl来自定义。
设置好之后,你的应用程序就可以跑起来了。He~
[root@WSN /]# ls
bin         helloarm    linuxrc     proc        sys         var
dev         home        lost+found  root        tmp
etc         lib         mnt         sbin        usr
[root@WSN /]# ./helloarm
Hello arm !
I am running on GNU/linux !
Bye-Bye!
[root@WSN /]#

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值