上节,我们使用到了重定位(就是在你重定位的地方运行你复制的代码,一般再链接脚本里面重定位),把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,不起作用。