嵌入式Linux-linux连接脚本
嵌入式Linux-linux连接脚本
介绍
每一个链接过程都由链接脚本(linker script, 一般以lds作为文件的后缀名)控制。 链接脚本主要用于规定如何把输入文件内的section放入输出文件内, 并控制输出文件内各部分在程序地址空间内的布局。
链接器在进行链接时,会根据链接脚本从输入的.o文件中挑选出感兴趣的section,把它们合并生成新的section,这些新产生的section归属于目标文件的某个segment(段),并出现在目标文件中。例如file1.o和file2.o分别有两个.text,它们在链接后生产的目标文件也会有一个.text,而这个.text既是由file1.o和file2.o的.text合并而来的。
终端中使用链接脚本
代码示例
//提供名为:test.dls的连接脚本
arm-linux-ld -Ttest.dls -o test test1.o test2.o
//不提供连接脚本
arm-linux-ld -o test test1.o test2.o
具体介绍
-
如果你不提供链接脚本,则链接器会使用一个缺省的脚本,这个脚本是被编译进链接器可执行文件的.
-
可以使用–verbose命令行显示缺省的链接器脚本的内容.
arm-linux-ld --verbose //查看arm-linux-ld连接的默认连接脚本 ld --verbose //ld连接的默认连接脚本
-
你可以使用-T命令行来提供你自己的链接脚本来替换缺省的链接脚本.
//提供名为:test.dls的连接脚本 arm-linux-ld -Ttest.dls -o test test1.o test2.o
-
-T选项可以指定数据段,代码段,bss段起始位置。(使用默认连接脚本的选项)
-Ttext startaddr -Tdata startaddr -Tbss startaddr //例子示范 ld –Ttext 0x00000000 –g led_on.o –o led_on_elf
语法基本介绍
SECTIONS命令:
The SECTIONS command tells the linker how to map input sections into output sections, and how to place the output sections in memory.
命令格式如下:
SECTIONS
{
sections-command
sections-command
......
}
地址计数器‘.’
该符号只能用于SECTIONS命令内部,初始值为‘0’,可以对该符号进行赋值,也可以使用该符号进行计算或赋值给其他符号。它会自动根据SECTIONS命令内部所描述的输出段的大小来计算当前的地址。
输出段描述
前面提到在SECTIONS命令中可以作输出段描述
命令格式如下:
section [address] [(type)] : [AT(lma)]
{
output-section-command
output-section-command ...
} [>region] [AT>lma_region] [:phdr :phdr ...] [=fillexp]
secname和contents是必须的,其他的都是可选的。下面挑几个常用的看看:
- secname:段名
- contents:决定哪些内容放在本段,可以是整个目标文件,也可以是目标文件中的某段(代码段、数据段等)
- start:定义本段连接(运行)的地址,如果没有使用
AT(ldadr)
,本段存储的地址也是start。GNU网站上说start可以用任意一种描述地址的符号来描述。(连接(运行)的地址:就是当可执行文件,运行时,段的位置) -
AT(ldadr):定义本段存储(加载)的地址。(本段存储的地址:就是编译完之后,相应段存储在ELF运行文件中的存储地址)
例子+解释
例子
/* nand.lds */
SECTIONS {
firtst 0x00000000 : { head.o init.o }
second 0x30000000 : AT(4096) { main.o }
}
解释
-
head.o放在0x00000000地址开始处,init.o放在head.o后面,他们的运行地址也是0x00000000,即连接和存储地址相同(没有AT指定);
-
main.o放在4096(0x1000,是AT指定的,存储地址)开始处,但是它的运行地址在0x30000000,运行之前需要从0x1000(加载处)复制到0x30000000(运行处),此过程也就用到了读取Nand flash。
-
这就是存储地址和连接(运行)地址的不同,称为加载时域和运行时域,可以在.lds连接脚本文件中分别指定
例子+解释
OUTPUT_ARCH(arm) //设置编译出来可执行文件的架构
ENTRY(_start) //设置可执行文件起始的的段位置。这里设置了_start,因此在在输入段中,一定存在一个名为_start的段。
SECTIONS {
. = 0xa3f00000; //设置起始位置
__boot_start = .; //复制参数,在最后用到了
.start ALIGN(4) :{
*(.text.start)
} //ALIGN(4)表示4字节对齐
.setup ALIGN(4) : {
setup_block = .;
*(.setup)
setup_block_end = .;
}
.text ALIGN(4) : {
*(.text)
}
.rodata ALIGN(4) : {
*(.rodata)
}
.data ALIGN(4) : {
*(.data)
}
.got ALIGN(4) : {
*(.got)
}
__boot_end = .;
.bss ALIGN(16) : {
bss_start = .;
*(.bss)
*(COMMON)
bss_end = .;
} //ALIGN(16)表示16字节对齐
.comment ALIGN(16) : {
*(.comment)
}
stack_point = __boot_start + 0x00100000;
loader_size = __boot_end - __boot_start;
setup_size = setup_block_end - setup_block;
}
参考
Linux链接脚本学习–lds(写的比较基础的,可以作为入门看看)
百度文库解释(前半部分还不错的)
lds链接脚本*三篇博客(超级长的,排版不好的文章.不过很多人转了,可以看一下。)