谢谢韦老大和独孤君
韦老大回答:
1. 裸板程序烧在FLASH上
一上电,肯定从0地址运行
2. 但是,0地址要么对应NOR FLASH,要么对应只有4K的片内内存
3. 程序要读写数据,或是程序大于4K,怎么办?
4. 程序就要复制到SDRAM里去执行
5. SDRAM那么大,复制到哪个地址去?能随便选择地址吗
6. 不能,要复制到它的链接地址去
7. 为什么一定要复制到它的链接地址去?
8. 因为这个链接地址是程序运行时“应该位于的地方”,比如要访问某个全局变量时,就是访问这个全局变量的链接地址
9. 既然链接地址是SDRAM的地址,那为什么一开始程序可以从0地址运行
10. 因为一开始的程序是“位置无关码”
【
位置无关码:CPU取指时,总是相对于本条执行指令的相对地址去取指。比如指行一个ADD指令时,PC要取下一指令的地址,就在原来的基础上+4。这就不管你代码放在存储器的任何位置,只要他们的相对地址没有改变,就能正常执行程序。一般上电复位那几条语句就必须是位置无关码指令。 位置相关码:可以这样来说,就是CPU每次取指都从绝对位置去取,而不是上面的相对位置。这个绝对地址就是相对起始地址0来说的。这样,就要求你在存放程序时,必须给连接脚本所规定的一样,把代码放到指定位置。
】
独孤君回答:
以下是我自己总结的,看对你有帮助不
加载时域与运行时域:可以这么理解,加载时域涉及到存储地址;运行时域涉及到连接地址(连接地址开始作用的时间是在使用伪指令ldr(adr、adrl)pc,=某符号或是某立即数时)。可执行程序在被下载到相应存储器件里时,它的存储地址可以通过oflash来选择;而当运行该程序时,一开始时PC寄存器的值是指向存储空间的起始地址(起始地址由oflash决定)的,但在遇到伪指令ldr(adr、adrl)pc,=某符号或是某立即数(位置无关相对跳转指令B与BL不影响运行地址)后,程序运行时的绝对地址(即PC寄存器的值)就发生了改变,它是以连接地址(由arm-linux-ld之-T选项设定)为基址,指令所在位置为相对地址共同组成的。总之:运行地址=-T指定的连接地址+相对偏移地址;存储地址=由oflash指定某存储器件的起始地址+由链接文件中AT指定的加载地址。