中容量STM32处理器启动代码的理解

发布者:SereneMeadow7最新更新时间:2019-03-24 来源: eefocus关键字:中容量  STM32  启动代码 手机看文章 扫描二维码
随时随地手机看文章

今天要介绍一下STM32的启动代码,我这里使用的是中容量的STM32f103c8t6,对应的启动文件就是startup_stm32f10x_md.s,我这里启动文件版本是V3.6.2


不多说废话,先上我注释过的源码:


 

/**

  ******************************************************************************

  * @file      startup_stm32f10x_md.s

  * @author    MCD Application Team

  * @version   V3.6.2

  * @date      28-February-2013

  * @brief     STM32F10x Medium Density Devices vector table for RIDE7 toolchain.

  *            This module performs:

  *                - Set the initial SP                                              -设置最初的SP寄存器值

  *                - Set the initial PC == Reset_Handler, -设置最初的PC寄存器值 == Reset_Handler

  *                - Set the vector table entries with the exceptions ISR address    -在有特殊的ISR(中断服务程序)地址的情况下设置向量表入口

  *                - Configure the clock system -设置时钟系统

  *                - Branches to main in the C library (which eventually -C库文件中由分支到主函数(最后调用了主函数)

  *                  calls main()).

  *            After Reset the Cortex-M3 processor is in Thread mode, 在线程模式下复位了CM3处理器后,此时优先级特权化,栈顶设置为主函数

  *            priority is Privileged, and the Stack is set to Main.

  ******************************************************************************

  * @attention

  *

  *

© COPYRIGHT 2013 STMicroelectronics

  *

  * Licensed under MCD-ST Liberty SW License Agreement V2, (the "License");

  * You may not use this file except in compliance with the License.

  * You may obtain a copy of the License at:

  *

  *        http://www.st.com/software_license_agreement_liberty_v2

  *

  * Unless required by applicable law or agreed to in writing, software 

  * distributed under the License is distributed on an "AS IS" BASIS, 

  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

  * See the License for the specific language governing permissions and

  * limitations under the License.

  *

  ******************************************************************************

  */

    

  .syntax unified                                                                   ;表示使用了统一汇编语言语法

.cpu cortex-m3                                                                  ;MCU为CM3

.fpu softvfp

.thumb                                                                          ;指令集

 

.global g_pfnVectors                                                                ;global使得g_pfnVectors可以被其他目标文件使用

.global Default_Handler

 

/* start address for the initialization values of the .data section.                初始化.data 块的起始地址,这个地址在链接脚本中被定义

defined in linker script */

.word _sidata

/* start address for the .data section. defined in linker script                    .data块的起始地址,这个地址在链接脚本中被定义*/  

.word _sdata

/* end address for the .data section. defined in linker script                      .data块的结束地址,这个地址在链接脚本中被定义*/

.word _edata

/* start address for the .bss section. defined in linker script                     .bss块的起始地址,这个地址在链接脚本中被定义*/

.word _sbss

/* end address for the .bss section. defined in linker script                       .bss块的结束地址,这个地址在链接脚本中被定义*/

.word _ebss

 

.equ  BootRAM, 0xF108F85F                                                           /*这里跟c中的宏定义类似,即BootRAM = 0xF108F85F,不参与编译*/

/**

 * @brief  This is the code that gets called when the processor first

 *          starts execution following a reset event. Only the absolutely

 *          necessary set is performed, after which the application

 *          supplied main() routine is called. 

 * @param  None

 * @retval : None

*/

 

    .section .text.Reset_Handler

.weak Reset_Handler

.type Reset_Handler, %function

Reset_Handler:

 

/* Copy the data segment initializers from flash to SRAM                             将初始化了的数据段从flash复制到SRAM*/  

  movs r1, #0                                                                       ;将立即数0赋值给r1寄存器

  b LoopCopyDataInit                                                                 ;程序转移到LoopCopyDataInit处

 

CopyDataInit:                                                                        ;从FLASH中拷贝地址在sdata和edata之间的代码到SRAM中

ldr r3, =_sidata                                                                 ;从存储器中将_sidata加载到寄存器r3中

