STM32
直播中

郝埃连

7年用户 1368经验值
私信 关注
[问答]

如何速查STM32F4外设配置?

mcu编程方式是什么?
如何速查STM32F4外设配置?

回帖(1)

李华瑞

2021-11-29 11:25:51
mcu编程方式



  • ICP(in circuit programming)

使用JTAG/SWD协议或bootloader下载用户应用程序到mcu


  • IAP(in application programming)

通过任意一种通信接口(IO口、USB、CAN、USART、I2C、SPI等)下载册灰姑娘徐或者应用数据到flash
stm32允许用户在应用程序中重新烧写flash中的内容
局限性:IAP至少需要有一部分程序已经使用ICP方式烧录到闪存存储器中,即需要有bootloader
作用:在不需要操作硬件平台的情况下实现远程升级
stm32系统存储器

所有stm32都带有系统存储器区域,ST在mcu出厂前已经将bootloader固化在mcu内部的系统存储器区域
系统存储器可以使用串口1(USART1)接收应用程序
stm32f4xx中系统存储器大小为30k,位于主存储区和OTP区域之间
将BOOT1=0,BOOT0=1即可从系统存储器区域加载
ICP/ISP下载流程



  • 通过串口1传输程序数据
  • 在ICP下,bootloader将其下载到0x8000000到0x0807FFFF之间的主存储器区域(在ISP下载中依靠JTAG/SWD协议直接写入)

IAP下载流程



  • ICP方式烧录bootloader区域代码
  • 将应用程序转换为二进制编码
  • 通过各种通信接口发送二进制文件到bootloader
  • bootloader接收程序并将其烧录在IAP应用程序存储区(主存储器)
  • bootloader跳转到程序区进行执行

一般程序执行过程(从startup.s启动)



  • 从0x08000000执行程序,同时从栈顶开始读取地址执行程序
  • 开栈、开堆、设置内存(共用4字节)
  • 开辟复位中断向量(中断向量表起始地址)——对应Reset_Handler地址
  • 跳转到复位中断程序入口Reset_Handler(void),执行相关程序
  • 相关程序包含systemInit和main函数入口SystemInit、__main,进入int main(void)
  • 执行主程序内容(主循环)
  • 当发生中断事件时,先进行现场保护,再到中断向量表中查找对应中断向量xxx_Handler——对应XXX_IRQHandler(void)函数地址
  • 执行中断服务函数内容
  • 执行完毕后,先恢复现场,再加载入主函数

加入IAP后程序执行过程

1-5同上


  • 到主函数后,进行IAP过程
  • IAP程序完毕后,跳回新的复位中断向量地址(0x08000004+n+m,n为相对IAP程序main函数入口偏移地址,m为相对bootloader复位中断向量偏移地址)
  • 在新程序中重复上述一般程序执行步骤
  • 中断执行时,跳转到原来的中断向量表,不依据新的中断向量表进行跳转
  • 执行完毕后,恢复现场并加载入新程序主函数

IAP升级应用程序过程



  • 检查是否需要对实际应用程序代码进行更新
  • 如果不需要,则跳转到实际应用部分代码执行
  • 如果需要,则执行更新操作后再跳转到实际应用部分代码执行

将第一部分的原始代码称为bootloader程序,第二部分代码称为APP,一般从最低地址存放bootloader,APP紧跟其后
stm32的APP不仅可以放到flash里执行,也可以放入SRAM里执行,两者原理一致

Stack_Size      EQU     0x00000400
                AREA    STACK, NOINIT, READWRITE, ALIGN=3
Stack_Mem       SPACE   Stack_Size
__initial_sp
; Heap Configuration
;     Heap Size (in Bytes) <0x0-0xFFFFFFFF:8>
;

Heap_Size       EQU     0x00000200
                AREA    HEAP, NOINIT, READWRITE, ALIGN=3
__heap_base
Heap_Mem        SPACE   Heap_Size
__heap_limit
                PRESERVE8
                THUMB
;中断向量表
; Vector Table Mapped to Address 0 at Reset
                AREA    RESET, DATA, READONLY
                EXPORT  __Vectors
                EXPORT  __Vectors_End
                EXPORT  __Vectors_Size
