原创 STM32启动代码概述

2009-8-4 08:39 1842 1 1 分类: MCU/ 嵌入式

来源 CEPARK网站http://www.cepark.com/Index.html  作者: hnrain


一般嵌入式开发流程就是先建立一个工程,再编写源文件,然后进行编译,把所有的*.s文件和*.c文件编译成一个*.o文件,再对目标文件进行链接和定位,编译成功后会生成一个*.hex文件和调试文件,接下来要进行调试,如果成功的话,就可以将它固化到flash里面去。


启动代码是用来初始化电路以及用来为高级语言写的软件作好运行前准备的一小段汇编语言,是任何处理器上电复位时的程序运行入口点。


比如,刚上电的过程中,PC机会对系统的一个运行频率进行锁定在一个固定的值,这个设计频率的过程就是在汇编源代码中进行的,也就是在启动代码中进行的。与此同时,设置完后,程序开始运行,注意,程序是在内存中运行的。这个时候,就需要把一些源文件从flash里面copy到内存中,又要对它们进行初始化读写,这又有频率的设置。这些都是初始化。


初始化完成后,我们又要设置一些堆栈,要跳到C语言的main函数里面运行。这就需要堆栈。对普通的ARM CPU有这样一个要求:在绝对地址为零的地方要放置一个异常向量表,但并不是所有的ARM CPU都留有这个一个空间,这就需要用到映射功能。我们可以将其它地方的一些空间映射到绝对地址里面。当发生异常时,ARM核来读取异常中断表的时候,它会使用映射之后的那个表,这个就可以接着往下执行,否则在绝对地址零的地方找不到任何信息,程序就会死掉。这些运行的环境全部建立好后,程序就会跳转到我们的main函数里面。


总之,启动代码,就是对最小系统的初始化。包括晶振,CPU频率等。


启动代码的最小系统是:异常向量表的初始化存储区分配初始化堆栈高级语言入口函数调用– main()函数。


程序的启动过程:


点击看大图


 


 


以下面这个例子为例,编译完后,DEBUG后,我们可以看到,光标指向绝对地址为零的地方,这里存放的就是一个异常向量表。


 


6136_200905201600352PsPF.jpg


 


它对应在startup.s里的源文件如下:


 


点击看大图


 


单步运行后,马上跳转到初始化CPU的频率。即初始化锁相环,将其锁在一个固定的频率。具体代码如下:


; Setup PLL


               IF     PLL_SETUP <> 0


               LDR    R0, =PLL_BASE


               MOV    R1, #0xAA


               MOV    R2, #0x55


<?XML:NAMESPACE PREFIX = O />

 


; Configure and Enable PLL


               MOV    R3, #PLLCFG_Val


               STR    R3, [R0, #PLLCFG_OFS]


               MOV    R3, #PLLCON_PLLE


               STR    R3, [R0, #PLLCON_OFS]


               STR    R1, [R0, #PLLFEED_OFS]


               STR    R2, [R0, #PLLFEED_OFS]


 


; Wait until PLL Locked


PLL_Loop       LDR    R3, [R0, #PLLSTAT_OFS]


               ANDS   R3, R3, #PLLSTAT_PLOCK


               BEQ    PLL_Loop


 


; Switch to PLL Clock


               MOV    R3, #(PLLCON_PLLE:OR:PLLCON_PLLC)


               STR    R3, [R0, #PLLCON_OFS]


               STR    R1, [R0, #PLLFEED_OFS]


               STR    R2, [R0, #PLLFEED_OFS]


               ENDIF  ; PLL_SETUP


 


然后再初始化每一种模式的堆栈,再进行单步运行的时候,下面我们可以看到,它自动跳转到main()函数:


; Enter the C code


 


               IMPORT __main


               LDR    R0, =__main


               BX     R0


 


 


               IF     :DEF:__MICROLIB


 


               EXPORT __heap_base


               EXPORT __heap_limit


 


               ELSE


 


这个时候,程序会运行各种scatterload函数,将我们的堆栈、全局变量等内容拷贝到内存中去。拷贝完后,就正式跳转到我们的main()函数中来执行了。


 


点击看大图


 


这就是启动代码执行的全过程,呵呵,平时我们看到以为只是执行main()函数就行了,是不是没有想到在执行 main() 函数后还有这么多学问呢?


 


如果觉得对你有有帮助,请顶一顶,这是对我很大的支持!

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
1
关闭 站长推荐上一条 /3 下一条