STM32L系列单片机内部提供了EEPROM存储区域,但实质上,其FLASH也是EEPROM类型,只不过有一块区域被开放出来专门用作EEPROM操作而已。STM32L的EEPROMSTM32L使用寿命设计为100000次擦写以上,容量为2K-4K,这对于一般设备的参数存储来说是非常理想的。但从EEPROM使用方式看,其不适用于被反复修改的数据存储使用,一般作为配置参数,其修改次数往往是比较少量的。 STM32L的EEPROM和FLASH是统一编址,操作共用同一个读写电路,所以在EEPROM读写的时候STM32L核对于FLASH的一切访问和操作都将暂停,只有当EEPROM的操作完成后,才继续执行后续代码,在这期间只有EEPROM的读写电路工作,CPU处于挂起状态。 读操作,和FLASH以及内存一样,EEPROM的数据读取直接用总线读周期读出即可,不需要进行额外操作和设置。 - #define EEPROM_BASE_ADDR 0x08080000
; x- C9 B' A' \* R - #define EEPROM_BYTE_SIZE 0x0FFF
复制代码 以上定义EEPROM区的起始位置和大小,给定偏移量之后,可以按字节/半字/字/双字方式读出,但要注意的是最好偏移地址都按四字节对齐,以免产生总线访问错误或是取不正确:- /*------------------------------------------------------------ 0 r. S4 V& x& H `+ u" u8 j# w
- Func: EEPROM数据按字节读出
$ J8 x# b- Z$ b0 `% k7 r$ r - Note: : ^2 f; A! D- @2 _" k% I
- -------------------------------------------------------------*/ - q: @; G% j9 B5 W& z
- void EEPROM_ReadBytes(uint16 Addr,uint8 *Buffer,uint16 Length) ; K# b* A; M9 W3 y3 }
- { ; |8 a0 q( t" d
- uint8 *wAddr; , {% H5 G' c) ?! Z
- wAddr=(uint8 *)(EEPROM_BASE_ADDR+Addr);
! B( G' }! k5 G( o# ~8 j - while(Length--){
3 Q, l9 j+ Q2 e: c# a - *Buffer++=*wAddr++; . P0 a; G+ A3 A, [9 N
- } : M5 q- C: X3 @7 D1 j3 `9 k
- }
复制代码- /*------------------------------------------------------------
, K: `0 e" K* y! b" @ k - Func: EEPROM数据读出5 o- E4 G0 H. F |% Z, Y; q: N6 [
- Note:$ ?, ]4 a7 D' O0 g
- -------------------------------------------------------------*/ 7 e/ d1 @( A6 \. w8 I$ ]
- void EEPROM_ReadWords(uint16 Addr,uint16 *Buffer,uint16 Length) 7 G, c# J ]: F1 R) T6 p, t/ x( T
- { 5 y- V" Q2 U5 F: |$ b) ?: N
- uint32 *wAddr; 8 Z& q- Y' \. e3 o+ G! s0 _5 ^
- wAddr=(uint32 *)(EEPROM_BASE_ADDR+Addr); 8 l# }( g. C Z% X
- while(Length--){
% }+ f2 {4 l0 p' l - *Buffer++=*wAddr++; 2 h/ E" |7 Z- W8 O
- }
3 m; {* x& E t- P, f$ b2 c l - }
复制代码 以上方法使用字节和字方式读出,在后面方法中,在一个字的存储空间内只使用了16个位,另16位不用,这样以避免产生对齐问题。 EEPROM的编程比读操作要复杂的多,本质上来说,擦除操作和写入操作是一样的,擦除只是在相应的地方写入0x00000000,但在STM32L的实现上,根据其手册说明貌似把这种擦除和写入区分开了,当写入0x00或0x0000或0x00000000时,自动执行一次擦除操作,在值为非0时,才执行一次所谓的写入操作。数据的写入过程先要对EEPROM进行解锁,这通过对特殊寄存器写入特殊序列实现,然后在写入之前进行擦除操作,其擦除是按字/ 双字/页进行的,推荐使用页擦除方式进行,先把参数读到内存,并修改,再进行页擦除,最后将参数写回,这种方式比较通用,否则很容易出现地址对齐或长度问题。在数据擦除完成之后,即可进行写入,每写一字节/半字/双字,都需要判断其是否写入完成,这和内部高压擦写电路有关,只有在上次操作完成之后再进行其它操作才有意义。最后,对EEPROM进行加锁,以保护数据。 下是手册给出的解锁命令码: - #define PEKEY1 0x89ABCDEF //FLASH_PEKEYR
4 q+ V* U+ Y! c& @; S! k - #define PEKEY2 0x02030405 //FLASH_PEKEYR
复制代码 以下分别实现按字节和字方式写入: - /*------------------------------------------------------------) _" y5 B: d) U- ~0 W4 g# I
- Func: EEPROM数据按字节写入/ R# M- q& u F- ? `
- Note:
7 t0 f5 J, r- A: O7 e, W - -------------------------------------------------------------*/
2 X Z* O% Z4 r. `* Q - void EEPROM_WriteBytes(uint16 Addr,uint8 *Buffer,uint16 Length)
3 g3 K5 g [* T/ |' k- p - {
2 Y5 f4 F0 f/ ^# ?% S6 _4 H% K - uint8 *wAddr;
1 p6 ~; W! y8 t T" q$ M - wAddr=(uint8 *)(EEPROM_BASE_ADDR+Addr);
, K5 W5 i, [/ r/ e6 L6 E - DIS_INT
8 O1 S, c0 P9 \1 Y; q; S - FLASH->PEKEYR=PEKEY1; //unlock % N: F# Z8 |3 M
- FLASH->PEKEYR=PEKEY2;
4 f7 u- b: n* o1 M# ~% b2 X9 f3 W - while(FLASH->PECR&FLASH_PECR_PELOCK);
5 B% w0 e% C! ?% r - FLASH->PECR|=FLASH_PECR_FTDW; //not fast write ' N3 |- u& u1 s) e, _
- while(Length--){ * { u; A6 w- I2 h' P' G
- *wAddr++=*Buffer++;
0 a0 A9 M! q- U$ d. P - while(FLASH->SR&FLASH_SR_BSY); - L3 [7 i" I: g8 `2 m! {
- }
2 B+ e$ w- {$ G# M" F& D - FLASH->PECR|=FLASH_PECR_PELOCK;
6 u% N7 o9 `& \ - EN_INT
3 u% P3 T/ |* a - }
复制代码- /*------------------------------------------------------------
! [- J, @: A0 p2 l' [/ k7 Y - Func: EEPROM数据按字写入: H/ C# p# I: A* p( A
- Note: 字当半字用
& i) C/ p ~& D - -------------------------------------------------------------*/
' z$ [: h$ A& g& z' k- I - void EEPROM_WriteWords(uint16 Addr,uint16 *Buffer,uint16 Length) & }5 K6 c; p( h9 L' x; i
- {
( B" u7 G8 P8 I# f - uint32 *wAddr; ) ]8 G( { K0 G% |: P! ~
- wAddr=(uint32 *)(EEPROM_BASE_ADDR+Addr);
+ G7 e" C+ }3 [$ k5 ^5 x - DIS_INT
% o5 j, p' j. Y5 [ - FLASH->PEKEYR=PEKEY1; //unlock 6 N6 _' y d- {+ |, y
- FLASH->PEKEYR=PEKEY2; 4 ~$ S8 G- A M8 m( ^& |
- while(FLASH->PECR&FLASH_PECR_PELOCK);
1 t9 t9 g$ E9 M - FLASH->PECR|=FLASH_PECR_FTDW; //not fast write
% b/ f3 z8 E+ s# \ - while(Length--){
; {5 s5 o* X5 j - *wAddr++=*Buffer++;
! j' e( ~( J8 N) f - while(FLASH->SR&FLASH_SR_BSY);
9 }. Q1 P g8 f! R7 D: ?: P - }
' i7 m5 R8 Y" }$ V9 @/ c - FLASH->PECR|=FLASH_PECR_PELOCK; ( f; d% z0 n: _
- EN_INT # V! _2 N9 }; l1 p& C* y
- }
复制代码 7 e; S& w" o7 _: j6 d( Y
以上代码中,在写入数据之前先关闭系统中断DIS_INT,写入完成之后打开系统中断EN_INT,这样避免在执行写操作的过程中被中断过程所打断,引起CPU异常或锁死,在在使用中一定要注意。在MDK环境中,两个可以这样定义:5 u- Y$ _7 }# ~. @$ B0 A
- #define EN_INT __enable_irq(); //系统开全局中断 . ^1 D9 n! v. F+ x
- #define DIS_INT __disable_irq(); //系统关全局中断
复制代码
9 L4 B8 f6 S" S L" h6 j9 d/ I |