嵌入式开发--STM32G030C8T6,写片上FLASH死机CFGBSY和写入出错

故障现象1

G0系列,写片上FLASH时,经常死机,而且按复位键都没用,属于不断电都救不回来的那种死法。这种情况一般是由硬件置位了某个标志导致,只有断电才能故障复原。

故障查找

检查FLASH的相关寄存器,发现死机后SR寄存器如下图:
在这里插入图片描述
正常情况下,SR的值应该是等于0,如果哪一位不为0,则是出现了相应的错误。
上面的截图不仅仅是在死机之后出现,即使按下复位键,仍然是这样,通过操作寄存器,可以消除PGAERR和PGSERR两个标志位,但不能消除CFGBSY。
查手册发现,CFGBSY位是硬件置位和清0的,所以无法手动消除。
看来按复位键后,仍然不能正常工作的原因找到了,只要想办法清除这个标志位即可。

故障解决

通过向Flash的任意一个地址,直接写一个数,当然是不按照FLASH的操作流程。
数值无所谓。反正也不会真正写入数据的。
这时硬件会对FLASH的CFGBSY位清0,并将PGAERR和PGSERR两个标志位置位。
再手动对PGAERR和PGSERR两个标志位清0即可。
代码如下:

//向任意地址写一个数,数值任意,在Flash的SR寄存器中,硬件会自动清除CFGBSY标志,同时产生2个错误
//再手动清除这两个错误标志即可清除CFGBSY标志
void LL_flash_clear_cfgbsy(void)    //CFGBSY标志无法软件清除,只能通过制造错误,让硬件清除
{
	if(__HAL_FLASH_GET_FLAG(FLASH_FLAG_CFGBSY) != 0x00U)
	{
			*(uint32_t *)(0x08001000) = 1;	//向FLASH任意位置写一个数
			__HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_PGAERR) ;//清除2个标志位
			__HAL_FLASH_CLEAR_FLAG(FLASH_FLAG_PGSERR) ;
	}
}

故障现象2

G0系列的FLASH写入,是双字模式,也就是一次写入8字节,即64位。
STM32G030C8T6的一页是2K字节,也就是256个u64。
当我写入的数据比较多时,比如一次写整整一页,这时就必然会在接近完成的时候发生写入错误,代码如下:

//直接写入FLASH,在写入前不检查FLASH内容是否为空
u8 LL_flash_write_without_check(u32 addr, u64* pdata64, u16 len)
{
  u16 i,j;
  u8 ret;
  u64 read;
  u64 write;
  u8 err=0;
  
  HAL_FLASH_Unlock();
  
  for(j=0; j<len; j++)
  {
    for(i=0; i<10; i++)    //不超过10次的重复操作,以保证写入成功
    {
      ret = HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD, addr+j*8, *(pdata64+j));
      if(ret == HAL_OK)
      {
        read = *(__IO uint64_t *)(addr+j*8);  //在MCU认为写入正确以后,再次读取数据,并进行比对,如果比对不成功,说明写入出错
        write = *(pdata64+j);
        if(read == write)
          break;
        else
        {
          ret = 255;
          return ret;
        }
      }
      else
        err = 1;
    }
  }
  
  HAL_FLASH_Lock();
  return 0;
}

每次都会在写入超过0xe0,也就是224个数据之后出现错误,也可能是225,或者226,227,不会再更大了。换了板子也是同样的故障。
具体情况就是HAL_FLASH_Program返回值为0,表示写入成功,但是再次读取数据时,读出的数据与写入的数据不相等。

故障解决

不得已对代码进行了修改,如下

//直接写入FLASH,在写入前不检查FLASH内容是否为空
u8 LL_flash_write_without_check(u32 addr, u64* pdata64, u16 len)
{
  u16 i,j;
  u8 ret;
  u64 read;
  u64 write;
  u8 err=0;
  
  HAL_FLASH_Unlock();
  
  for(j=0; j<len; j++)
  {
    for(i=0; i<10; i++)    //不超过10次的重复操作,以保证写入成功
    {
      write = *(pdata64+j);
      ret = HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD, addr+j*8, write);
      if(ret == HAL_OK)
      {
        read = *(__IO uint64_t *)(addr+j*8);  //在MCU认为写入正确以后,再次读取数据,并进行比对,如果比对不成功,说明写入出错
        if(read == write)
          break;
        else
        {
          ret = 255;
          HAL_FLASH_Lock();
          return ret;
        }
      }
      else
      {
        err = 1;
      }
    }
  }
  
  HAL_FLASH_Lock();
  return 0;
}

其实修改的就是

      ret = HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD, addr+j*8, *(pdata64+j));

改成了

      write = *(pdata64+j);
      ret = HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD, addr+j*8, write);

故障排除。而这两种方式是完全等价的啊。难道是编译器出问题了?

唉,问题解决就行。

  • 1
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32G030C8T6是一款高性能的ARM Cortex-M0+微控制器。为了实现对其Flash的读写操作,我们可以按照以下步骤进行操作: 1. 首先,我们需要在程序中引入相关的头文件,以便使用相关的库函数。例如,可以引入"stm32g0xx_hal.h"头文件,其中包含了对STM32G0系列微控制器的HAL库函数定义。 2. 然后,我们需要进行Flash的解锁操作。由于对Flash的读写操作需要进入特权模式,所以需要使用相关的函数进行解锁。可以使用"HAL_FLASH_Unlock()"函数进行解锁操作。 3. 接下来,我们可以使用"HAL_FLASH_Program()"函数进行Flash写入操作。该函数需要传入一个Flash地址以及要写入的数据。例如,可以将数据写入Flash的某个页中。 4. 如果需要读取Flash中的数据,可以使用"HAL_FLASH_Program()"函数进行读取。该函数需要传入一个Flash地址以及一个指向接收数据的指针。例如,可以读取Flash某个地址中的数据。 5. 写入或读取操作完成后,我们需要对Flash进行锁定操作,以保护Flash的数据不被误操作。可以使用"HAL_FLASH_Lock()"函数进行锁定操作。 需要注意的是,进行Flash操作时需要小心,避免不当的操作导致数据丢失或Flash损坏。同时,由于Flash有限的写入次数,应尽量避免频繁的写入操作,以延长Flash的使用寿命。 以上是使用STM32G030C8T6进行Flash读写操作的基本步骤,可以根据具体的需求进行适当的调整和优化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值