您的位置:首页 > 其它

LDS文件格式简单用法备注

2016-03-17 15:07 197 查看
在分析u-boot和linux内核源代码的时候经常需要分析lds文件来判断代码的运行地址和加载地址。对于这两个地址我曾经困扰了很久,今天参考这篇文章(/article/9082711.html)终于搞清楚了他们的区别。

1. LDS文件的基本格式

SECTIONS{
...
secname start BLOCK(align)(NOLOAD):AT(ldadr)
{contents}>region:phdr = fill
...
}


其中:

1)secname是必须存在的字段,是该段的名称;

2)start 为段的重定位地址,即本段在内存中链接(运行)时的地址,一般也是需要指出的;

3)align是内存对其的方式;

4)AT是本段存储加载的地址(在生成的文件中的地址);

5)content决定哪些内容放在本段,可以使整个目标文件,也可以是目标文件中的某段(代码段,数据段等)。

需要指出的是,如果没有用AT指令给出该段存放的地址,那么该段就会放在和运行地址相同的地方,他们只是地址相同,一个在内存,一个在生成的可执行文件里面。

2. 从一个简单的例子来分析

1. SECTIONS {
2.       . = 0x00000000;
3.       .init : AT(0){ head.o init.o nand.o}
4.       . = 0x30000000;
5.       .text : AT(4096) { *(.text) }
6.       .rodata ALIGN(4) : AT((LOADADDR(.text)+SIZEOF(.text)+3)&~(0x03)) {*(.rodata*)}
7.       .data ALIGN(4)   : AT((LOADADDR(.rodata)+SIZEOF(.rodata)+3)&~(0x03)) { *(.data) }
8.       __bss_start = .;
9.       .bss ALIGN(4)  : { *(.bss)  *(COMMON) }
10.       __bss_end = .;
11. }


在这个例子中,

第2行代码将当前链接地址设为0,此时该段的加载(存储)地址也是0,第3行定义了init段,它的链接地址和加载地址都为0,存放了head.o、init.o和nand.o三个文件,按先后顺序从0地址开始存放。

第4行又将链接地址设为了0x30000000,第5行定义了代码段,它的链接地址为0x30000000而加载地址(我喜欢叫做存储,不是很准确,但好理解)为4096,就是说代码段存放在了距离生成的可执行文件4096字节的地方,程序运行的时候就需要将这个位置的代码拷贝到0x30000000的位置(内存中)去运行,忘了说了该段包含所有代码的代码段,因为是*(.text)。

第6行定义了只读数据段,该段4字节对齐,加载地址为(LOADADDR(.text)+SIZEOF(.text)+3)&~(0x03),即(代码段的位置+代码段的长度+3)&~(0x03)的位置,这里加3和与~0x03做与运算是为了4字节对齐,该段也包含了所有代码的只读数据段。

第7行定义了数据段,该段也是4字节对齐,加载地址为(LOADADDR(.rodata)+SIZEOF(.rodata)+3)&~(0x03),即(只读数据段的位置+只读数据段的长度+3)&~(0x03)的位置,这里加3和与~0x03做与运算是为了4字节对齐,该段也包含了所有代码的数据段。

第8到第10行定义了堆栈段,这个部分在内存中,0x30000000+代码段的长度+对齐后的只读数据段的长度+对齐后的数据段的长度后的位置。

理解的不到位的地方,请读者提出并指教,我将不甚感激。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: