Linux下的lds链接脚本<一>

    xiaoxiao2021-03-25  126

    一、 概论 每一个链接过程都由 链接脚本 (linker script, 一般以lds作为文件的后缀名)控制.  链接脚本 主要用于规定如何把输入文件内的section放入输出文件内, 并控制输出文件内各部分在程序地址空间内的布局. 但你也可以用连接命令做一些其他事情. 连接器有个默认的内置连接脚本, 可用 ld –verbose查看. 连接选项-r和-N可以影响默认的连接脚本(如何影响?). -T选项用以指定自己的链接脚本, 它将代替默认的连接脚本。你也可以使用以增加自定义的链接命令. 以下没有特殊说明,连接器指的是静态连接器.   二、基本概念 链接器把一个或多个输入文件合成一个输出文件. 输入文件: 目标文件或链接脚本文件. 输出文件: 目标文件或可执行文件. 目标文件(包括可执行文件)具有固定的格式, 在UNIX或GNU/Linux平台下, 一般为ELF格式 有时把输入文件内的section称为 输入section (input section), 把输出文件内的section称为 输出section (output sectin). 目标文件的每个section至少包含两个信息:  名字大小. 大部分section还包含与它相关联的一块数据, 称为 section contents(section内容). 一个section可被标记为“loadable(可加载的)”或“allocatable(可分配的)”. loadable section在输出文件运行时, 相应的section内容将被载入进程地址空间中. allocatable section内容为空的section可被标记为“可分配的”. 在输出文件运行时, 在进程地址空间中空出大小同section指定大小的部分. 某些情况下, 这块内存必须被置零. 如果一个section不是“可加载的”或“可分配的”, 那么该section通常包含了调试信息. 可用 objdump -h命令查看相关信息. 每个“可加载的”或“可分配的”输出section通常包含两个地址VMA (virtual memory address虚拟内存地址或程序地址空间地址)和 LMA( load memory address加载内存地址或进程地址空间地址). 通常VMA和LMA是相同的. 在目标文件中, loadable或allocatable的输出section有两种地址:  VMA(virtual Memory Address)和 LMA(Load Memory Address). VMA是执行输出文件时section所在的地址, 而LMA是加载输出文件时section所在的地址. 一般而言, 某section的VMA == LMA. 但在嵌入式系统中, 经常存在加载地址和执行地址不同的情况: 比如将输出文件加载到开发板的flash中(由LMA指定), 而在运行时将位于flash中的输出文件复制到SDRAM中(由VMA指定). 可这样来理解VMA和LMA, 假设: (1) .data section对应的VMA地址是0×08050000, 该section内包含了3个32位全局变量, i、j和k, 分别为1,2,3. (2) .text section内包含由”printf( “j=%d “, j );”程序片段产生的代码. 连接时指定.data section的VMA为0×08050000, 产生的printf指令是将地址为0×08050004处的4字节内容作为一个整数打印出来。 如果.data section的LMA为0×08050000,显然结果是j=2 如果.data section的LMA为0×08050004,显然结果是j=1 还可这样理解LMA: .text section内容的开始处包含如下两条指令(intel i386指令是10字节,每行对应5字节): jmp 0×08048285 movl $0×1,
    转载请注明原文地址: https://ju.6miu.com/read-6748.html

    最新回复(0)