本帖最后由 stm1024 于 2019-1-18 11:15 编辑 " }2 a3 O4 j5 Y
2 A3 z* S$ m# F) L- [* s
Flash现已成为MCU的标配,Flash的特性在决定了MCU的一些功能,如代码大小、执行效率、FDU等。了解一款MCU的Flash显得很重要,这里测试一下G071的Flash。
* b. r4 q5 k* m8 S5 l; ^! u8 ~& a$ }先看看和F0存在的不同。flash的大小就没必要深究了,主要的区别如下:
2 H0 L! u! V! s! l" U% ~- h
0 Q I+ V' T5 Y
1.指令缓存。Flash控制器含有一个8字节的预取指缓冲器和一个16字节的指令缓存。
. v- X# V( ~- m! I- A/ I! o2.OTP区域。所谓的OTP,也就是one time programming,这个区域可以按双字方式写入数据,而且只能写一次,如果写入的数据有一个bit不为0,则以后这个区域就不可更改了,可用于存取非擦写的用户数据。
0 ~1 a$ W! N8 V# K4 d3.快速编程。官方给出的数据是22ms的擦除时间和写入每双字时间为82us。& M, t2 p( i! c* r# W
4.PCROP。这个是proprietary code readout protection的缩写,用于保护代码,其代码只能用来执行,而不能被读取或写入。至于Securable Memory,则是安全储存器,该区域不能被非安全区域外访问,
0 Q0 j) T6 p: w4 A5.ECC。ECC是Error Correction and Checking,即错误检查和纠正。能检查2 bits的错误,并能纠正1 bit翻转错误。; L% E! u) T+ C# R3 d# A
) }0 [; H2 A- ~3 L7 }& U/ @
再看看Flash的组织。见下图:
8 `- ]2 O2 s7 U: x8 P
9 s& c; i! C# [1 }0 W3 D. c
Flash主要分为两块,分别是主块和信息块。主块就是常规使用的Flash,通常也是我们程序的入口地址,这一块可以随便使用。1 X: W4 L/ L, ]8 u
而信息块则包含了系统储存器,OTP区域和选项字节,Flash的操作方式按下图:2 p& S. x0 i0 V# L! N! Y# x
8 ?5 p* }( R; ]2 o$ T; |- Y编程需要一次写入8字节,也就是粒度为64bits(实际上是72bits,还有8bits的ECC)快速编程为256字节(仅支持Main Memory),快速编程因为不检查所写入的地址,因此比标准编程快了约37%。擦除可按2KB的页或者整个擦除(128KB),但花费的时间相当。
" g! E; i! L" B9 L! N6 z) a/ u4 g- j/ @4 j
在选项字节的配置上,和F0系列没有差别(或者我还没有发现)。3 ?3 c$ W) v: F- `/ p
: s6 F" z- e8 D* o# R4 I5 z
以下Flash编程测试代码:# h2 `1 `$ A" e$ s
- int main(void)- U: C; p5 {) ?1 v! r& \
- {# Y, R, A* R0 U: H2 V2 f
- uint16_t i;
( n) R P; m* f0 H - HAL_Init();' X+ M0 @- g7 O5 I) p* ^$ ~
- SystemClock_Config();/ Y* O2 r6 n8 W2 N, x
- MX_GPIO_Init();
2 b: z3 n! O% B K - HAL_FLASH_Unlock();
0 y. l' V. b# ~* Q# ? - EraseInitStruct.TypeErase = FLASH_TYPEERASE_PAGES;
/ `1 L1 H' \4 N& ~/ Z" i - EraseInitStruct.Page = GetPage(FLASH_USER_START_ADDR);
; s$ K( ^( E8 y7 I - EraseInitStruct.NbPages = 1;
& ?" `+ o6 a/ p) b2 z {- q - HAL_FLASHEx_Erase(&EraseInitStruct, &PageError);
0 R7 g" ~: |: ~+ w1 k - //8*256=2KB,1 page
5 O5 u" L( H; m8 ?; } - for(i=0;i<0x100;i++)$ X0 U. z. c% i. D+ F3 Y
- HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD, FLASH_USER_START_ADDR+i*8, 0xefcdab8967452301);
. Q! a* Q6 ?) T/ X+ h$ G - HAL_FLASH_Lock();
$ U% x1 [8 G! C9 L. V/ ~ - while (1)9 A* X4 B% P0 q$ q. Z- I
- {
: c% Y5 {2 i+ `& ] B9 e - 4 C% Q. ?) z7 D: s+ Y
- }
! }( P( h D5 Y) U - }
复制代码 效果嘛,就是把一页的数据都改写为0x012345678abcdef这种模式,因为ARM核是Little-Endian的,所以低字节在前,高字节在后:2 f8 r$ U$ ]9 N0 V
, D" |! `( L5 u1 p1 G
不得不说,有ST Link Utility就是好,直接Dump Flash。
3 d2 t2 d0 C) N4 v3 C+ U) M此外,可以看到刚好改写数据是1页:
+ p; t6 Y! q) A& e9 z% ?9 I$ Y
- L' r0 T. R; g9 j1 t
因为0x2000-0x1800=0x800,也就是1 Page的大小。" Q7 H y6 x+ d9 A' L6 m
% K+ E% s$ g U/ s9 Q
2 b$ Q: }0 S4 \/ K- t
5 b/ q* m1 \1 I0 Y$ J) p: l5 g6 }. E
|
啊,这个没复制进去。
一行代码:9 l$ p) ]9 r! T4 K7 v& B: ], S
这命名方式不太符合ST家的风格,而且目前从官网看,也就只有G070,G071,G081三个大系列。2 W9 F: Z9 T; I, z# h4 d
这个可以参考HAL库的文档,我都是调用的HAL库函数
* |0 x2 J# \+ U; `5 z
谢谢!( C" r5 o6 x! b. r6 J) D. _
另外还有页设置的函数GetPage()是在哪个文件中定义的?我在编译时提示未定义。( `, [* @) X% X# C+ n4 @
这个问题已经解决了,编译通过了。7 }! g4 O3 T7 m- J8 |
额,这个还没见到过,不过一般用的好像都是F103C8T6
感谢分享
大佬过奖了~
过奖了,学了点皮毛
这个文章中的,一位买了STLINK 主控是stm32gc102cb