超级版主
 
- 积分
- 5707
- 金钱
- 5707
- 注册时间
- 2019-5-8
- 在线时间
- 1540 小时
|
|
第四十九章 FLASH模拟EEPROM实验
1)实验平台:正点原子STM32H7R7开发板
2)章节摘自【正点原子】STM32H7R7开发指南 V1.1
3)购买链接: https://detail.tmall.com/item.htm?id=820823382459
4)全套实验源码+手册+视频下载地址:http://www.openedv.com/docs/boards/stm32/zdyz_stm32h7rx.html
5)正点原子官方B站:https://space.bilibili.com/394620890
6)正点原子STM32开发板技术交流群:756580169
STM32H7R7本身没有自带EEPROM,但是STM32H7R7具有IAP(在应用编程)功能,所以我们可以把它的FLASH当成EEPROM来使用。本章,我们将利用STM32H7R7内部的FLASH来实现第四十章实验类似的效果,不过这次我们是将数据直接存放在STM32H7R7内部,而不是存放在NOR FLASH。
本章分为如下几个小节:
49.1 STM32H7R7 FLASH简介
49.2 硬件设计
49.3 程序设计
49.4 下载验证
49.1 STM32H7R7 FLASH简介
STM32H7R7内部只有64KB FLASH,STM32H7R7的闪存模块组织如表49.1.1所示:
表49.1.1 STM32H7R7闪存模块组织
STM32H7R7只有一个块Bank,每个扇区容量为8KB,并具128位FLASH字,每个字有9个ECC(误码校正)位。
关于STM32H7R7内部FLASH的详细说明,详见《STM32H7Rx参考手册_V6(英文版)》第5.3节相关内容。
在执行闪存写操作时,任何对闪存的读操作都会锁住总线,在写操作完成后读操作才能正确地进行。既在进行写或擦除操作时,不能进行代码或数据的读取操作。
49.1.1 闪存的读取
为了准确读取 Flash 数据,必须根据 HCLK 时钟 (flash_aclk) 频率和Vcore电压范围在 Flash 存取控制寄存器 (FLASH_ACR) 中正确地设置等待周期数 (LATENCY)。Flash 等待周期与HCLK时钟频率之间的对应关系,如表49.1.1.2所示:
表49.1.1.2 HCLK时钟频率对应的FLASH等待周期表
等待周期通过FLASH_ACR寄存器的LATENCY[3:0]四个位设置。系统复位后,CPU时钟频率为内部64M RC振荡器,LATENCY默认是1,即2个等待周期。为了得到更佳的FLASH访问性能,我们设置Vcore电压范围为VOS0级别(1.30V~1.40V)。rcc_aclk和rcc_hclk的频率是一样的,都是来自RCC_BMCFGR的BMPRE[3:0]分频,rcc_hclk我们一般设置的是300MHz,这样rcc_aclk也是300MHz的频率。我们设置等待周期为8(LATENCY[3:0]=7),否则FLASH读写可能出错,导致死机。
STM32H7R7的FLASH读取是很简单的。例如,我们要从地址addr,读取一个字(一个字为32位),可以通过如下的语句读取:
Data = *(volatile uint32_t *)faddr; 将faddr强制转换为volatile uint32_t指针,然后取该指针所指向的地址的值,即得到了faddr地址的值。类似的,将上面的volatile uint32_t改为volatile uint8_t,即可读取指定地址的一个字节。相对FLASH读取来说,STM32H7R7 FLASH的写就复杂一点了,下面我们介绍STM32H7R7闪存的编程和擦除。
49.1.2 闪存的编程和擦除
在对 STM32H7R7的Flash执行写入或擦除操作期间,任何读取Flash的尝试都会导致总线阻塞。只有在完成编程操作后,才能正确处理读操作。这意味着,写/擦除操作进行期间不能从Flash中执行代码或数据获取操作。
STM32H7R7用户闪存的编程一般由5个32位寄存器控制,他们分别是:
FLASH访问控制寄存器(FLASH_ACR)
FLASH秘钥寄存器(FLASH_KEYR)
FLASH状态寄存器(FLASH_SR)
FLASH控制寄存器(FLASH_CR)
FLASH CRC控制寄存器(FLASH_CRCCR)
注意:这里的FLASH_KEYR、FLASH_SR、FLASH_CR、FLASH_CRCCR分别对应Bank1的相关寄存器,所以单个Bank的控制寄存器由:FLASH_KEYR、SR、CR和CCR等四个寄存器控制。下面,我们直接以FLASH_KEYR、FLASH_CR、FLASH_SR和FLASH_CCR来介绍相关操作。
STM32H7R7复位后,FLASH编程操作是被保护的,不能写入FLASH_CR寄存器;通过写入特定的序列(0X45670123和0XCDEF89AB)到FLASH_KEYR寄存器才可解除写保护,只有在写保护被解除后,我们才能操作相关寄存器。
FLASH_CR的解锁序列为:
1,写0X45670123到FLASH_KEYR
2,写0XCDEF89AB到FLASH_KEYR
通过这两个步骤,即可解锁FLASH_CR,如果写入错误,那么FLASH_CR将被锁定,直到下次复位后才可以再次解锁。
FLASH配置步骤
STM32H7R7的FLASH在编程的时候,也必须要求其写入地址的FLASH是被擦除了的(也就是其值必须是0XFFFFFFFF),否则无法写入。STM32H7R7的标准编程步骤如下:
1,检查FLASH_CR的LOCK是否解锁,如果没有则先解锁
2,检查FLASH_SR中的BUSY位,确保当前未执行任何FLASH操作。
3,将FLASH_CR寄存器中的PG位置1,激活FLASH编程。
4,等待BUSY位清零,完成一次编程。
按以上步骤,就可以完成一次FLASH编程。不过需要注意:编程前,要确保要写入地址的FLASH已经擦除。
在STM32H7R7的FLASH编程的时候,要先判断缩写地址是否被擦除了,所以,我们有必要再介绍一下STM32H7R7的闪存擦除,STM32H7R7的闪存擦除分为两种:扇区擦除和块擦除。
扇区擦除步骤如下:
1,检查FLASH_CR的LOCK是否解锁,如果没有则先解锁。
2,检查FLASH_SR寄存器中的BUSY 位,确保当前未执行任何FLASH操作。
3,在FLASH_CR寄存器中,将SER位置1,并设置SSN[2:0]=需要擦除的扇区号。
4,将FLASH_CR寄存器中的START位置1,触发擦除操作。
5,等待BUSY位清零。
经过以上五步,就可以擦除某个扇区。本章,我们只用到了STM32H7R7的扇区擦除功能。块擦除功能我们在这里就不介绍了,想了解的朋友请看《STM32H7Rx参考手册_V6(英文版).pdf》的相关内容。
49.1.3 FLASH寄存器
Flash访问控制寄存器(FLASH_ACR)
Flash访问控制寄存器描述如图49.1.3.1所示:
图49.1.3.1 FLASH_ACR寄存器
WRHIGHFREQ[1:0]位,用于控制FLASH编程操作时的延迟,必须根据FLASH操作频率(rcc_aclk)进行正确的设置:00,rcc_aclk≤80Mhz;01,rcc_aclk≤160Mhz;10,rcc_aclk≤240Mhz;11,rcc_aclk≤320Mhz;我们的flash_aclk设置的是300Mhz,设置WRHIGHFREQ[1:0]=11即可。
LATENCY[3:0]位,用于控制FLASH读延迟,必须根据我们MCU内核的工作电压和频率,来进行正确的设置,否则,可能死机。
FLASH密钥寄存器(FLASH_KEYR)
FLASH密钥寄存器描述如图49.1.3.2所示:
图49.1.3.2 FLASH_KEYR寄存器
该寄存器主要用来解锁FLASH_CR,必须在该寄存器写入特定的序列(KEY1和KEY2)解锁后,才能对FLASH_CR寄存器进行写操作。
存储区1的FLASH控制寄存器(FLASH_CR)
存储区1的FLASH控制寄存器描述如图49.1.3.3所示:
图49.1.3.3 FLASH_CR寄存器
LOCK位,该位用于指示FLASH_CR寄存器是否被锁住,该位在检测到正确的解锁序列后,硬件将其清零。在一次不成功的解锁操作后,在下次系统复位之前,该位将不再改变。
PG位,该位用于选择编程操作,在往FLASH写数据的时候,该位需要置1。
SER位,该位用于选择扇区擦除操作,在扇区擦除的时候,需要将该位置1。
START位,该位用于开始一次擦除操作。在该位写入1 ,将执行一次擦除操作。
SSN[2:0]位,这3个位用于选择要擦除的扇区编号,取值范围为0~7。
其他位,我们就不在这里介绍了,请大家参考《STM32H7Rx参考手册_V6(英文版).pdf》 。
存储区1的FLASH状态寄存器(FLASH_SR)
存储区1的FLASH状态寄存器描述如图49.1.3.4所示:
图49.1.3.4 FLASH_SR寄存器
BUSY位:表示BANK当前正在执行编程操作,必须等待该位为0,才可以执行其他操作。
WBNE位:表示BANK写BUFFER是否为空。当该位为1时,表示写BUFFER里面还有数据待写入FLASH,需要等待该位为0,才表示数据写入全部完成了。
QW位:表示操作序列里面是否还有编程操作需要执行,需要等待该位为0,才表示所有的编程操作完成了。
最后,FLASH清除控制寄存器FLASH_CRCCR用于清除相关错误,这里我们就不做介绍了,详见《STM32H7Rx参考手册_V6(英文版).pdf》第5.9.8节。
49.2 硬件设计
1. 例程功能
按键KEY1控制写入FLASH的操作,按键KEY0控制读出操作,并在TFTLCD模块上显示相关信息。LED0闪烁用于提示程序正在运行。
2. 硬件资源
1)LED灯
LED:LED0 – PD14
2)串口1(PB14/PB15连接在板载USB转串口芯片CH340上面)
3)正点原子2.8/3.5/4.3/7/10寸TFTLCD模块(包括MCU屏和RGB屏,都支持)
4)独立按键 :
KEY0 – PE9
KEY1 – PE8
49.3 程序设计
49.3.1 FLASH的HAL库驱动
FLASH在HAL库中的驱动代码在stm32h7rsxx_hal_flash.c和stm32h7rsxx_hal_flash_ex.c文件(及其头文件)中。
1. HAL_FLASH_Unlock函数
解锁闪存控制寄存器访问的函数,其声明如下:
- HAL_StatusTypeDef HAL_FLASH_Unlock(void);
复制代码 函数描述:
用于解锁闪存控制寄存器的访问,在对FLASH进行写操作前必须先解锁,解锁操作也就是必须在FLASH_KEYR寄存器写入特定的序列(KEY1和KEY0)。
函数形参:无
函数返回值:HAL_StatusTypeDef枚举类型的值。
2. HAL_FLASH_Lock函数
锁定闪存控制寄存器访问的函数,其声明如下:
- HAL_StatusTypeDef HAL_FLASH_Lock (void);
复制代码 函数描述:
用于锁定闪存控制寄存器的访问。
函数形参:无
函数返回值:HAL_StatusTypeDef枚举类型的值。
3. HAL_FLASH_Program函数
闪存写操作函数,其声明如下:
- HAL_StatusTypeDef HAL_FLASHEx_Program(uint32_t TypeProgram, uint32_t Address,
- uint64_t Data);
复制代码 函数描述:
该函数用于FLASH的写入。
函数形参:
形参1是TypeProgram用来区分要写入的数据类型。
形参2是Address用来设置要写入数据的FLASH地址
形参3是Data是要写入的数据类型。
函数返回值:
HAL_StatusTypeDef枚举类型的值。
4. HAL_FLASHEx_Erase函数
闪存擦除函数,其声明如下:
- HAL_StatusTypeDef HAL_FLASHEx_Erase(FLASH_EraseInitTypeDef *pEraseInit,
- uint32_t *SectorError);
复制代码 函数描述:
该函数用于大量擦除或擦除指定的闪存扇区。
函数形参:
形参1是FLASH_EraseInitTypeDef结构体类型指针变量。
- typedef struct
- {
- uint32_t TypeErase; /* 擦除类型 */
- uint32_t Sector; /* 擦除的Sector编号 */
- uint32_t NbSectors; /* 擦除Sector的数量 */
- } FLASH_EraseInitTypeDef;
复制代码 形参2是uint32_t类型指针变量,存放错误码,0xFFFFFFFF值表示扇区已被正确擦除,其它值表示擦除过程中的错误扇区。
函数返回值:HAL_StatusTypeDef枚举类型的值。
49.3.2 程序解析
1. STMFLASH驱动代码
这里我们只讲解核心代码,详细的源码请大家参考光盘本实验对应源码。STMFLASH驱动源码包括两个文件:stmflash.c和stmflash.h。
stmflash.h头文件做了一些比较重要的宏定义,定义如下:
- /* FLASH起始地址 */
- #define STM32_FLASH_BASE 0x08000000 /* STM32 FLASH 起始地址 */
- #define STM32_FLASH_SIZE 0x10000 /* STM32 FLASH 总大小64KByte */
复制代码 STM32_FLASH_BASE和STM32_FLASH_SIZE分别是FLASH的起始地址和FLASH总大小,这两个宏定义随着芯片是固定的,STM32H7R7芯片的FLASH是64K字节,所以STM32_FLASH_SIZE宏定义值为0x10000。
下面我们开始介绍stmflash.c的程序,具体程序源码如下:
该部分代码,我们重点介绍一下stmflash_write函数,该函数用于在STM32H7R7的指定地址写入指定长度的数据,有几个要注意的点:
1,写入地址必须是用户代码区以外的地址。
2,写入地址必须是32的倍数。
3,单次写入长度必须是32字节的倍数(4个字)。
4,第 1 点比较好理解,如果把用户代码给擦除了,可想而知你运行的程序可能就被废了,从而很可能出现死机的情况。另外,STM32H7R7的扇区大小为8KB,而FLASH的擦除方式是以扇区为单位进行擦除的,所以建议大家使用该函数的时候,写入地址定位到用户代码占用扇区以外的扇区,比较保险。
5,第2点和第3点则是由于STM32H7R7的FLASH特性,每次写入必须是128位宽,也就是16字节,因此写入首地址必须是16字节的倍数,且写入数据长度必须是32字节的倍数。
由于我们使用了分散加载(ATK-DNH7R7_flash_ROMxspi1.sct),stmflash.c编译后是自动存放到外部QSPI FLASH的,所以不需要做额外的设置。关于分散加载说明,详见:7.2小节。
2. main.c代码
在main.c里面编写如下代码:
- /* 要写入到STM32 FLASH的字符串数组 */
- uint8_t g_text_buf[] = {"STM32 FLASH TEST"};
- #define TEXT_LENTH sizeof(g_text_buf) /* 数组长度 */
- /*SIZE表示半字长(4字节), 大小必须是4的整数倍, 如果不是的话, 强制对齐到4的整数倍 */
- #define SIZE TEXT_LENTH / 4 + ((TEXT_LENTH % 4) ? 1 : 0)
- #define FLASH_SAVE_ADDR 0x0800E000
- /* 设置FLASH 保存地址(必须大于用于代码区地址范围,且为4的倍数 */
- int main(void)
- {
- uint8_t key = 0;
- uint8_t i = 0;
- uint32_t datatemp[SIZE];
- sys_mpu_config(); /* 配置MPU */
- sys_cache_enable(); /* 使能Cache */
- HAL_Init(); /* 初始化HAL库 */
- sys_stm32_clock_init(300, 6, 2); /* 配置时钟,600MHz */
- delay_init(600); /* 初始化延时 */
- usart_init(115200); /* 初始化串口 */
- usmart_dev.init(300); /* 初始化USMART */
- led_init(); /* 初始化LED */
- key_init(); /* 初始化按键 */
- hyperram_init(); /* 初始化HyperRAM */
- lcd_init(); /* 初始化LCD */
- lcd_show_string(30, 50, 200, 16, 16, "STM32", RED);
- lcd_show_string(30, 70, 200, 16, 16, "FLASH EEPROM TEST", RED);
- lcd_show_string(30, 90, 200, 16, 16, "ATOM@ALIENTEK", RED);
- lcd_show_string(30, 110, 200, 16, 16, "KEY1:Write KEY0:Read", RED);
- while (1)
- {
- key = key_scan(0);
- if (key == KEY1_PRES) /* KEY1按下,写入STM32 FLASH */
- {
- lcd_fill(0, 150, 239, 319, WHITE); /* 清除半屏 */
- lcd_show_string(30,150, 200, 16, 16, "Start Write FLASH....", RED);
- stmflash_write(FLASH_SAVE_ADDR, (uint32_t *)g_text_buf, SIZE);
- lcd_show_string(30,150, 200, 16, 16, "FLASH Write Finished!", RED); /* 提示传送完成 */
- }
- if (key == KEY0_PRES) /* KEY0按下,读取字符串并显示 */
- {
- lcd_show_string(30,150, 200, 16, 16, "Start Read FLASH.... ", RED);
- stmflash_read(FLASH_SAVE_ADDR, (uint32_t *)datatemp, SIZE);
- lcd_show_string(30,150, 200, 16, 16, "The Data Readed Is: ", RED); /* 提示传送完成 */
- lcd_show_string(30,170, 200, 16, 16, (char *)datatemp, BLUE); /* 显示读到的字符串 */
- }
- i++;
- delay_ms(10);
- if (i == 20)
- {
- LED0_TOGGLE(); /* 提示系统正在运行 */
- i = 0;
- }
- }
- }
复制代码 主函数代码逻辑比较简单,当检测到按键KEY1按下后往FLASH指定地址开始的连续地址空间写入一段数据,当检测到按键KEY0按下后读取FLASH指定地址开始的连续空间数据。
最后,我们将stmflash_read_word和test_write函数加入USMART控制,这样,我们就可以通过串口调试助手,调用STM32H7R7的FLASH读写函数,方便测试。
49.4 下载验证
将程序下载到开发板后,可以看到LED0不停的闪烁,提示程序已经在运行了。LCD显示的内容如图49.4.1所示:
图49.4.1程序运行效果图
通过先按KEY1按键写入数据,然后按KEY0读取数据,得到如图49.4.2所示:
图49.4.2 操作后的显示效果图 |
|