LDS文件格式简单用法备注
2016-03-17 15:07
197 查看
在分析u-boot和linux内核源代码的时候经常需要分析lds文件来判断代码的运行地址和加载地址。对于这两个地址我曾经困扰了很久,今天参考这篇文章(/article/9082711.html)终于搞清楚了他们的区别。
其中:
1)secname是必须存在的字段,是该段的名称;
2)start 为段的重定位地址,即本段在内存中链接(运行)时的地址,一般也是需要指出的;
3)align是内存对其的方式;
4)AT是本段存储加载的地址(在生成的文件中的地址);
5)content决定哪些内容放在本段,可以使整个目标文件,也可以是目标文件中的某段(代码段,数据段等)。
需要指出的是,如果没有用AT指令给出该段存放的地址,那么该段就会放在和运行地址相同的地方,他们只是地址相同,一个在内存,一个在生成的可执行文件里面。
在这个例子中,
第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+代码段的长度+对齐后的只读数据段的长度+对齐后的数据段的长度后的位置。
理解的不到位的地方,请读者提出并指教,我将不甚感激。
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+代码段的长度+对齐后的只读数据段的长度+对齐后的数据段的长度后的位置。
理解的不到位的地方,请读者提出并指教,我将不甚感激。
相关文章推荐
- Android开发_控制硬加速hardwareAccelerated
- 【慕课笔记】6-1 序列化基本操作
- Java:单例模式的7种写法
- oracle DBLink的使用
- Writable collections
- 配置本地yum源的方法
- 时间按格式转字符串
- [2016-3-17 Test]
- oracle user_users,dba_users,all_users
- log4j简单使用
- eclipse中Debug简单记忆
- eclipse中Debug简单记忆
- 排序、查找
- 基于Android的ELF PLT/GOT符号重定向过程
- 博客专栏第一篇文章——我眼里的DBA
- Can't connect to local MySQL server through socket '/var/run/mysqld/mysqld.sock'解决
- iOS涂鸦-自由画笔可以用到的代码
- HTTP请求WebTool
- 常用的一些正则验证
- oracle 11g 分区