__Vectors       DCD     __initial_sp               ; Top of Stack
                DCD     Reset_Handler              ; Reset Handler复位中断向量
                DCD     NMI_Handler                ; NMI Handler非可屏蔽中断向量
                DCD     HardFault_Handler          ; Hard Fault Handler
                DCD     MemManage_Handler          ; MPU Fault Handler
                DCD     BusFault_Handler           ; Bus Fault Handler
                DCD     UsageFault_Handler         ; Usage Fault Handler
                DCD     0                          ; Reserved
                DCD     0                          ; Reserved
                DCD     0                          ; Reserved
                DCD     0                          ; Reserved
                DCD     SVC_Handler                ; SVCall Handler
                DCD     DebugMon_Handler           ; Debug Monitor Handler
                DCD     0                          ; Reserved
                DCD     PendSV_Handler             ; PendSV Handler
                DCD     SysTick_Handler            ; SysTick Handler
                ; External Interrupts
                DCD     WWDG_IRQHandler                   ; Window WatchDog                       
                DCD     PVD_IRQHandler                    ; PVD through EXTI Line detection
                DCD     TAMP_STAMP_IRQHandler             ; Tamper and TimeStamps through the EXTI line
                DCD     RTC_WKUP_IRQHandler               ; RTC Wakeup through the EXTI line
                DCD     FLASH_IRQHandler                  ; FLASH
                DCD     RCC_IRQHandler                    ; RCC
                DCD     EXTI0_IRQHandler                  ; EXTI Line0
                DCD     EXTI1_IRQHandler                  ; EXTI Line1
                DCD     EXTI2_IRQHandler                  ; EXTI Line2
                DCD     EXTI3_IRQHandler                  ; EXTI Line3
                DCD     EXTI4_IRQHandler                  ; EXTI Line4
                DCD     DMA1_Stream0_IRQHandler           ; DMA1 Stream 0
                DCD     DMA1_Stream1_IRQHandler           ; DMA1 Stream 1
                DCD     DMA1_Stream2_IRQHandler           ; DMA1 Stream 2
                DCD     DMA1_Stream3_IRQHandler           ; DMA1 Stream 3
                DCD     DMA1_Stream4_IRQHandler           ; DMA1 Stream 4
                DCD     DMA1_Stream5_IRQHandler           ; DMA1 Stream 5
                DCD     DMA1_Stream6_IRQHandler           ; DMA1 Stream 6
                DCD     ADC_IRQHandler                    ; ADC1, ADC2 and ADC3s
                DCD     CAN1_TX_IRQHandler                ; CAN1 TX
                DCD     CAN1_RX0_IRQHandler               ; CAN1 RX0
                DCD     CAN1_RX1_IRQHandler               ; CAN1 RX1
                DCD     CAN1_SCE_IRQHandler               ; CAN1 SCE
                DCD     EXTI9_5_IRQHandler                ; External Line[9:5]s
                DCD     TIM1_BRK_TIM9_IRQHandler          ; TIM1 Break and TIM9                  
                DCD     TIM1_UP_TIM10_IRQHandler          ; TIM1 Update and TIM10                 
                DCD     TIM1_TRG_COM_TIM11_IRQHandler     ; TIM1 Trigger and Commutation and TIM11
                DCD     TIM1_CC_IRQHandler                ; TIM1 Capture Compare
                DCD     TIM2_IRQHandler                   ; TIM2
                DCD     TIM3_IRQHandler                   ; TIM3
                DCD     TIM4_IRQHandler                   ; TIM4
                DCD     I2C1_EV_IRQHandler                ; I2C1 Event
                DCD     I2C1_ER_IRQHandler                ; I2C1 Error
                DCD     I2C2_EV_IRQHandler                ; I2C2 Event
                DCD     I2C2_ER_IRQHandler                ; I2C2 Error
                DCD     SPI1_IRQHandler                   ; SPI1
                DCD     SPI2_IRQHandler                   ; SPI2
                DCD     USART1_IRQHandler                 ; USART1
                DCD     USART2_IRQHandler                 ; USART2
                DCD     USART3_IRQHandler                 ; USART3
                DCD     EXTI15_10_IRQHandler              ; External Line[15:10]s
                DCD     RTC_Alarm_IRQHandler              ; RTC Alarm (A and B) through EXTI Line
                DCD     OTG_FS_WKUP_IRQHandler            ; USB OTG FS Wakeup through EXTI line
                DCD     TIM8_BRK_TIM12_IRQHandler         ; TIM8 Break and TIM12                  
                DCD     TIM8_UP_TIM13_IRQHandler          ; TIM8 Update and TIM13                 
                DCD     TIM8_TRG_COM_TIM14_IRQHandler     ; TIM8 Trigger and Commutation and TIM14
                DCD     TIM8_CC_IRQHandler                ; TIM8 Capture Compare
                DCD     DMA1_Stream7_IRQHandler           ; DMA1 Stream7
                DCD     FSMC_IRQHandler                   ; FSMC
                DCD     SDIO_IRQHandler                   ; SDIO
                DCD     TIM5_IRQHandler                   ; TIM5
                DCD     SPI3_IRQHandler                   ; SPI3
                DCD     UART4_IRQHandler                  ; UART4
                DCD     UART5_IRQHandler                  ; UART5
                DCD     TIM6_DAC_IRQHandler               ; TIM6 and DAC1&2 underrun errors
                DCD     TIM7_IRQHandler                   ; TIM7                  
                DCD     DMA2_Stream0_IRQHandler           ; DMA2 Stream 0
                DCD     DMA2_Stream1_IRQHandler           ; DMA2 Stream 1
                DCD     DMA2_Stream2_IRQHandler           ; DMA2 Stream 2
                DCD     DMA2_Stream3_IRQHandler           ; DMA2 Stream 3
                DCD     DMA2_Stream4_IRQHandler           ; DMA2 Stream 4
                DCD     ETH_IRQHandler                    ; Ethernet
                DCD     ETH_WKUP_IRQHandler               ; Ethernet Wakeup through EXTI line
                DCD     CAN2_TX_IRQHandler                ; CAN2 TX
                DCD     CAN2_RX0_IRQHandler               ; CAN2 RX0
                DCD     CAN2_RX1_IRQHandler               ; CAN2 RX1
                DCD     CAN2_SCE_IRQHandler               ; CAN2 SCE
                DCD     OTG_FS_IRQHandler                 ; USB OTG FS
                DCD     DMA2_Stream5_IRQHandler           ; DMA2 Stream 5
                DCD     DMA2_Stream6_IRQHandler           ; DMA2 Stream 6
                DCD     DMA2_Stream7_IRQHandler           ; DMA2 Stream 7
                DCD     USART6_IRQHandler                 ; USART6
                DCD     I2C3_EV_IRQHandler                ; I2C3 event
                DCD     I2C3_ER_IRQHandler                ; I2C3 error
                DCD     OTG_HS_EP1_OUT_IRQHandler         ; USB OTG HS End Point 1 Out
                DCD     OTG_HS_EP1_IN_IRQHandler          ; USB OTG HS End Point 1 In
                DCD     OTG_HS_WKUP_IRQHandler            ; USB OTG HS Wakeup through EXTI
                DCD     OTG_HS_IRQHandler                 ; USB OTG HS
                DCD     DCMI_IRQHandler                   ; DCMI
                DCD     CRYP_IRQHandler                   ; CRYP crypto
                DCD     HASH_RNG_IRQHandler               ; Hash and Rng
                DCD     FPU_IRQHandler                    ; FPU
