位置无关码

上节,我们使用到了重定位(就是在你重定位的地方运行你复制的代码,一般再链接脚本里面重定位),把norflash上的代码复制到SDRAM上,所以我们得先初始化SDRAM,不然重定位到SDRAM上也运行不了。

这里面有很多讲究  就比如初始化SDRAM  :  bl sdram_init或者b sdram_init   只能这样用不能使用ldr sdram_init。

因为我们原先把代码重定位到了SDRAM也就是0x30000000以上的地址,先看下我们的链接脚本吧。

所以这些代码段,数据段,bss段地址都会是在SDRAM上的,看看反汇编

看这个start从30000000开始  其实这个30000000我觉得应该就是无关码,假的实际运行还是从0地址开始运行的也就是还在先从nor上运行的,他这样写是方便我们好用ldr跳转,就假装有个禁锢,使用b,bl等调转是跳不出这个禁锢的,只能使用默认的跳转方式(跳转地址=pc(当前地址)+offset(偏移地址,忘记怎么算了)),反正还是在nor上面小范围的调。ldr可以一下跳很远。

上面只是跳转的问题,还有个问题就是,初始值的设置,不能使用数组,全局变量,因为他们会存放在rodata,data段里面,

我们访问它都是以实际地址访问的,他们都保存在栈中,像下面我们没有使用数组和全局变量。这些0x22000000等都是用r2,r3这些寄存器暂时保存一下。

我们看一下用数组初始化sdram里面的寄存器的函数。

ldr r3, 0x300005a8  ,去0x300005a8地方把值给r3, 然后 mov ip,r3  然后去30000708的地方把值给ip寄存器。

看30000708的值就是初始化SDRAM里面写的值。他们是保存在栈里面。都是实际地址跳转的时候

总结一下。重定位的话,bl,b值默认跳转只能在nor里面跳,ldr可以重定位跳转,可以调到sdram里面。

sdram初始化里面不能用数组和全局变量,因为那是还没有初始化sdram那样你相当于在还没有初始化的sdram里面初始化sdram,不起作用。

猜你喜欢

转载自blog.csdn.net/qq_41609723/article/details/81502087