STM32f10x启动代码分析
读这个文件的时候用到了很多资料,大部分来自网络,作者不详,感谢他们。
其中知名的有,理想智芯科技的李想老师,玩转stm32的作者野火团队,感谢他们。
这是arm公司的技术支持可以用中文http://infocenter.arm.com/help/index.jsp。
;EQU伪指令 为0x00000400取个别名Stack_Size
;Stack_Size指定栈的大小
Stack_Size
;AREA伪指令定义一个段STACK(栈段),属性为:NOINIT(不初始化), READWRITE(可读可写), ALIGN=3(8字
;节对齐)。
;APACE伪指令分配栈空间
;Stack_Mem变量在栈低
;__initial_sp变量的栈顶
Stack_Mem
__initial_sp
;下面代码语法同上,定义堆空间
Heap_Size
__heap_base
Heap_Mem
__heap_limit
;定义REST段,属性DATA(数据类型),READONLY(只读)
;下面代码完成中断向量的存储工作,
;DCD xxxx_xxxx 意为安字存储xxxx_xxxx变量的首地址
;__Vectors变量存储的内容为__initial_sp变量的地址
;__Vectors的地址为中断向量表的首地址
;__Vectors_End
;__Vectors_Size 代表向量标的大小
__Vectors
__Vectors_End
__Vectors_Size
;定义|.text|段,属性为:CODE(代码), READONLY(只读)。
;|.text|习惯性命名,表示由C编译程序产生的代码段,或用于以某种方式与C库关联的代码段
;下面代码相当于定义了Reset_Handler这个函数,我们通过Reset_Handler变量的地址就可以跳转过来
;由于这个段是这个文件唯一的代码段,且把main函数地址作为变量导入所以,cpu上电后会从下面程序开始
;运行。
Reset_Handler
;EXPORT
;那么这里定义的xxxx,1.有效,2.作为输出供其他文件中的函数调用。如果其文件定义了xxxx那么这个文件定
;义的xxxx无效。可以再arm的汇编指南中查阅到。
;导入变量__main与SystemInit。一个是用户主程序,一个是系统时钟初始化程序
;有返回的调用SystemInit函数,初始化系统时钟
;可以看出,在进入main函数之前,这里先初始换了系统时钟。在主函数中,就不用调用这个函数了
;SystemInit这个函数是在文件system_stm32f10x.c中实现的。system_stm32f10x.c在固件库的目录为
;STM32F10x_StdPeriph_Lib_V3.5.0LibrariesCMSISCM3DeviceSupportSTSTM32F10x
;无返回的调用main函数,把控制全交给main函数。
;下面从NMI_Handler到SysTick_Handler PROC定义的是内核中断函数。
;在这里可以看出,如果用户不定义这些函数,经使用下面的定义。由伪指令EXPORT [WEAK]决定
;一定中断发生且被允许,将执行B .。这是个死循环相当于jump $
NMI_Handler
HardFault_Handler
MemManage_Handler
BusFault_Handler
UsageFault_Handler
SVC_Handler
DebugMon_Handler
PendSV_Handler
SysTick_Handler PROC
;下面中断函数有st给出,同样用户如果不自己定义,
;一定中断允许且发生,将进入死循环
Default_Handler PROC
WWDG_IRQHandler
PVD_IRQHandler
TAMPER_IRQHandler
RTC_IRQHandler
FLASH_IRQHandler
RCC_IRQHandler
EXTI0_IRQHandler
EXTI1_IRQHandler
EXTI2_IRQHandler
EXTI3_IRQHandler
EXTI4_IRQHandler
DMA1_Channel1_IRQHandler
DMA1_Channel2_IRQHandler
DMA1_Channel3_IRQHandler
DMA1_Channel4_IRQHandler
DMA1_Channel5_IRQHandler
DMA1_Channel6_IRQHandler
DMA1_Channel7_IRQHandler
ADC1_2_IRQHandler
USB_HP_CAN1_TX_IRQHandler
USB_LP_CAN1_RX0_IRQHandler
CAN1_RX1_IRQHandler
CAN1_SCE_IRQHandler
EXTI9_5_IRQHandler
TIM1_BRK_IRQHandler
TIM1_UP_IRQHandler
TIM1_TRG_COM_IRQHandler
TIM1_CC_IRQHandler
TIM2_IRQHandler
TIM3_IRQHandler
TIM4_IRQHandler
I2C1_EV_IRQHandler
I2C1_ER_IRQHandler
I2C2_EV_IRQHandler
I2C2_ER_IRQHandler
SPI1_IRQHandler
SPI2_IRQHandler
USART1_IRQHandler
USART2_IRQHandler
USART3_IRQHandler
EXTI15_10_IRQHandler
RTCAlarm_IRQHandler
USBWakeUp_IRQHandler
TIM8_BRK_IRQHandler
TIM8_UP_IRQHandler
TIM8_TRG_COM_IRQHandler
TIM8_CC_IRQHandler
ADC3_IRQHandler
FSMC_IRQHandler
SDIO_IRQHandler
TIM5_IRQHandler
SPI3_IRQHandler
UART4_IRQHandler
UART5_IRQHandler
TIM6_IRQHandler
TIM7_IRQHandler
DMA2_Channel1_IRQHandler
DMA2_Channel2_IRQHandler
DMA2_Channel3_IRQHandler
DMA2_Channel4_5_IRQHandler
;下面这些代码初始化堆栈。
;他们是相当复杂的,以你为我们的main和__main是不同的,__main进过编译后的main,
;编译器要对进入main做了些手脚,初始化C语言环境,其中就要堆栈。
;下面这些就是在那时候用的。可以跟踪到他们在进入main前的执行。
;当然要在disassembly先才可以看到。
;见解来自Triquinne from elecfans
;http://www.elecfans.com/emb/danpianji/20120910287921_6.html
__user_initial_stackheap
好了就这么多了,上面的解释也差不多了。
评论