__Vectors_End


__Vectors_Size  EQU  __Vectors_End - __Vectors
                AREA    |.text|, CODE, READONLY
; Reset handler
Reset_Handler    PROC
                 EXPORT  Reset_Handler             [WEAK]
        IMPORT  SystemInit
        IMPORT  __main
                 LDR     R0, =SystemInit
                 BLX     R0
                 LDR     R0, =__main
                 BX      R0
                 ENDP
; Dummy Exception Handlers (infinite loops which can be modified)
NMI_Handler     PROC
                EXPORT  NMI_Handler                [WEAK]
                B       .
                ENDP
HardFault_Handler
                PROC
                EXPORT  HardFault_Handler          [WEAK]
                B       .
                ENDP
MemManage_Handler
                PROC
                EXPORT  MemManage_Handler          [WEAK]
                B       .
                ENDP
BusFault_Handler
                PROC
                EXPORT  BusFault_Handler           [WEAK]
                B       .
                ENDP
UsageFault_Handler
                PROC
                EXPORT  UsageFault_Handler         [WEAK]
                B       .
                ENDP
SVC_Handler     PROC
                EXPORT  SVC_Handler                [WEAK]
                B       .
                ENDP
DebugMon_Handler
                PROC
                EXPORT  DebugMon_Handler           [WEAK]
                B       .
                ENDP
PendSV_Handler  PROC
                EXPORT  PendSV_Handler             [WEAK]
                B       .
                ENDP
SysTick_Handler PROC
                EXPORT  SysTick_Handler            [WEAK]
                B.
                ENDP


Default_Handler PROC
                EXPORT  WWDG_IRQHandler                   [WEAK]
;后面都是类似的 EXPORT xxx_IRQHandler [WEAK] 中断函数名声明
;此处省略中断向量名
                B.
                ENDP
                ALIGN
;*******************************************************************************
; User Stack and Heap initialization
;*******************************************************************************
                 IF      :DEF:__MICROLIB
                 EXPORT  __initial_sp
                 EXPORT  __heap_base
                 EXPORT  __heap_limit
                 ELSE
                 IMPORT  __use_two_region_memory
                 EXPORT  __user_initial_stackheap
__user_initial_stackheap
                 LDR     R0, =  Heap_Mem
                 LDR     R1, =(Stack_Mem + Stack_Size)
                 LDR     R2, = (Heap_Mem +  Heap_Size)
                 LDR     R3, = Stack_Mem
                 BX      LR
                 ALIGN
                 ENDIF
                 END
