链接脚本,文字无关码,运行地址,…

位置无关代码的概念,指代码不在连接时制定的运行地址空间,也可以执行,它一段加载到任意地址空间都能执行的特殊代码。

SECTIONS{//定义域中所包含的的段
...
secname start BLOCK(align)(NOload) :AT(laddr)
{contents}>region:phdr=fill
...
}
secname 和contents是必须的,其他都是可选的;
secname:段名,理解为该段输出
contents:决定哪些内容放在本段,可以理解为该段输入
start:本段运行地址,若不存在AT(ladr),本段存储的也为start。start可以用任意一种描述地址的符号来描述,若代码中含有位置无关码,程序运行时必须放在这个地址上。
AT(laddr):定义本段存储地址


SECTIONS {
first 0x000000 : {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连接脚本文件中分别指定。
            编写好的.lds文件,在用arm-linux-ld连接命令时带-Tfilename来调用执行,如arm-linux-ld –Tnand.lds x.o y.o –o xy.o。也用-Ttext参数直接指定连接地址,如arm-linux-ld –Ttext 0x30000000 x.o y.o –o xy.o。
总之:
          连接地址<==>运行地址
          存储地址<==>加载地址
既然程序有了两种地址,就涉及到一些跳转指令的区别,下面就来具体看看这些跳转指令。
ARM汇编中,常有两种跳转方法:b跳转指令、ldr指令向PC赋值。
(1)       b step1 :b跳转指令是相对跳转,依赖当前PC的值,偏移量是通过该指令本身的bit[23:0]算出来的,这使得使用b指令的程序不依赖于要跳到的代码的位置,只看指令本身。这就是位置无关码。
(2)       ldr pc, =step1 :该指令是从内存中的某个位置(step1)读出数据并赋给PC,同样依赖当前PC的值,但是偏移量是那个位置(step1)的连接地址(运行时的地址),所以可以用它实现从Flash到RAM的程序跳转。

对于程序:
链接脚本:
SECTION {
...
.text 0x30000000 ={main.o}
...
}
...
b main //此时main是一个相对地址,可以理解为x86汇编里面的偏移地址,故不管在不在main有没有被加载在0x30000000,都能运行。
...
ldr pc,=main //此时main对应的是一个绝对地址,在.lds里面或-Ttext中确定,只有main被加载到0x30000000才能运行。
由上面的论述可以得知,如果你的这段代码需要实现位置无关,那么你就不能使用绝对寻址指令,否则的话就是位置有关了。
如u-boot.lds:
OUTPUT_FORMAT("elf32-littlearm", "elf32-littlearm", "elf32-littlearm")
指定输出可执行文件是elf格式,31位arm指令,小端
OUTPUT_ARCH(arm) //输出可执行文件的体系架构是arm
ENTRY(_start)   //可执行文件的入口是_start段
SECTIONS
{
. = 0x00000000; 从0x0位置开始

. = ALIGN(4);   代码以四字节对齐
.text       : //.text的基址由LDFLAGS中-Ttext $(TEXT_BASE)指定
{
  cpu/arm920t/start.o (.text)
  *(.text)
}

. = ALIGN(4);
.rodata : { *(.rodata) } 只读数据段

. = ALIGN(4);
.data : { *(.data) } 读/写数据段

. = ALIGN(4);
.got : { *(.got) } 

. = .;
__u_boot_cmd_start = .;
.u_boot_cmd : { *(.u_boot_cmd) }
__u_boot_cmd_end = .;

. = ALIGN(4);
__bss_start = .;   //bss段
.bss : { *(.bss) }
_end = .;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值