ldr r3, [r3, r1]                                                                 ;从地址r3+r1处读取一个字(32bit)到r3中

str r3, [r0, r1]                                                                 ;把寄存器r3的值存储到存储器中地址为r0+r1地址处

adds r1, r1, #4                                                               ;r1 = r1 + 4

    

LoopCopyDataInit:

ldr r0, =_sdata                                                                  ;从存储器中将_sidata加载到寄存器r0中

ldr r3, =_edata                                                                  ;从存储器中将_edata加载到寄存器r3中

adds r2, r0, r1                                                               ;r2=r0+r1

cmp r2, r3                                                                       ;计算r2 - r3,若小于0,标志位为0,反之为1

bcc CopyDataInit                                                                 ;如果标志位为0(无借位)即r2

ldr r2, =_sbss                                                                   ;从存储器中将_sbss加载到寄存器r2中

b LoopFillZerobss                                                              ;无条件跳转到LoopFillZerobss

/* Zero fill the bss segment. */  

FillZerobss:

movs r3, #0                                                                   ;将立即数0存入寄存器r3

str r3, [r2], #4                                                                 ;将寄存器r3的值存储到地址为r2寄存器值得地址处后,r2 = r2 + 4

    

LoopFillZerobss:

ldr r3, = _ebss                                                                  ;从存储器中将_ebss加载到寄存器r3中

cmp r2, r3                                                                       ;比较r2,r3,然后更新标志位

bcc FillZerobss                                                                  ;如果标志位为0(无借位),则跳转到FillZerobss处

/* Call the clock system intitialization function.*/

/* bl  SystemInit */

/* Call the application's entry point.*/

bl main                                                                         ;转移到main函数起始处

bx lr                                                                           ;转移到地址lr处

.size Reset_Handler, .-Reset_Handler                                               ;相当于mov pc lr    lr即寄存器r14

 

/**

 * @brief  This is the code that gets called when the processor receives an 

 *         unexpected interrupt. This simply enters an infinite loop, preserving

 *         the system state for examination by a debugger.

 * @param  None     

 * @retval None       

*/

    .section .text.Default_Handler,"ax",%progbits

Default_Handler:

Infinite_Loop:                                                                           

b Infinite_Loop                                                                ;无条件跳转到Infinite_Loop

.size Default_Handler, .-Default_Handler

/******************************************************************************

*

* The minimal vector table for a Cortex M3.  Note that the proper constructs         Cortex M3的最小向量表

* must be placed on this to ensure that it ends up at physical address

* 0x0000.0000.

*

******************************************************************************/    

  .section .isr_vector,"a",%progbits                                            ;定义中断向量表的数据段

.type g_pfnVectors, %object

.size g_pfnVectors, .-g_pfnVectors

    

    

g_pfnVectors:

.word _estack                                                                  ;在当前位置放置一个word型的值,这个值为_estack;后面同理

.word Reset_Handler                                                            ;这一块程序的意思就是说以g_pfnVectors为初始地址,然后依次

.word NMI_Handler                                                              ;以字为单位写入相应的数据

.word HardFault_Handler

.word MemManage_Handler

.word BusFault_Handler

.word UsageFault_Handler

.word 0

.word 0

.word 0

.word 0

.word SVC_Handler

.word DebugMon_Handler

.word 0

.word PendSV_Handler

.word SysTick_Handler

.word WWDG_IRQHandler

.word PVD_IRQHandler

.word TAMPER_IRQHandler

.word RTC_IRQHandler

.word FLASH_IRQHandler

.word RCC_IRQHandler

.word EXTI0_IRQHandler

.word EXTI1_IRQHandler

.word EXTI2_IRQHandler

.word EXTI3_IRQHandler

.word EXTI4_IRQHandler

.word DMA1_Channel1_IRQHandler

.word DMA1_Channel2_IRQHandler

.word DMA1_Channel3_IRQHandler

.word DMA1_Channel4_IRQHandler

.word DMA1_Channel5_IRQHandler

.word DMA1_Channel6_IRQHandler

