打印
[STM32F1]

读写flash的注意事项

[复制链接]
2044|52
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
jackcat|  楼主 | 2024-7-18 03:06 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
解锁前确认状态: 在执行FLASH解锁操作之前,确实应该确认FLASH是否已被锁定。直接对一个未锁定的FLASH执行解锁指令可能会导致硬件错误(HardFault),因为解锁指令通常是设计用来移除写保护的,对一个已解锁的FLASH再次执行解锁操作是没有意义的,并且在某些情况下可能导致不可预料的行为。

擦除前关闭数据缓存: 这一点至关重要。在执行FLASH擦除操作前,必须确保数据缓存(DCEN位)是关闭状态。如果不关闭,会导致擦除不成功或出现其它未知错误

擦除后执行缓存复位: 擦除操作完成之后,执行数据缓存复位(DCRST位)是一个好的实践。这样做可以确保缓存中任何可能残留的、与FLASH当前状态不符的数据被清除,防止后续读取操作出现混乱。特别是对于连续的写入-擦除-读取序列,这一步骤能够保证读取到的数据是最新的。(遇到的真实情况是:未复位擦除后去访问flash数据,最开始访问的数据可能是擦除前的数据,而不是此刻真实的flash中的数据)

/*
* @Author: duanhuisheng
* @Date: 2021-11-03 21:36:14
* @LastEditTime: 2022-05-24 18:22:23
* @LastEditors: duanhuisheng
* @Description:
* stm32f407用于参数平衡擦写读写flash的注意事项: 解锁前一定要确定是锁住的,或者执行解锁指令会产生 hardfauat
* 错误中断  导致程序跑飞 ,另外在擦除扇区前一定要关闭数据缓存
* 或者擦除失败,擦除完成后要执行一下缓存复位,不然擦除后去访问flash数据,最开始访问的数据可能是擦除前的数据,而不是此刻真实的flash中的数据
*
*/
#include <MString.h>

static void delay(volatile int32_t length)
{
    while (length)
        length--;
}

// FLASH起始地址
#define STM32_FLASH_BASE 0x08000000  // STM32 FLASH的起始地址
// FLASH解锁键值
#define FLASH_KEY1 0X45670123
#define FLASH_KEY2 0XCDEF89AB

// FLASH 扇区的起始地址
#define ADDR_FLASH_SECTOR_0 ((u32)0x08000000)   // 扇区0起始地址, 16 Kbytes
#define ADDR_FLASH_SECTOR_1 ((u32)0x08004000)   // 扇区1起始地址, 16 Kbytes
#define ADDR_FLASH_SECTOR_2 ((u32)0x08008000)   // 扇区2起始地址, 16 Kbytes
#define ADDR_FLASH_SECTOR_3 ((u32)0x0800C000)   // 扇区3起始地址, 16 Kbytes
#define ADDR_FLASH_SECTOR_4 ((u32)0x08010000)   // 扇区4起始地址, 64 Kbytes
#define ADDR_FLASH_SECTOR_5 ((u32)0x08020000)   // 扇区5起始地址, 128 Kbytes
#define ADDR_FLASH_SECTOR_6 ((u32)0x08040000)   // 扇区6起始地址, 128 Kbytes
#define ADDR_FLASH_SECTOR_7 ((u32)0x08060000)   // 扇区7起始地址, 128 Kbytes
#define ADDR_FLASH_SECTOR_8 ((u32)0x08080000)   // 扇区8起始地址, 128 Kbytes
#define ADDR_FLASH_SECTOR_9 ((u32)0x080A0000)   // 扇区9起始地址, 128 Kbytes
#define ADDR_FLASH_SECTOR_10 ((u32)0x080C0000)  // 扇区10起始地址,128 Kbytes
#define ADDR_FLASH_SECTOR_11 ((u32)0x080E0000)  // 扇区11起始地址,128 Kbytes

// 解锁STM32的FLASH
static void STMFLASH_Unlock(void)
{
    FLASH->KEYR = FLASH_KEY1;  // 写入解锁序列.
    FLASH->KEYR = FLASH_KEY2;
}
// flash上锁
static void STMFLASH_Lock(void)
{
    FLASH->CR |= (u32)1 << 31;  // 上锁
}
// 得到FLASH状态
// 返回值:
//  0,操作完成
//  1,忙
//  2,操作异常
static uint8_t STMFLASH_GetStatus(void)
{
    u32 res = 0;
    res     = FLASH->SR;
    if (res & (1 << 16))
        return 1;  // 忙
    else if (res & (1 << 4))
        return 2;  // 操作异常
    else if (res & (1 << 5))
        return 2;  // 操作异常
    else if (res & (1 << 6))
        return 2;  // 操作异常
    else if (res & (1 << 7))
        return 2;  // 操作异常
    return 0;      // 没有任何状态/操作完成.
}
// 等待操作完成
//  time:要延时的长短(单位:10us)
// 返回值:
//  0,完成
//  2,操作异常
//  0XFF,超时
static uint8_t STMFLASH_WaitDone(u32 time)
{
    uint8_t res;
    do
    {
        res = STMFLASH_GetStatus();
        if (res != 1)
            break;  // 非忙,无需等待了,直接退出.
        // delay_us(10);
        time--;
    } while (time);
    if (time == 0)
        res = 0xff;  // TIMEOUT
    return res;
}

