STM32的bootloader程序设计
7 c( X- y" A- H6 }0 I3 e6 _" `通常在使用MCU进行开发设计的电子产品中,只存在固件的概念,并没有所谓的 Boot Loader。其实在此类设备中,同样是可以存在 Bootloader的。正因为有Bootloader,所以不需要拆机就能对产品进行固件升级。万一产品固件有缺陷,那么利用 Bootloader进行升级非常方便,非常节省精力和成本,因此越来越多的基于; [- b/ I* ~8 a0 _# m
MCU的产品也开始使用 Bootloader.
& S8 p- W7 j v( @/ \2 u2 o5 E8 N8 f3 o3 R0 X3 L8 y, R/ F& T
要实现这种设计会用到IAP技术。IAP是 In Application Programming(在应用中编程)。一般情况下芯片的代码区只有一个用户程序。而在IAP方案下将代码区划分为两部分,各存放一个程序,一个为 Bootloader(引导加载程序),另一个为user application(用户应用程序)。 Bootloader在产品出厂时就固定下来了,当需要变更 user application I时,只需要通过触发 Bootloader对 user application的擦除和重新写入,就可完成对用户应用程序的升级
) h( R( S# e5 V! d* }2 D0 Z
+ l. p* j8 m" u/ D7 i, F! [0 U9 ` W6 B
: z8 `9 @0 q4 u* l" U实现的机制如下。' l/ `# o6 C4 s. Z. d
当复位发生时,开始执行 Bootloader。 Bootloader会检测一个特定的条件,比如
+ r- h. y$ x6 Q& g: F说按下组合键。当条件触发时, Bootloader会进行一个分支判断,以更新用户应用程6 z: ?6 i+ G3 x/ @* A
序或者直接执行用户应用程序。8 r, _, x1 h$ H3 |8 E3 B
# M1 Y% z" \3 v) O. @. G
用户应用程序需要和 Bootl oader分开。最常见的做法是将 Bootloader放置在存储 Flash中的最开始处,并将用户应用程序放置在其后的可用的存储 Flash中;这样就可以独立配置两块区域的内存保护。7 W9 i, z$ g7 ^
2 _- Z* _; X% f
boot 工程配置截图如下
3 o3 t# e; c4 ]; N) b3 o- M
+ T, V7 [1 ^2 K
# Z2 A7 t d9 l7 I- e9 `' r
# O [+ Y+ O7 U/ G0 Z, o* g% z程序流程图8 w" C% p! k- h4 |8 c9 x* j
9 ^6 @3 g C) O* j6 R% H h
# c( g# o5 t) x
4 n( b" `6 ?) X- int main(void)
7 i# e& Y3 `9 F7 h5 z0 { - {* a' N) Y& ?" K! t) f+ v
- /* H7 V1.6.0版本 Hal库 2020-03-04更新 */
9 P+ b5 _& i( P) i. R - HAL_Init();0 ]+ k+ H. x; n* L$ O
: e3 d& O* y V X+ a0 o3 O- /* 系统时钟 */( Z) L: r0 S4 C% A8 z7 @ u n$ s" G
- SystemClock_Config();: {7 x9 M* Q) m, d
- 4 V4 o2 A2 c2 _% u! A: R; h; ^
- /* GPIO初始化 */
& V0 ]0 b7 V3 r! b% D- W1 v" d - MX_GPIO_Init();' s8 M: G/ c5 F/ B. H3 |8 W. w/ v
- - ?6 q/ M) \6 ]4 f" }
- /* 打印串口LPUART */
a7 |5 h, p! f# c8 n - MX_LPUART1_UART_Init();
+ R' @/ }' U! K - : X' {$ G, f- L# f1 B
- /* 读取应用程序 */
3 R6 b7 P2 v1 l - ReadAppPro();
, V& [! o* _) Y2 U1 h! s8 M - 5 p' ~$ j `9 l* a* H
- /* 跳转到应用程序区 -> 0x08020000(512K) */ s; v' M. ] J; F% N- r+ Z
- Quit_IAP_Process();# O9 x4 c# g9 O, u2 U, Y
- ( L$ y7 s0 H2 c
- }
复制代码
& H4 H* ^% P* j7 \, }ReadAppPro();
7 `5 h# ]5 S. `- void ReadAppPro(void)
4 R3 R7 R0 M. c - {6 z `, \' s4 d' Z1 M. s, y$ u8 l
- updateInfo_TypeDef updateInfo; /*升级信息参数*/: v0 O' c: {% o* k a2 E
- uint32_t NbOfSectors = 0; /*占用Flash扇区数*/
* H" Z' F% Y/ }8 w$ o$ r5 B/ F - uint32_t count = 0; /*分包写入个数*/
9 S2 ?4 y; a* j6 g l4 g4 `- V - uint32_t i = 0;
9 |2 ^. i" I9 ^ K - uint32_t calcCRC = 0;2 x. Z! p- q, k& c4 T, E* _
- uint32_t programCRC = 0;
% U2 g( e0 f8 K1 B, u6 P3 D - uint32_t fileLen = 0;. E4 {* y/ p- {8 L& o
- uint8_t *papp_data = (uint8_t *)(0x24000000);/* 内存起始位置 */ q1 r0 u( j& x# O
- $ C+ |" E; q! I: C' j) Y$ W
- /* Step1: 读取升级信息 */
/ t1 Z1 f. Q9 y6 Q, i/ T7 i* | - ReadUpdateInfo(&updateInfo);
. @/ I4 k) m% x* ?- h
2 M# w$ x" c3 Z/ | z* z- /* Step2: 判断是否需要升级 */5 I2 ^0 p( c0 X# t5 k* m
- if (1 == updateInfo.status)
1 b& C) T* H+ t' S* s! l - { V# n8 J' K6 j! \
- /* 获取升级文件信息长度 */* I/ o6 D# s1 [6 e) w: u& C
- fileLen = updateInfo.len; 0 r$ [0 t3 c; y4 Q7 D
- /* 占用Flash扇区数计算 */ [4 n5 u# U# r2 R! C
- if (0 == updateInfo.len % FLASH_ONE_SECTOR_BYTES)
" q9 i& u7 B9 v8 W - {
. y" z" }! M, ?, D, ?; D. p - NbOfSectors = updateInfo.len / FLASH_ONE_SECTOR_BYTES;
. q! A8 W" f3 f Q2 b - }" y K6 s+ G* S1 a* u+ s9 c) x
- else2 F$ W0 N8 F" u, h+ x8 b" t
- {
8 a) a5 M# B; x- H6 s+ e - NbOfSectors = updateInfo.len / FLASH_ONE_SECTOR_BYTES + 1;( i. `7 p2 `+ d% ]
- }
" W9 \- Z! }$ E3 V% o - + z0 d/ e/ F$ S4 ~0 r
- /* Step3: 读取升级程序 */
6 B6 W' r& l! @$ l* U/ V# a - /* 分段段数计算 */8 S+ f# j) A! L( i9 w/ \
- count = (updateInfo.len%4096)?(updateInfo.len/4096+1):(updateInfo.len/4096);# W' s1 |9 r- Q) e2 G% K
- for(i = 0; i < count; i ++)
" X+ }$ F5 M- B. l# Q3 q6 U1 J" K - {
( k. o: I! q' D& x& G - /* 程序数据从0x24000000开始,写入到APP_MAIN_ADDR 区域 */+ `) B( }. q# s1 D& ]& ^
- (void)FlashDataRead(APP_MAIN_ADDR+i*4096,papp_data+i*4096,4096);' i; b! v% P2 G, e/ S
- }0 Q- d. [4 U+ L# d. B1 e: N4 G7 ~
( y8 [9 B* g$ J5 p- /* Step4: check crc */& f; D" k9 W! K5 F3 W+ c9 {
- calcCRC = ComputeCRC32(papp_data,fileLen-4);
& t! E# D# @- ?$ U4 ?4 o" {( p - programCRC = *(uint32_t *)(&papp_data[fileLen-4]);/ V# d+ F! O* Q7 k5 }8 X& p
- ; k2 n0 V6 i2 T W+ E# t
- /* 如果CRC计算通过 */& i2 x2 ]) X8 z# Y" A$ L9 K
- if(programCRC == calcCRC)
" v+ t- A _" m - {
t" B3 B z* E% U - /* Step5: 需要升级时,擦除APP区 */4 I) ^" a# A/ e0 s; F U
- (void)FlashErase(APPLICATION_ADDRESS , NbOfSectors);1 V" H& `4 {& y+ R$ O, ]
- for(i = 0; i < count; i ++)
- v- S; P% r2 J- n3 t3 X - {
! N& d1 T4 Z& g( J' @6 u8 ? P+ z - /* 程序数据从0x24000000开始,写入APPLICATION_ADDRESS 区域 */+ {) E- S- p9 _5 L
- (void)FlashDataWrite(APPLICATION_ADDRESS+i*4096,papp_data+i*4096,4096);
( C) T: ]$ X) m! ] - }
$ |/ Q. g9 @( y# e - /* check:写入程序之后,立即再校验,防止出差 */
, T( l7 x: @+ q: Z, W3 D - uint8_t *pflash_data = (uint8_t *)(APPLICATION_ADDRESS);
, a9 t3 ^! B# k& P - calcCRC = ComputeCRC32(pflash_data, fileLen-4);
7 v( U8 c8 Z1 y& D - if(programCRC != calcCRC){$ e! r- O; f4 _9 ?' w
- printf("app flash data write err\r\n");
8 a+ O1 } W& Y3 W$ c$ l# a - return;
( d# I3 L& U# s7 u% {; w- x0 D, X3 U+ V - }
5 N' W. j% h& [- p( P5 D! P - }
% o( H3 ?8 J! A3 l: l - else
; ^# a* x3 e: ^+ S( f - {0 |: @# Y; Z; D1 e! i, u7 Q# W
- printf("APP_MAIN_ADDR CRC err: programCRC 0x%x, calcCRC 0x%x\r\n", programCRC, calcCRC);7 t/ I+ c) Q/ Q- v/ j" f
- }
; T' l8 y# C: u" L
% h+ p/ R0 Q5 c0 R3 ~" g) N- /* Step5: 升级完成之后,改变升级信息 */ : G# e7 [# w5 V8 w; Q- B, p
- memset(&updateInfo, 0 , sizeof(updateInfo_TypeDef)) ; # e& n+ b) j0 E+ @
- memcpy(gTempBuf, &updateInfo, sizeof(updateInfo_TypeDef));
) t' h+ L: L4 Z/ j& J+ u - (void)FlashEraseAndWrite(APP_UPDATA_INFO_ADDR, 1, (uint8_t *)gTempBuf, sizeof(updateInfo_TypeDef));- R4 n5 A \" r/ X! w. V
- }* p9 H3 B' X- y6 Z7 x
- ! l5 W0 a3 ?$ g3 M% G5 z& I
- return;' P0 X2 [% V7 l; |- _( O
4 p( d" ]0 q1 ]- }
复制代码
$ I$ d1 C% d0 P% D- r+ hQuit_IAP_Process(); 跳转到应用程序区
" R3 v* [; E1 }+ {- f2 J8 O- void Quit_IAP_Process(void)
: V5 q) z6 V+ u7 S& }* U - {9 s" }( ?0 e u% \2 q+ [, }" _
- /*
8 {0 B+ m7 _7 H - * 把用户代码的首地址里面的数据拿出来,看看是不是以0x24000000开头。9 }) | R& ^8 P3 O2 {3 T
- * (X & 0x2FFE0000 ) == 0x24000000意思是说X是不是在0x24000000与24080000之间,即栈顶是不是在以0x24000000开始的512K
" i2 R3 P2 l2 k5 c- f$ {5 P9 \ - * 前面说了用户代码的首地址放的是堆栈指针,而堆栈应该指向RAM,而RAM的首地址是0x24000000,
# ^% I! i" t4 m - * 所以这句话用来判断是不是已经有一个正确的堆栈指针地址写在了用户代码的首地址,& T9 l; n$ S% k, b
- * 进一步可以推测是不是有一个正确的用户代码写在了用户代码区,如果已经写入了,就可以跳过去运行,如果没有写入,就不跳过去。
/ u7 H4 `. x# r; I4 } - */5 j# R4 R( @( d n) G& U
- __disable_irq();
@: a: ^/ h1 C! n - JumpAddress = *(__IO uint32_t*)APPLICATION_ADDRESS; /* APPLICATION_ADDRESS is 0x08020000,app 程序的跳转起始位置*/
L- `' I6 s! |6 a, I; n* u - if ( ((JumpAddress&0x2FFE0000)==0x24000000) || ((JumpAddress&0x2FF80000)==0x24000000))
2 ]7 W7 q% H: y* o8 v# {" j( P g - {$ L* Y1 O% I* [9 X
- #ifdef PRINT_DEBUG
' Q* o6 m0 f- f - printf("Start Jump IAP!\r\n");
9 N; O) b% ? K$ [; ]' v - #endif! W( L+ y! l6 N% G
- /* 是把用户代码的复位地址赋值给JumpAddress。 */ Q! X+ q5 v/ ]3 X* M
- JumpAddress = *(__IO uint32_t*) (APPLICATION_ADDRESS + 4);$ i! o6 s6 \2 V3 L& v
- /* 是把用户代码的堆栈地址写入堆栈指针 */2 r! S* f) [8 o1 u8 g% B7 `" X
- Jump_To_Application = (pFunction) JumpAddress;% }" z" p& ?; H# B" O
- __set_MSP(*(__IO uint32_t*) APPLICATION_ADDRESS);$ ^, B: Q- R; o
- /* 是把用户代码的复位地址付给PC指针 */
/ h1 i: @5 O: G0 X. o9 z# X1 W$ a+ U - Jump_To_Application();
. q- B2 @4 p3 `, Z$ M" A, t0 y - }
1 u* |& {9 E3 v; n' I; T" D - ( w; y. t9 _5 A! A1 l* a. f) [6 s* v
- file_size=0xFFFFFF;
, w( w: o, w: A - while(file_size--)
7 ^% k/ K6 N! X" Z& S5 u - {
* ~$ h* Q' y: X5 L; u - __nop();
" x0 Y j$ o" q \. t4 E" V0 d - }
7 U2 O; T8 ~9 s/ r- w( n4 A - + h! e9 ~/ S H" y: e
- NVIC_SystemReset();2 v9 A$ T) w6 ?' l1 X6 x1 z1 c* `
& q! w# N- B1 c9 N: }: ~- }
复制代码 + K2 v0 @1 u8 W. B8 }4 ]% K
问题1:, \4 L4 v- n$ Z! l6 z. }5 n( V
P7 r. V5 {/ a1 M+ Z' y X
JumpAddress 地址为什么指向 APPLICATION_ADDRESS + 4 ;7 V& {) {6 V( c- i( G
MSP 主堆栈指针为什么指向 APPLICATION_ADDRESS ;1 H0 S" N. o$ q" o) b
1 l1 j- ~! B- B- J从 startup_stm32fxxx.s 中的启动代码可以看出:程序开始第一条指令地址为 CSTACK,第二8 s8 b6 \; \/ n _3 B5 \
条指令地址为复位指令 ,参考代码中的红色部分(蓝色为注释);9 q' T& |( w" O( C+ [8 h
4 P# \8 r& l7 w* O6 n/ o__Vectors DCD __initial_sp ; Top of Stack ( APPLICATION_ADDRESS )2 r1 w6 |% k+ `; }$ k; \
DCD Reset_Handler ; Reset Handler ( APPLICATION_ADDRESS + 4)/ b6 A: \; j/ V0 X5 d( q8 d ?+ H( H
当程序启动时首先要执行复位程序,因此 JumpAddress 地址指向复位指令地址(即APPLICATION_ADDRESS + 4) ;# y+ \: k4 G% V7 G
MSP 对应的是主堆栈指针,指向 __initial_sp 地址(即 APPLICATION_ADDRESS).+ S, \. _* v! L# q
* [6 f+ ]2 K- V, e: X# p
问题2:- _6 }0 R4 W7 k) ` } G( }
$ L# a: x" O% |! R' }为什么需要做以下判断 : p( K% V. n$ B2 A q+ v
' I0 h0 ]5 j% P5 X) ?6 gif (((JumpAddress&0x2FFE0000)==0x24000000) || ((JumpAddress&0x2FF80000)==0x24000000)) {4 W+ [* @0 ?: y9 F
$ v: s' c: K9 ~: e* P
}
& l; g" e% O6 D3 t: j3 [% }/ UApplicationAddress 存放的是程序的主堆栈地址,__initial_sp堆栈地址指向 RAM,而 RAM 的
9 g. s4 j- |" z# E% N起始地址是 0x24000000;
8 ~1 \, f" W5 Q因此上面的判断语句执行:判断用户代码的堆栈地址是否落在:0x24000000~0x24080000区间5 { x+ U6 x9 P" F. U9 r
中,这个区间的大小为 512K.
9 O; {+ i( s3 U6 ?& A例程中使用芯片 RAM=512K,因此做上面的判断;如果芯片 RAM 比 512K 大的话,可以在
3 x; @* `. _3 \9 X- {- d; n此判断语句做调整! d+ Y6 c' ^% L. S* f! A0 F6 E: [
* y! S/ j1 z1 |/ `/ D# Y4 y r6 ~
以上分析可以看出,程序跳转部分的设计与 APP 启动代码和芯片 RAM 的大小有关系;移植和
; ^9 m; x3 V* f' ~3 {* i8 U调试时需加以注意+ M/ e( U) V7 N( e# g3 z
, t& x. l: U7 u) b1 v9 P9 XAPP 程序.
$ G" \/ R8 J0 x; I2 \7 uAPP 程序起始地址设置方法' u" s# L: W3 @& s4 f
a- X& g0 K0 Q% ^: l/ D! b1 _2 M5 e; v4 q& `+ U
$ G8 Q2 V1 p; z& ~中断向量表的偏移量设置方法+ U" s. B9 \, j( j% Z
system_stm32h7xx.c, |8 ?3 F' v" A9 @) C: [
7 F$ a6 a9 o& s' R- /** @addtogroup STM32H7xx_System_Private_Defines {4 m1 z/ \0 @& i
- * @{$ s8 X! j7 Z; B$ B+ R" U7 U
- */
# U1 s, d& {6 K& C8 \% ?" M6 O
% M. u& q! u7 i- /************************* Miscellaneous Configuration ************************/* F, l& V; X: T0 c9 A* F+ J5 ~
- /*!< Uncomment the following line if you need to use initialized data in D2 domain SRAM (AHB SRAM) */) Q1 M2 |9 D' I) d f& h
- /* #define DATA_IN_D2_SRAM */
, E9 H. @4 P& U5 J - #define DATA_IN_D2_SRAM
5 i. l8 [- B: Z* G) X
! w7 _0 f3 M, G- /*!< Uncomment the following line if you need to relocate your vector Table in: ]1 \ U0 Q# d) Q
- Internal SRAM. */5 X3 Z6 C$ w" X
- /* #define VECT_TAB_SRAM */" l" u. P# ~' k9 P" S( R; ^
- #define VECT_TAB_OFFSET 0x20000UL /*!< Vector Table base offset field. This value must be a multiple of 0x200. */
0 l l# ?- j5 \% \! q( f) w9 p
/ s6 ~/ T5 \* O' [2 z- //#define VECT_TAB_OFFSET 0x00000UL
9 y6 y! v, a4 w( _( v - /******************************************************************************/
复制代码
. h$ x# ]$ o4 ^1 w7 L5 h3 c. K/ A7 j6 q1 M2 h
STM32H743 复位后,还是从 0X08000004 地址取出复位中断向量的地址,并跳转到复位中断服务程序在运行完复位中断服务程序之后跳转到 IAP 的 main 函数,在执行完 IAP 以后(即将新的 APP 代码写入 STM32H743 的 FLASH,新程序的复位中断向量起始地址为 0X08000004+N+M),跳转至新写入程序的复位向量表,取出新程序的复位中断向量的地址,并跳转执行新程序的复位中断服务程序,随后跳转至新程序的 main 函数,并且注意到此时 STM32H743 的 FLASH,在不同位置上,共有两个中断向量表
9 q% ~" M; s w/ C( W7 ?3 O, m: C5 F7 k' o
通过以上两个过程的分析,知道 IAP 程序必须满足两个要求:
' H5 l5 x. u, y* X# g1) 新程序必须在 IAP 程序之后的某个偏移量为 x 的地址开始;
* j0 o/ G: c$ z. f- B& N+ t2) 必须将新程序的中断向量表相应的移动,移动的偏移量为 x
9 T* K N5 _1 d3 Q0 G/ {7 M$ ?
' u- e- Y' m# v% R8 l% H) S6 r9 H/ i6 o l G( \
|