.word DMA1_Channel7_IRQHandler

.word ADC1_2_IRQHandler

.word USB_HP_CAN1_TX_IRQHandler

.word USB_LP_CAN1_RX0_IRQHandler

.word CAN1_RX1_IRQHandler

.word CAN1_SCE_IRQHandler

.word EXTI9_5_IRQHandler

.word TIM1_BRK_IRQHandler

.word TIM1_UP_IRQHandler

.word TIM1_TRG_COM_IRQHandler

.word TIM1_CC_IRQHandler

.word TIM2_IRQHandler

.word TIM3_IRQHandler

.word TIM4_IRQHandler

.word I2C1_EV_IRQHandler

.word I2C1_ER_IRQHandler

.word I2C2_EV_IRQHandler

.word I2C2_ER_IRQHandler

.word SPI1_IRQHandler

.word SPI2_IRQHandler

.word USART1_IRQHandler

.word USART2_IRQHandler

.word USART3_IRQHandler

.word EXTI15_10_IRQHandler

.word RTCAlarm_IRQHandler

.word USBWakeUp_IRQHandler

  .word 0

.word 0

.word 0

.word 0

.word 0

.word 0

.word 0

.word BootRAM          /* @0x108. This is for boot in RAM mode for             这个是为stm32f10系列中容量系列设备的boot中的RAM模式准备的

                            STM32F10x Medium Density devices. */

   

/*******************************************************************************

*

* Provide weak aliases for each Exception handler to the Default_Handler. 

* As they are weak aliases, any function with the same name will override 

* this definition.

*

*******************************************************************************/

    

  .weak NMI_Handler                                                                 ;定义一个弱别名NMI_Handler

.thumb_set NMI_Handler,Default_Handler                                          ;如果这个不重写这个弱别名,那么默认执行Default_Handler,反之这执行重写过的NMI_Handler


  .weak HardFault_Handler

.thumb_set HardFault_Handler,Default_Handler


  .weak MemManage_Handler

.thumb_set MemManage_Handler,Default_Handler


  .weak BusFault_Handler

.thumb_set BusFault_Handler,Default_Handler

 

.weak UsageFault_Handler

.thumb_set UsageFault_Handler,Default_Handler

 

.weak SVC_Handler

.thumb_set SVC_Handler,Default_Handler

 

.weak DebugMon_Handler

.thumb_set DebugMon_Handler,Default_Handler

 

.weak PendSV_Handler

.thumb_set PendSV_Handler,Default_Handler

 

.weak SysTick_Handler

.thumb_set SysTick_Handler,Default_Handler

 

.weak WWDG_IRQHandler

.thumb_set WWDG_IRQHandler,Default_Handler

 

.weak PVD_IRQHandler

.thumb_set PVD_IRQHandler,Default_Handler

 

.weak TAMPER_IRQHandler

.thumb_set TAMPER_IRQHandler,Default_Handler

 

.weak RTC_IRQHandler

.thumb_set RTC_IRQHandler,Default_Handler

 

.weak FLASH_IRQHandler

.thumb_set FLASH_IRQHandler,Default_Handler

 

.weak RCC_IRQHandler

.thumb_set RCC_IRQHandler,Default_Handler

 

.weak EXTI0_IRQHandler

.thumb_set EXTI0_IRQHandler,Default_Handler

 

.weak EXTI1_IRQHandler

.thumb_set EXTI1_IRQHandler,Default_Handler

 

.weak EXTI2_IRQHandler

.thumb_set EXTI2_IRQHandler,Default_Handler

 

.weak EXTI3_IRQHandler

.thumb_set EXTI3_IRQHandler,Default_Handler

 

.weak EXTI4_IRQHandler

.thumb_set EXTI4_IRQHandler,Default_Handler

 

.weak DMA1_Channel1_IRQHandler

.thumb_set DMA1_Channel1_IRQHandler,Default_Handler

 

.weak DMA1_Channel2_IRQHandler

.thumb_set DMA1_Channel2_IRQHandler,Default_Handler

 

.weak DMA1_Channel3_IRQHandler