// 在FLASH指定地址写一个字
//  faddr:指定地址(此地址必须为4的倍数!!)
//  dat:要写入的数据
// 返回值:0,写入成功
//     其他,写入失败
static uint8_t STMFLASH_WriteWord(u32 faddr, u32 dat)
{
    uint8_t res;
    FLASH->SR |= 0xc0;
    res = STMFLASH_WaitDone(0XFFFFFf);
    if (res == 0)  // OK
    {
        FLASH->CR &= ~(3 << 8);  // 清除PSIZE原来的设置
        FLASH->CR |= 2 << 8;
        FLASH->CR |= 1 << 0;  // 编程使能
        while ((FLASH->CR & 0x01) != 0x01)
        {
            FLASH->CR |= 1 << 0;
        }
        *(vu32 *)faddr = dat;                          // 写入数据
        res            = STMFLASH_WaitDone(0XFFFFFf);  // 等待操作完成,一个字编程,最多100us.
        if (res != 1)                                  // 操作成功
        {
            FLASH->CR &= ~(1 << 0);  // 清除PG位.
        }
    }
    return res;
}

// 擦除扇区
//  sectoraddr:扇区地址,范围是:0~11.
//  0~3,16K扇区;4,64K扇区;5~11,128K扇区.
// 返回值:执行情况
static uint8_t STMFLASH_EraseSector(u32 sectoraddr)
{
    uint8_t res = 0;
    res         = STMFLASH_WaitDone(5000000000);  // 等待上次操作结束,最大5s
    if (res == 0)
    {
        FLASH->CR &= ~(3 << 8);               // 清除PSIZE原来的设置
        FLASH->CR |= 2 << 8;                  // 设置为32bit宽,确保VCC=2.7~3.6V之间!!
        FLASH->CR &= ~(0X1F << 3);            // 清除原来的设置
        FLASH->CR |= sectoraddr << 3;         // 设置要擦除的扇区
        FLASH->CR |= 1 << 1;                  // 扇区擦除
        FLASH->CR |= 1 << 16;                 // 开始擦除
        res = STMFLASH_WaitDone(5000000000);  // 等待操作结束,最大5s
        if (res != 1)                         // 非忙
        {
            FLASH->CR &= ~(1 << 1);  // 清除扇区擦除标志.
        }
    }
    return res;
}

void CHAL_Flash_init(void)
{
    STMFLASH_Lock();  //
    /* 解锁前必须要确保是已经上锁的  或者执行解锁指令会导致程序进入 hard_fault */
    STMFLASH_Unlock();    // 解锁
    FLASH->CR |= 2 << 8;  // 设置为32bit宽,确保VCC=2.7~3.6V之间!!
}

uint32_t CHAL_Flash_Rd(void *addr, void *buffer, uint32_t size)
{
    uint32_t len = size;
    uint8_t *sou = (uint8_t *)addr;
    uint8_t *des = (uint8_t *)buffer;
    // if(FLASH->CR&0x80000000)
    // {
    //   STMFLASH_Unlock();         // 解锁
    // }

    while (len--)
    {
        *des++ = *sou++;
    }
    return size;
}

uint8_t CHAL_Flash_EraseSector(uint8_t *addr, uint32_t sct)
{
    int i   = 0;
    int ret = 1;
    __disable_irq();
    if (FLASH->CR & 0x80000000)
    {
        STMFLASH_Unlock();  // 解锁
    }

    // delay(1000);
    FLASH->CR &= ~(3 << 8);  // 清除PSIZE原来的设置
    FLASH->CR |= 2 << 8;     // 设置为32bit宽,确保VCC=2.7~3.6V之间!!

    FLASH->ACR &= ~(1 << 10);  // FLASH擦除期间,必须禁止数据缓存!!!搞了我两晚上才发现这个问题!
    if ((uint32_t)(addr) < ADDR_FLASH_SECTOR_3)
    {
        STMFLASH_EraseSector(2);
    }
    else
    {
        STMFLASH_EraseSector(3);
    }
    FLASH->ACR |= 1 << 12;  // 数据缓存复位
    delay(10);
    FLASH->ACR &= ~(1 << 12);  // 一定要执行缓存复位  有可能导致 读flash的数据可能是擦除前缓存的数据
    FLASH->ACR |= 1 << 10;  // FLASH擦除结束,开启数据fetch
    delay(10);

    //  delay(1000);
    STMFLASH_Lock();  // 上锁
    __enable_irq();   // 开启总中
    return ret;
}

uint32_t CHAL_Flash_Wt(void *addr, const void *buffer, uint32_t size)
{
    int i = 0, len = size / 4;
    if (FLASH->CR & 0x80000000)
    {
        STMFLASH_Unlock();  // 解锁
    }
    __disable_irq();
    /* Program data. */
    FLASH->ACR &= ~(1 << 10);
    for (i = 0; i < len; i++)
    {
        STMFLASH_WriteWord((uint32_t)addr + i * 4, ((uint32_t *)buffer)[i]);
    }
    FLASH->ACR |= 1 << 10;  // FLASH擦除结束,开启数据fetch
    STMFLASH_Lock();        // 上
                            //    delay(1000);
    __enable_irq();         // 开启总中
    return size;
}


