【问题】STM32G0芯片擦除flash失败,发现死在FLASH

您所在的位置:网站首页 flash无法打印 【问题】STM32G0芯片擦除flash失败,发现死在FLASH

【问题】STM32G0芯片擦除flash失败,发现死在FLASH

2023-09-27 23:23| 来源: 网络整理| 查看: 265

本人开发stm32g030和g070都遇到过擦除flash失败的问题,HAL库中按照例程方式擦除存在擦除失败问题。例程基本流程如下:

HAL_FLASH_Unlock(); __HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_ALL_ERRORS) ; HAL_FLASHEx_Erase(&pEraseInit,&PageError); HAL_FLASH_Lock() ;

按照这种方式其实很容易擦除失败。网上多数是说中断问题等,还有些是看门狗喂狗问题。

以上问题之外,本人遇到的是FLASH_FLAG_CFGBSY一直被置位,导致擦除函数里判断等待超时以后就略过去了。

尝试注释掉这一部分的话,会在执行擦除时之后触发硬件错误中断。所以该方法不可取。作者开始寻找在擦除前通过某些操作把FLASH_SR_CFGBSY标志位给置0,由于该标志位是硬件置位的,软件没法置0,因此只能通过其他方式。

作者在尝试了诸多方法都没法把FLASH_SR_CFGBSY标志置0,想换个方式解决,就是找到发生该标志被置1的原因,结果发现本人的项目是因为串口1的发送执行之后,FLASH_SR_CFGBSY就被置1了。

调用串口1发送前下图:

调用串口1发送后下图:

本人项目对打印日志的需求还是很需要的,所以不可能将就不使用打印的,于是又是在网上漫长地寻找答案。某些论坛上也遇到同病相怜的开发者发表了一样CFGBSY标志置1不恢复的问题,没人给出合适的答案。

偶然间自己尝试通过iar的debug强制修改字符结果神奇的事情发生。

随便写个‘1’

 

按下回车键确定,发生如下图的结果:

 CFGBSY置0了,同时报了3个错误标志来。

作者由此得来灵感,在unlock前,判断CFGBSY是否置位1,如果在flash代码区域中在自己觉得没用的区域执行写入操作,就能够把CFGBSY置0,同时报其他flash错误标记,然后通过清除这些标记的手段实现擦除动作。具体代码参考如下:

#define FLASH_FLAG_ALL_ERRORS (FLASH_FLAG_OPERR | FLASH_FLAG_PROGERR | FLASH_FLAG_WRPERR | \ FLASH_FLAG_PGAERR | FLASH_FLAG_SIZERR | FLASH_FLAG_PGSERR | \ FLASH_FLAG_MISERR | FLASH_FLAG_FASTERR | \ FLASH_FLAG_OPTVERR | FLASH_FLAG_ECCC | FLASH_FLAG_ECCD) uint8_t lc_flash_ram_to_rom(void) { FLASH_EraseInitTypeDef pEraseInit ; uint32_t PageError = 0; pEraseInit.TypeErase = FLASH_TYPEERASE_PAGES ; pEraseInit.Page = WORK_STORE_TABLE_PAGE ; pEraseInit.NbPages = 1 ; // pEraseInit.Banks = FLASH_BANK_1 ; again: if(__HAL_FLASH_GET_FLAG(FLASH_FLAG_CFGBSY) != 0x00U) { *(uint32_t *)(WORK_STORE_TABLE_ADDR+240) = 12323;//flash随意写入一个值,使flash触发错误 __HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_ALL_ERRORS) ; } if(HAL_FLASH_Unlock() != HAL_OK) { goto again ; } FLASH_WaitForLastOperation(10) ;//等待一下,提升擦除成功率 HAL_FLASHEx_Erase(&pEraseInit,&PageError); if(PageError != 0xFFFFFFFF) { DEBUG_printf("\r\nerase fail\r\n") ; goto again ;//失败时重新执行 } uint32_t flash_unit_n = (sizeof(work_store_table) / FLASH_UNIT_VALUE) + 1 ; uint64_t * flash_pdata = (uint64_t *)(&work_store_table) ; for(uint32_t i = 0 ; i < flash_unit_n ; ++i) { HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD,WORK_STORE_TABLE_ADDR+i*8,*(flash_pdata+i)) ; } HAL_FLASH_Lock() ; return 1 ; }

      关键代码:

  *(uint32_t *)(WORK_STORE_TABLE_ADDR+240) = 12323

该方法只是随意在flash块中在自己想要擦除的地方随意写一个值。

通过多次测试,成功率100%,认为已经解决该问题,至于该工程的串口1为什么会触发CFGBSY置位,已经不用关系。虽然找不到根本原因,但是只要能解决问题的话作者已经不在深究。



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3