.thumb_set DMA1_Channel3_IRQHandler,Default_Handler

 

.weak DMA1_Channel4_IRQHandler

.thumb_set DMA1_Channel4_IRQHandler,Default_Handler

 

.weak DMA1_Channel5_IRQHandler

.thumb_set DMA1_Channel5_IRQHandler,Default_Handler

 

.weak DMA1_Channel6_IRQHandler

.thumb_set DMA1_Channel6_IRQHandler,Default_Handler

 

.weak DMA1_Channel7_IRQHandler

.thumb_set DMA1_Channel7_IRQHandler,Default_Handler

 

.weak ADC1_2_IRQHandler

.thumb_set ADC1_2_IRQHandler,Default_Handler

 

.weak USB_HP_CAN1_TX_IRQHandler

.thumb_set USB_HP_CAN1_TX_IRQHandler,Default_Handler

 

.weak USB_LP_CAN1_RX0_IRQHandler

.thumb_set USB_LP_CAN1_RX0_IRQHandler,Default_Handler

 

.weak CAN1_RX1_IRQHandler

.thumb_set CAN1_RX1_IRQHandler,Default_Handler

 

.weak CAN1_SCE_IRQHandler

.thumb_set CAN1_SCE_IRQHandler,Default_Handler

 

.weak EXTI9_5_IRQHandler

.thumb_set EXTI9_5_IRQHandler,Default_Handler

 

.weak TIM1_BRK_IRQHandler

.thumb_set TIM1_BRK_IRQHandler,Default_Handler

 

.weak TIM1_UP_IRQHandler

.thumb_set TIM1_UP_IRQHandler,Default_Handler

 

.weak TIM1_TRG_COM_IRQHandler

.thumb_set TIM1_TRG_COM_IRQHandler,Default_Handler

 

.weak TIM1_CC_IRQHandler

.thumb_set TIM1_CC_IRQHandler,Default_Handler

 

.weak TIM2_IRQHandler

.thumb_set TIM2_IRQHandler,Default_Handler

 

.weak TIM3_IRQHandler

.thumb_set TIM3_IRQHandler,Default_Handler

 

.weak TIM4_IRQHandler

.thumb_set TIM4_IRQHandler,Default_Handler

 

.weak I2C1_EV_IRQHandler

.thumb_set I2C1_EV_IRQHandler,Default_Handler

 

.weak I2C1_ER_IRQHandler

.thumb_set I2C1_ER_IRQHandler,Default_Handler

 

.weak I2C2_EV_IRQHandler

.thumb_set I2C2_EV_IRQHandler,Default_Handler

 

.weak I2C2_ER_IRQHandler

.thumb_set I2C2_ER_IRQHandler,Default_Handler

 

.weak SPI1_IRQHandler

.thumb_set SPI1_IRQHandler,Default_Handler

 

.weak SPI2_IRQHandler

.thumb_set SPI2_IRQHandler,Default_Handler

 

.weak USART1_IRQHandler

.thumb_set USART1_IRQHandler,Default_Handler

 

.weak USART2_IRQHandler

.thumb_set USART2_IRQHandler,Default_Handler

 

.weak USART3_IRQHandler

.thumb_set USART3_IRQHandler,Default_Handler

 

.weak EXTI15_10_IRQHandler

.thumb_set EXTI15_10_IRQHandler,Default_Handler

 

.weak RTCAlarm_IRQHandler

.thumb_set RTCAlarm_IRQHandler,Default_Handler

 

.weak USBWakeUp_IRQHandler

.thumb_set USBWakeUp_IRQHandler,Default_Handler

 

/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/


如果有注释不准确的地方还希望能够给我指出,感谢感谢~~