使用特权

评论回复
沙发
tpgf| | 2024-7-19 08:26 | 只看该作者
请问楼主如何确认FLASH是否已被锁定呢

使用特权

评论回复
板凳
paotangsan| | 2024-7-19 09:43 | 只看该作者
如果flash已经是解锁状态 此时我还进行解锁操作 会有什么后果呢

使用特权

评论回复
地板
renzheshengui| | 2024-7-19 20:00 | 只看该作者
在读写flash的时候如果没有关闭中断使能 会有什么后果呢

使用特权

评论回复
5
keaibukelian| | 2024-7-19 21:39 | 只看该作者
为什么数据缓存不关闭就会导致擦除不成功呢

使用特权

评论回复
6
heimaojingzhang| | 2024-7-19 22:45 | 只看该作者
如果是在保护状态下进行写操作 会有什么错误码返回吗

使用特权

评论回复
7
guanjiaer| | 2024-7-19 23:51 | 只看该作者
是否也可以不用判断 无论如何就先进行解锁操作?

使用特权

评论回复
8
4c1l| | 2024-7-27 11:15 | 只看该作者
在执行FLASH擦除操作之前,需要关闭数据缓存(DCEN位)。

使用特权

评论回复
9
工程师犹饿死| | 2024-7-30 18:10 | 只看该作者
通过检查相关寄存器的状态位来确认。执行解锁指令对一个未锁定的FLASH可能会导致不可预料的行为或硬件错误

使用特权

评论回复
10
裤脚口感好| | 2024-7-31 23:57 | 只看该作者
开启的数据缓存可能会导致擦除操作不成功或出现数据不一致的情况

使用特权

评论回复
11
ulystronglll| | 2024-8-12 22:21 | 只看该作者
在执行FLASH解锁操作之前,确认FLASH是否已被锁定是非常重要的。

使用特权

评论回复
12
kmzuaz| | 2024-8-13 03:05 | 只看该作者
因为解锁指令的设计初衷是移除写保护,如果FLASH已经处于解锁状态,再次执行解锁操作可能会干扰正常的硬件操作。

使用特权

评论回复
13
zerorobert| | 2024-8-13 05:57 | 只看该作者
在执行Flash操作时,应该实现适当的错误处理逻辑来检测并响应任何可能的错误条件,比如写入保护错误、编程错误或擦除错误。

使用特权

评论回复
14
maudlu| | 2024-8-13 13:16 | 只看该作者
STM32的固件库提供了函数来检查FLASH的锁定状态,例如FLASH_OB_GetLockStatus()。这些函数封装了直接访问寄存器的复杂性,提供了更易用的接口。

使用特权

评论回复
15
robincotton| | 2024-8-13 17:59 | 只看该作者
在STM32中,FLASH解锁操作涉及以下几个步骤:

写保护:默认情况下,STM32的FLASH是写保护的,这意味着不能对FLASH进行编程或擦除操作。
解锁:要解除写保护,需要按照一定的顺序写入特定的密钥到FLASH的解锁寄存器(FLASH_KEYR)。
编程/擦除:解锁之后,可以对FLASH进行编程或擦除操作。
自动锁定:在编程或擦除操作完成后,FLASH会自动恢复到写保护状态。

使用特权

评论回复
16
1988020566| | 2024-8-13 19:43 | 只看该作者
在确认FLASH锁定后,可以使用解锁指令(如对FLASH_KEYR寄存器写入特定的解锁键值)。这个步骤会解除对FLASH的写保护,允许程序或固件更新操作。

使用特权

评论回复
17
pixhw| | 2024-8-13 21:30 | 只看该作者
在执行FLASH操作时,应设计错误处理机制,包括处理解锁失败、编程错误等情况,保证系统稳定和数据安全。

使用特权

评论回复
18
pl202| | 2024-8-14 22:56 | 只看该作者
在尝试解锁Flash之前,应该先检查Flash的当前状态。如果Flash已经是解锁状态,再次执行解锁操作可能会导致不可预测的结果,包括硬件错误(HardFault)。

使用特权

评论回复
19
beacherblack| | 2024-8-15 11:41 | 只看该作者
如果Flash是锁定状态,需要按照正确的解锁序列来解锁Flash。对于大多数STM32系列,这个序列通常涉及向特定的寄存器写入特定的值。例如,对于STM32F系列的微控制器,通常需要连续两次写入特定的密钥值到FLASH_KEYR寄存器。

使用特权

评论回复
20
uytyu| | 2024-8-15 15:04 | 只看该作者
在STM32中,可以通过读取特定的寄存器(通常是FLASH_CR寄存器中的LOCK位)来判断FLASH是否已被锁定。如果LOCK位显示FLASH已锁定,那么执行解锁操作是合理的。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

7

主题

1456

帖子

0

粉丝