ld

作者: m风满楼 | 来源:发表于2018-09-21 10:35 被阅读12次
    • ld命令,称为链接编辑程序或绑定程序,组合对象文件、归档文件并导入文件到一个输出对象文件,同时解析外部引用。生成可运行的可执行对象文件。缺省情况下, ld 命令创建并输出到 a.out 文件里。
    链接方式
    归档文件
    共享对象
    导入导出
    库:
    • 库是名称以.a,也可能是以 .so 结束的文件。要指定库,可以指定一个完整或相对路径,或以 -lName 形式使用 -l(小写字母 L)标志。最后格式指示在若干目录下搜索 libName.a 文件或(使用 rtl 选项时)libName.so 文件。这些搜索包含 -L 标识指定的所有目录及标准库目录 /usr/lib 和 /lib。
    处理:
    • ld 命令以相同方式处理所有输入文件,不管是不是归档。它包含所有对象的符号表,并废弃重复现有符号的符号定义。不象 ld 命令的其他版本,不必排列归档文件的顺序,因此引用在定义之前。
    • ld 命令标志的顺序不影响如何处理,除非标志和输入对象文件、库和导入文件配合使用。标志是:-L、-f、-l(小写字母 L)、-bkeepfile 和 -bI(大写字母)。标志以下列顺序处理
    1. -L标志添加目录到搜索目录列表来定位由-l(小写字母 L)标志指定的库。按指定的顺序搜索目录。所有 -L 标志处理在处理任意-l 标志之前进行。
    2. ld 命令处理 InputFile参数,以指定顺序由-f标志指定的文件和由 -l(小写字母 L)标志指定的库。
    3. 在处理所有其他对象文件和库之后,ld 命令处理由 -bI(大写字母 i)标志指定的导入文件。如果在处理对象文件之前必须处理该文件,可以不用 -bI 参数指定一个导入文件作为输入文件。这种情况下,导入文件的第一行必须以 #!(# 和惊叹号)符号开始,导入文件和其他输入文件按第 2 步描述的处理。
    4. -bkeepfile 选项命名一个 ld 命令不在其上执行垃圾收集的输入文件。如果已制定的输入文件作为一个 InputFile 参数指定或列在由 -f 标志指定的文件里,-bkeepfile 选项不影响文件处理的顺序。否则,文件和其他输入文件一起按第 2 步描述的顺序处理。

    库的搜索路径顺序

    • 静态库链接时搜索路径的顺序: 1. ld会去找gcc/g++命令中的参数-L;2. 再找gcc的环境变量LIBRARY_PATH,它指定程序静态链接库文件搜索路径;
      export LIBRARY_PATH=$LIBRARY_PATH:data/home/billchen/lib 3. 再找默认库目录 /lib /usr/lib /usr/local/lib,这是当初compile gcc时写在程序内的。
    • 动态链接时、执行时搜索路径顺序: 1. 编译目标代码时指定的动态库搜索路径-l;
      1. 环境变量LD_LIBRARY_PATH指定动态库搜索路径,它指定程序动态链接库文件搜索路径;export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:data/home/billchen/lib 3. 配置文件/etc/ld.so.conf中指定的动态库搜索路径; 4. 默认的动态库搜索路径/lib; 5. 默认的动态库搜索路径/usr/lib。

    环境变量

    以下环境变量影响 ld 命令执行:
    项目 描述

    • LIBPATH 如果定义了 LIBPATH,使用它的值作为缺省库路径信息。否则缺省库路径信息是 /usr/lib:/lib。如果没有指定 -L 标志和 -blibpath 选项,那么缺省库路径信息写到输出文件的装入程序部分。不管指定的任意选项,当搜索命令行中指定的库时不使用 LIBPATH。
    • TMPDIR 如果输出文件已经存在或在远程文件系统中,ld 命令生成临时的输出文件。临时输出文件创建在通过 TMPDIR 指定的目录下。如果没有定义 TMPDIR,当输出文件是远程的或在与现有的输出文件同样的目录下时,临时输出文件创建在 /tmp下。
    • OBJECT_MODE 如果既不使用 -b32 也不使用 -b64 选项,检测 OBJECT_MODE 环境变量确定链接方式。如果 OBJECT_MODE 值是 32 或 64,分别使用 32 位或 64 位方式。如果是 32_64 或任意其他值,链接程序打印错误消息并且返回非零值退出。否则,使用 32 位方式。

    注意事项:

    • ld在链接的时候如果路径下相同库名的静态库和动态库都有,那么会优先使用动态链接库,动态链接库不存在才会使用静态链接库,如果指明要使用静态链接库,那么需要在编译命令中加入-static参数;-static是一个全局参数,并不是只对一个链接生效,对所有的链接都生效。也就是一旦写上之后所有库都只会去找静态库,就不能指定某些库链静态库,而其它库链动态库;
    • gcc编译链接时写法都是-L库所在路径, -l库名,

    查找动态库路径

    查资料之后发现运行的时候查找动态库路径顺序如下:

    1. 可执行文件中自带的编译时写入的RPATH、RUNPATH
    2. LD_LIBRARY_PATH 指定的地方
    3. ldconfig 指定的地方:根据/etc/ld.so.cache查找
    4. /lib
    5. /usr/lib

    使用ldconfig修改/etc/ld.so.cache

    • ldconfig命令的用途主要是在默认搜寻目录/lib和/usr/lib以及动态库配置文件/etc/ld.so.conf内所列的目录下,搜索出可共享的动态链接库(格式如lib.so),进而创建出动态装入程序(ld.so)所需的连接和缓存文件(一般默认为/etc/ld.so.cache),此文件保存已排好序的动态链接库名字列表

    • ldconfig通常在系统启动时运行,而当用户安装了一个新的动态链接库时,就需要手工运行这个命令。

    • 由于我们是新安装了redis的动态链接库,因此需要手动把它加到缓存文件中去,具体做法为:

     /etc/ld.so.conf下面加一行/usr/local/lib /usr/local/mysql/lib
    //(需要root权限),保存过后ldconfig一下(需要root权限)
    

    参考链接:

    ld指南
    ld动态库静态库链接问题
    制作静态库和动态库

    相关文章

      网友评论

        本文标题:ld

        本文链接:https://www.haomeiwen.com/subject/tfkanftx.html