这里以STM32程序运行过程为轴线,依次介绍单片机启动过程中的流程。


  在给单片机上电前,先通过给BOOT0和BOOT1连接高低电平来设置程序启动的初始位置。STM32程序启动的位置有三种:


  1:Main Flash memory


  是STM32内置的Flash,一般我们使用JTAG或者SWD模式下载程序时,就是下载到这个里面,重启后也直接从这启动程序。


  2:System memory


  从系统存储器启动,这种模式启动的程序功能是由厂家设置的。一般来说,这种启动方式用的比较少。


  3:Embedded Memory


  内置SRAM,既然是SRAM,自然也就没有程序存储的能力了,这个模式一般用于程序调试。


 


  在这三种方式里面用的最多的是第一种,下面以第一种启动方式为例进行介绍:


  在重启芯片时,SYSCLK的第4个上升沿,BOOT引脚的值将被锁存。然后进入启动文件,具体的启动文件为startup_stm32f10x_md.s。


  首先启动文件中有一个Reset_Handler,这个汇编函数的作用有两个:


   a.将FLASH中的程序代码拷贝到SRAM中


   b.将系统堆栈初始化(清0)


  然后进入main,从而进入c的世界。




但是启动文件到这里还没有结束,后面还有一个部分。


这个部分的作用是定义了一个段来存放中断向量表,然后以字的形式分别填入了中断的指针。


在这以后还有一个部分,后面这个部分的作用是给中断服务定义了一个弱别名,这个弱别名的作用就是在有中断被触发后,如果没有重写对应的弱别名,那么程序就默认执行默认的中断处理函数(Default_Handler),反之则执行重写了的中断处理函数


关键字:中容量  STM32  启动代码 引用地址:中容量STM32处理器启动代码的理解

上一篇:解决STM32 HardFault_Handler错误记录
下一篇:STM32硬件复位后无法启动

推荐阅读最新更新时间:2024-03-16 16:26