IAP的实现
IAP程序的要求:1.APP必须在bootloader之后某个偏移量为x的地址开始;2.必须将新程序的中断向量表相应移动,移动的偏移量为x


APP程序的生成步骤
设置APP的起始地址和储存空间大小
设置中断向量表偏移量(通过SCB->VTOR寄存器进行设置)
在APP的main()函数内写入


SCB->VTOR=FLASH_BASE|0x10000;
1
来设置偏移地址


设置MDK编译后运行fromelf.exe,生成对应二进制文件(通过设置MDK User选项卡为“编译后调用fromelf.exe,根据axf文件生成bin文件,用于IAP更新”)
设置以下指令


${workspace}/bin/fromelf.exe --bin -o ../OBJ/${APP}.bin ../OBJ/${APP}.axf
1
到User选项卡中"After Build/Rebuild"一项


编译后显示"After Build -User command xxxxxx"且无报错可判断为生成成功bin文件


下载并写入APP程序
IAP-Bootloader的具体代码实现
选择通信接口
选择写入地址
设置跳转
注意编写代码前引入stm32f4xx_flash.c源文件、stmflash.c文件和对应头文件


usart.c


//仅列出改动内容
#define USART_REC_LEN 120*1024 //120k缓存


//注意这里设置为全局变量
u8 USART_RX_BUF[USART_REC_LEN] __attribute__ (at(0x20001000));//将数据存放在SRAM起始地址为0x20001000的位置


void USART1_IRQHandler(void)
{
    u8 res;
   
    if(USART_GetITStatus(USART1,USART_IT_RXNE)!=RESET)//接收中断
    {
            res=USART_ReceiveData(USART1);//读取接收到的数据   
        if(USART_RX_CNT         {
            USART_RX_BUF[USART_RX_CNT]=res;//将数据保存在缓存
            USART_RX_CNT++;
        }
    }
}


加入IAP文件夹与iap.c、iap.h文件


iap.h


#ifndef __IAP_H
        #define __IAP_H
        #include "sys.h"
        typedef void (*iapfun)(void);//定义一个函数类型的参数
               
        #define FLASH_APP1_ADDR 0x0801000 //第一个应用程序起始地址(存放在FLASH)
        //保留0x08000000-0x0800FFFF的64k空间给bootloader
       
        void IAP_load_app(u32 appxaddr);//跳转到APP执行
        void IAP_write_appbin(u32 appxaddr,u8* appbuf,u32 applen);//在指定地址开始处写入APP二进制文件       
#endif


iap.c


#include "iap.h"
#include "stmflash.h"


IAPfun jmp2app;
u32 IAPbuf[512];//2k字节缓存


//写入APP到FLASH
//appxaddr APP起始地址
//appbuf APP代码缓存
//applen APP大小
void IAP_write_appbin(u32 appxaddr,u8* appbuf,u32 applen)
{
        u32 t;
        u32 temp;
        u32 fwaddr=appxaddr;
        u16 i=0;
        u8* dfu=appbuf;


        for(t=0;t         {
                temp=(u32)dfu[3]<<24;
                temp|=(u32)dfu[2]<<16;
                temp|=(u32)dfu[1]<<8;
                temp|=(u32)dfu[0];//每8个字节封装成32位(8个字节)
               
                dfu+=4;//偏移4个字节
                IAPbuf[i++]=temp;//将封装后的数据传给写入缓存
               
                if(i==512)//每满足512个字节写入一次
                {
                        i=0;
                        STMFLASH_write(fwaddr,IAPbuf,512);
                        fwaddr+=2048;//fwaddr偏移512*4=2048位数据,进行下一轮操作直到将接收数据全部写入
                }
        }
        if(i)//如果最后有剩余
                STMFLASH_write(fwaddr,IAPbuf,i);//将剩余的内容字节写入FLASH
}


//跳转到APP段
//appxaddr APP代码起始地址
void IAP_load_app(u32 appxaddr)
{
        if(((*(vu32*)appxaddr)&0x2FFE0000)==0x20000000)//检查栈顶地址是否合法
        {
                jmp2app=(IAPfun)*(vu32*)(appxaddr+4);//用户代码区第二个字为程序开始地址(复位地址)
               
                /*这里使用了汇编程序MSR_MSP设置堆栈*/
                MSR_MSP(*(vu32*)appxaddr);//初始化APP堆栈指针
                jmp2app();//跳转到APP段
        }
}


使用方法
烧录bootloader至stm32
打开串口
使用串口助手将文件发送王mcu
等待固件更新
开始程序
举报

更多回帖

发帖
×
20
完善资料,
赚取积分