
STM32系列的芯片里都有个特别的存储区---FLASH选项字区域,这里简单聊下该话题并顺便给出基于STM32F0芯片的代码实现及几点提醒。 . `; E) i9 q' ^9 _) @6 _6 O4 L 这块特定的FLASH存储区域,通常用来存放有关芯片内部FLASH读保护、写保护、看门狗使能方式、芯片启动、RAM校验、电源监控等配置信息。具体内容以及格式因不同的STM32系列可能有差异。 6 G9 V/ i8 S2 G# o9 e; q( M 一般来讲,各个选项字由选项字节组成,各选项字节由原始字节项和互补字节项组成。如下图存储方式,绿色栏为原始选项字节,黄色栏代码互补选项字节。不同系列可能有差异。【下文中截图如无特别说明,均是来自STM32F0芯片的参考手册。】 + F; T; U n# T$ t9 ? L/ o8 C ![]() . T, p; l4 o* {9 c8 C2 B FLASH选项字一般安排在某固定地址起始的一块连续的地址空间。下图就是STM32FO芯片内部FLASH选项字的地址及内容结构安排。这里包括读保护选项字节、用户选项字节、扇区写保护选项字节。( Q, U3 E7 L7 u* n ![]() 5 m H8 a8 I# s3 h+ ] ( K, A+ V- W0 p$ |5 t 对于出厂的芯片,FLASH选项字往往具有初始出厂值。下图就是STM32F0芯片出厂时FLASH选项字的初始值。) i( T' m9 J2 J ! e& C% t) d' X# d. _2 {9 d 0 k* W2 R' \7 k) r$ J- n. }% D ![]() 5 u6 v' P" d; o0 C! @ 但在我们的实际应用中往往需要结合实际应用情况,对FLASH选项字进行重新配置。而对FLASH选项字进行重新配置一般有两种方式: 第一种,通过编程烧录工具进行选项字的配置并写入。' }( j" `3 M3 c/ i3 C: X - q/ N" ?$ v/ s% z9 n& Q4 \ 比方使用STLinkUtiliy或STM32CubeProgrammer或其它类似编程工具来实现。这种方式相对较为简单,一般对操作人员往往有些要求。尤其在有意或无意弄错配置选项的情况下,没法快速发现并及时纠正。 ![]() 0 I- ^1 x4 z; M5 ?/ x ![]() $ s$ }8 m' B; ?0 N9 m" T 另外一种方式就是用户在应用代码里根据应用需求做FLASH选项字的配置编程。 9 P6 c" Z9 r6 F1 ^ B; s$ P 这种方式,对开发人员来讲会增加了一些工作量。但烧录时只管烧录FLASH执行代码即可,关于芯片配置方面的信息在代码里自行完成。即使烧录时出现选项字的误操作,用户程序代码也可以将其自动纠正过来。. E+ N# x$ w; l/ Y3 b# v: W4 x 6 M7 ~7 G. I0 D) j& Y ; W6 t$ x; Q7 S# P7 \- u 不论使用哪种方式修改FLASH选项字的配置,要想新的选项字信息真正起作用,还有个对FLASH选项字信息进行加载的环节,即将FLASH选项字的配置信息加载到选项字寄存器,从而作用于芯片的相关功能,也就是上面提到过的读保护、写保护、看门狗使能、电源控制、启动选择等功能信息。一般来说,将FLASH选项字信息加载到选择字寄存器需要借助系统复位或上电复位来完成。【当在调试状态下修改FLASH选项字时,它的加载需借助上电复位】: s4 q; }" c1 y) n$ s* j" Z8 ~ : L9 M5 \, d( k3 F+ V! j+ J( c. f% d ! ?" u! O, G/ o5 |+ F ![]() 9 b8 |% N; Y9 o* E$ ` " P. r2 ]3 H( b 在进行FLASH选项字加载之前,芯片硬件会先对选项字信息进行基本地判断与确认,即将各个原始选项字节与互补选项字节的进行匹配比对。如果比对失败将产生出错事件,并将比对失败的选项字节【原始字节和互补字节】强制修改为0xFF或其它指定值,视不同的STM32系列而定。3 E. Y' d$ `! Z+ m1 K$ \5 O. i. Y * B, m/ Z' O0 t7 n& e 比方,下面就是STM32L4系列做选项字加载前发生原始项与互补项信息比对不匹配时的处理原则: ! _1 L9 v3 ?+ D. p8 _ % E G {$ |+ a) N, V ![]() , a+ C8 E2 c7 ^2 P* l* f5 N5 T 关于FLASH选项字信息地编程修改,这里特别提醒几点:/ Q" [: K# _7 ^! {) z 5 _' o0 ?# w7 ^2 d- h! ^ , d, x% e: @/ P- g2 k3 R2 E. W 第1点,当准备好要修改的选项字信息,在对FLASH选项字进行编程修改之前,需先对FLASH选项字区域进行擦除。发出擦除指令即可。擦除完毕之后再将新的选项字信息写入选项字区域。 . @( [7 I# @( b2 \* h ![]() ![]() 5 u3 p1 n6 N6 F3 J8 V: A+ e. o ( l0 E, N$ u& C8 _$ ] 那么为了保障选项字正确有序的编程写人,在做FLASH选项字的编程时,一定要保持电源的稳定。【比方说,代码里稍作延时等上电稳定后再操作】如果正在做FLASH选项字编程时发生电源电压剧烈波动或重启,很可能发生FLASH选项字区域被擦除了但又没有正确写入新的选项字信息的情况。而且,前面也提到了,如果选项字因为没有被完整地正确写入,在做FLASH选项字加载前,还会由于原始项与互补项做信息比对失败而被强行修改为OxFF或其它指定值。2 }9 X! M+ |8 N% K / E! a/ M# ?- E Q 下面两幅截图就是使用烧录工具软件【STM32CubeProgrammer】,在做flash选项字编程时通过模拟芯片被强行断电所发生的情形。* z# ]/ C' k/ w* K: L8 P * q! h1 J& v7 f% n) c , W- S$ z. A5 V; ^ 本来,开始做FLASH选项字编程时,RDP保护配置都是选择的LEVEL_0。在选项字编程时芯片被中途断电重新后连接所看到结果却是芯片被读保护了,其它配置字也被擦除了。从截图中可以看到,此时RDP=0xFF,既不是0xAA也不是0xCC,那么芯片的读保护等级就相应地变成了LEVEL_1读保护状态了。有人碰到此情形时往往感到纳闷不解,明明自己没有修改RDP的配置怎么RDP的保护等级变了呢?就是因为中途断电,选项字区域刚被擦除又还没来得及完整、正常写入。* C0 U% V' m2 Y) Z* M; w; ~! | 3 s# r" k `3 M5 E- g& Y ![]() $ z( ~# l3 [" m* A% p2 Z) _( ~ ![]() 当然,做FLASH选项字编程时除了要求电源稳定外,编程时序也需遵照手册介绍的来操作,以保证选项字编程的正确性。 第2点,对于某些双BANK的STM32芯片,比如STM32L4,STM32F42X等系列,在做选项字编程调整时,也是先将两个BANK的用户选项页的内容进行擦除,然后依据用户准备好的选项寄存器的内容对所有选项字内容重新编程更新。; C4 q; K7 B: ~( K # v: T$ K) u7 N 8 J. q! p# E, y2 q5 @9 w _ ![]() ) N! y) ~+ F# ]2 U1 _ 这里要注意的是,我们不能只是单独地针对双BANK其中的某一个BANK进行用户选项字调整,而是需要将BANK1和BANK2的配置信息都准备好并写入相应的选项字寄存器,从而实现对两个BANK的选项字信息的编程修改。1 G- N9 R i" d+ j. T: o% s 6 q( a* H# e- N1 j9 r9 R 第3点,在做选项字编程过程中,如果修改读保护选项时,一定要清楚-----如果是从LEVEL_1调整为LEVEL_0将会发生整个芯片内部的FLASH内容被全部擦除的情况。另外,很多STM32系列支持LEVEL_2读保护,该保护等级具有不可逆性。如果选择该等级,除非你自己在芯片内部准备好了升级引导代码,否则你是不能再对片内代码内容做任何更新,更不可能基于该芯片再做调试了。 ; l4 A/ X1 n) J7 h7 ] 6 r4 K8 J7 q( M, x0 A" a) n( } 好,关于STM32芯片的选项字的编程提醒就聊到这里。感觉上内容应该不多,但由于STM32家族拥有众多系列,同时各个系列间在选项字这部分内容又或多或少存在着差异,让内容突然庞杂了很多。这里只能抛砖引玉似地给些提醒,具体应用时请参考各个STM32系列的参考手册及编程手册。4 P( W e g7 y% \& d/ n5 p 下面就以STM32F072芯片为核心的Nucleo板,给出一个实现用户选项字编程的示例。 / M4 |+ d% M9 m& z7 a; { 先看看STM32F072片内选项字区域的基本内容框架,如下图:- x& b" u7 ? l! g/ b7 s 3 `( u; u, H$ w k ![]() 一、选项字内容的规划与准备: 0 D6 g& q& Q% Y% w$ y+ ~" H 4 q: y, t' d0 l0 {% Q6 M' _) p7 S) ? 这里我将RDP选项字节配置为0xbb,即LEVEL_1,则其互补选项字节内容就是0x44;将USER选项字节下面红色方框内的三个选项置1,另外两个保留位置1,那么USER选项字节的内容就是OxEA,它的互补字节就是0x15.# l+ f5 V. F# T* o @6 I ! Y s* Y, h1 ~% e2 P% J U" b4 @ , t$ {2 y& e) V' ~ ![]() 选项域中的Data0字节配置为0x99,则其互补字节nData0则为0x66; : o4 {* |* T/ J$ K6 C3 k; | q 2 R& m7 U* I+ R: u) ~1 Y4 n S 选项域中的Data1字节配置为0x88,则其互补字节nData0则为0x77; & R% n& W3 G0 X' D$ E 按照上面规划准备好数据,如下图所示。各选项数据按原项和互补项组成半字写入。' h# e y, J& ~( B" s# O 3 R b! _- B8 q/ X( B! U$ E$ R 【顺便说下,这里没有对扇区写保护做配置。当然要做也是完全可以的】 + T% e8 _9 U1 E. t3 m 6 Z1 e3 r ~% j, g% M" E1 U ![]() 1 Z2 y* d& f+ A9 o5 \# S! h' n 二、编写相应程序代码,编译后下载到芯片: ![]() 相关代码不长,较为简单。按照手册描述的来编写。我将上述代码分成6部分。 - \* A: a( `) [5 `! f0 ~ 第1部分,做有关选项字节内容的判断,是否均为预期的选项配置内容。9 E5 P! `) m$ ?& F+ W9 ~ 0 D: Y9 T0 R0 v# C3 b! Q. f 第2部分,对FLASH、OPTION编程做开锁操作。7 X( E9 w. w! ?2 j: l( @ . x ]# }5 t9 H k- j 第3部分,对选项域进行擦除,然后退出擦除状态。 " t9 \/ b. [# p5 Q/ ^7 M : v2 F$ Q: C5 d% |" N5 }- E. @% x 第4部分,对欲修改的选项字节进行编程修改。$ n7 y$ L% S/ z* [ 第5部分,完成选项字的修改与编程后,上锁并退出。; @+ f/ D5 Q% q6 p( Q" R # W9 \. V( {6 Y 第6部分,触发系统复位,将新的选择域内容加载到选项控制寄存器。 g5 H' \7 B6 D; x& ?# Q$ A i( C, W . w3 r9 ~, o- u$ W8 ] 6 i6 C3 Y: ^8 \" T, G0 ` 三、运行程序,验证结果 0 v+ v4 g- ~$ y$ j+ E" `# m 运行用户程序实现选项域的编程修改操作。可连接到STLINK Utlity之类的工具查看选项配置结果,看看是否跟预期规划的一致。( b% s" K4 Y/ B; K ![]() " E+ x4 K( Y+ u . D: \- Q# W- Z3 y 上图是通过STLINK Utility工具查看到的STM32F072芯片运行用户程序后的选项域的结果。不难看出,跟我们预期的结果一样【打勾的选项表示置1的选项】。我们还可以基于STLINK_Utility工具随意修改RDP选项字以外的其它选项内容,通过它做选项字的修改编程。然后再次运行用户程序,可以发现各个选项字内容又会回归到用户预设的配置内容。$ w* u6 N3 n( j- i2 |8 O# W* }- {3 s7 ?& G ! e3 c! j+ d" Y& q+ ? |