STM32 基础系列教程 18 – IWDG
前言 学习stm32 独立看门狗(IWDG)接口使用,学会用STM32内部独立看门狗(IWDG)实现程序异常时自复位功能。 STM32F10xxx内置两个看门狗,提供了更高的安全性、时间的精确性和使用的灵活性。两个看门狗设备(独立看门狗和窗口看门狗)可用来检测和解决由软件错误引起的故障;当计数器达到给定的超时值时,触发一个中断(仅适用于窗口型看门狗)或产生系统复位。独立看门狗(IWDG)由专用的低速时钟(LSI)驱动,即使主时钟发生故障它也仍然有效。窗口看门狗由从APB1时钟分频后得到的时钟驱动,通过可配置的时间窗口来检测应用程序非正常的过迟或过早的操作。 IWDG最适合应用于那些需要看门狗作为一个在主程序之外,能够完
[单片机]
<font color='red'>STM32</font> 基础系列教程 18 – IWDG
再造STM32---第六部分:自己写库—构建库函数雏形
本章参考资料: 《STM32F4xx 中文参考手册》 、 《STM32F429 规格书》 虽然我们上面用寄存器点亮了 LED,乍看一下好像代码也很简单,但是我们别侥幸以后就可以一直用寄存器开发。在用寄存器点亮 LED 的时候,我们会发现 STM32 的寄存器都是 32 位的,每次配置的时候都要对照着《STM32F4xx 参考手册》中寄存器的说明,然后根据说明对每个控制的寄存器位写入特定参数,因此在配置的时候非常容易出错,而且代码还很不好理解,不便于维护。所以学习 STM32 最好的方法是用软件库,然后在软件库的基础上了解底层,学习遍所有寄存器。 6.1 什么是 STM32 函数库: 以上所说的软件
[单片机]
再造STM32---第六部分:自己写库—构建库函数雏形
STM32 + GCC + J-Link :开源的力量
发现Keil自带的编译器有时候优化有问题,跑得跟没优化时完全不一样。以前使用GCC,现在自然而然想再次使用GCC。 搭配1:Keil+GCC+Ulink2 这类的介绍也较多,主要是一个链接脚本、启动代码、IDE选项配置的问题,大家可以google一下,个人觉得搭配一般般。 搭配2:emIDE+JLink 纯粹的Opensource血统,这个EMIDE已经带了GCC,只需要一个JLink,然后构建好工程就可以debug了,如果发展迅猛,估计Keil也会扛不住了。
[单片机]
STM32半主机模式
在keil中编程时常会遇到__use_no_semihosting_swi的警告,这时你就是进入了半主机模式。 在嵌入式的编程中你是避免不了使用printf、fopen、fclose等函数的但是因为嵌入式的程序中并没有对这些函数的底层实现,使得设备运行时会进入软件中断BAEB处,这时就需要__use_no_semihosting_swi这 个声明,使程序遇到这些文件操作函数时不停在此中断处,具体操作如下,将下列程序加入你的工程中: #pragma import(__use_no_semihosting_swi) #pragma import(_main_redirection) const char __stdin_name ;
[单片机]
STM32与FPGA通信写数据出错问题解决方法
项目中需要使用STM32和FPGA通信,使用的是地址线和数据线,在FPGA中根据STM32的读写模式A的时序完成写入和读取。之前的PCB设计中只使用了8跟数据线和8根地址线,调试过程中没有发现什么问题,在现在的PCB中使用了8根地址线和16根数据线,数据宽度也改成了16位,刚开始是读取数据不正确,后来发现了问题,STM32在16位数据宽度下有个内外地址映射的问题,只需要把FPGA中的设定的地址乘以2在STM32中访问就可以了,但是在写操作的时候会出现写当前地址的时候把后面的地址写成0的情况,比如说我给FPGA中定义的偏移地址0x01写一个16位数据,按照地址映射,在STM32中我把地址写入0x02,。实际测试发现这个地址上的数据是
[单片机]
关于STM32芯片的三种烧写方式对比
我们做STM32产品开发,最终是要将编写的程序代码写进芯片存储器,通常指Flash存储器【含可以映射到芯片存储空间的片外存储器】,让程序正常运行起来以实现相应的功能。 一般来讲,将我们准备好的机器代码HEX或BIN文件等写进片上FLASH可以有三种常用的烧写方式。【温馨提示:下面提到的烧写方式的术语或称谓不是绝对的,对于不同的器件可能有不同的表述甚至内涵,这里仅针对STM32,知道怎么回事就好】 第一种,ICP【In circuit programming】,即在电路编程或在线编程【注:不必太纠结该术语名字或称谓。平常称之为ICP也挺好】。一般是指利用调试器通过调试接口将程序代码写进芯片存储器的过程。调试器可以是ST官方的
[单片机]
关于<font color='red'>STM32</font>芯片的三种烧写方式对比
stm32专题二十一:LCD显示原理
显示器的基本参数: 1 像素 像素是组成图像的最基本单元要素,显示器的像素指它成像最小的点,即液晶原理中提到的一个显示单元。 2 分辨率 一些嵌入式设备的显示器常常以“行像素值 x 列像素值”表示屏幕的分辨率。如分辨率 800x480 表示该显示器的每一行有 800 个像素点,每一列有 480 个像素点,也可理解为有 800 列,480 行。 3 色彩深度 色彩深度指显示器的每个像素点能表示多少种颜色,一般用“位”(bit)来表示。如单色屏的每个像素点能表示亮或灭两种状态(即实际上能显示 2 种颜色),用 1 个数据位就可以表示像素点的所有状态,所以它的色彩深度为 1bit,其它常见的显示屏色深为16bit和 24bi
[单片机]
<font color='red'>stm32</font>专题二十一:LCD显示原理
ST推四套价格低廉STM32评估及开发工具 均由第三方开发
Hitex、IAR、Keil和Raisonance公司的评估及开发工具与ST的量产的ARM Cortex-M3内核微控制器同步上市 中国,2007年10月15日 — 意法半导体(纽约证券交易所代码:STM)今天推出四套价格低廉的评估及开发工具,这些工具是由ST的主要的第三方软件供应商开发的,支持ST最近推出的基于具有突破性的ARM Cortex-M3内核的STM32微控制器。这些开发入门工具的开发厂商分别是Hitex、IAR、Keil和Raisonance,通过使用这些开发工具,用户可以轻松了解STM32的特性和功能,只需很少的时间和资金就能开始正式的应用程序开发,这些开发工具的价位在109美元到239美元之间。 全系列产品
[新品]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
随便看看
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved