STM32G030F6P6读写flash失败问题(HAL)

STM32G030是F0系列的升级版,其在性能上比F0要好很多,具体G0参数如下:

最开始做项目选用的单片机是STM32F030F4P6,但是在后期使用中发现,我的FLASH(16K)不够用了,就选择了STM32G030F6P6来进行项目使用,主要是价格便宜,资源够用。

在F030使用的flash拿到G030上来发现不可使用,就进行了一些修改,但是这个时候就出现了报错,在进行flash擦除的时候报错

HAL_FLASHEx_Erase(&EraseInitStruct,&PageError);

通过发现擦除有问题,我就去查看其底层函数。

cpp 复制代码
HAL_StatusTypeDef HAL_FLASHEx_Erase(FLASH_EraseInitTypeDef *pEraseInit, uint32_t *PageError)
{
  HAL_StatusTypeDef status;
  uint32_t index;

  /* Check the parameters */
  assert_param(IS_FLASH_TYPEERASE(pEraseInit->TypeErase));

  /* Process Locked */
  __HAL_LOCK(&pFlash);

  /* Reset error code */
  pFlash.ErrorCode = HAL_FLASH_ERROR_NONE;

  /* Wait for last operation to be completed */
  status = FLASH_WaitForLastOperation(FLASH_TIMEOUT_VALUE);

  if (status == HAL_OK)
  {
#if !defined(FLASH_DBANK_SUPPORT)
    /* For single bank product force Banks to Bank 1 */
    pEraseInit->Banks = FLASH_BANK_1;
#endif /* FLASH_DBANK_SUPPORT */

    if (pEraseInit->TypeErase == FLASH_TYPEERASE_MASS)
    {
      /* Proceed to Mass Erase */
      FLASH_MassErase(pEraseInit->Banks);

      /* Wait for last operation to be completed */
      status = FLASH_WaitForLastOperation(FLASH_TIMEOUT_VALUE);
    }
    else
    {
      /*Initialization of PageError variable*/
      *PageError = 0xFFFFFFFFU;

      for (index = pEraseInit->Page; index < (pEraseInit->Page + pEraseInit->NbPages); index++)
      {
        /* Start erase page */
        FLASH_PageErase(pEraseInit->Banks, index);

        /* Wait for last operation to be completed */
        status = FLASH_WaitForLastOperation(FLASH_TIMEOUT_VALUE);

        if (status != HAL_OK)
        {
          /* In case of error, stop erase procedure and return the faulty address */
          *PageError = index;
          break;
        }
      }

      /* If operation is completed or interrupted, disable the Page Erase Bit */
      CLEAR_BIT(FLASH->CR, FLASH_CR_PER);
    }
  }

  /* Process Unlocked */
  __HAL_UNLOCK(&pFlash);

  /* return status */
  return status;
}

其大致意思就是两种擦除方式,片擦除 以及全部擦除 。然后发现其status是在 **FLASH_WaitForLastOperation(FLASH_TIMEOUT_VALUE);**里面进行报错,其函数实现如下:

cpp 复制代码
HAL_StatusTypeDef FLASH_WaitForLastOperation(uint32_t Timeout)
{
  uint32_t error;
  /* Wait for the FLASH operation to complete by polling on BUSY flag to be reset.
     Even if the FLASH operation fails, the BUSY flag will be reset and an error
     flag will be set */
  uint32_t timeout = HAL_GetTick() + Timeout;

  /* Wait if any operation is ongoing */
#if defined(FLASH_DBANK_SUPPORT)
  error = (FLASH_SR_BSY1 | FLASH_SR_BSY2);
#else
  error = FLASH_SR_BSY1;
#endif /* FLASH_DBANK_SUPPORT */

  while ((FLASH->SR & error) != 0x00U)
  {
    if (HAL_GetTick() >= timeout)
    {
      return HAL_TIMEOUT;
    }
  }

  /* check flash errors */
  error = (FLASH->SR & FLASH_SR_ERRORS);

  /* Clear SR register */
  FLASH->SR = FLASH_SR_CLEAR;

  if (error != 0x00U)
  {
    /*Save the error code*/
    pFlash.ErrorCode = error;
    return HAL_ERROR;
  }

  /* Wait for control register to be written */
  timeout = HAL_GetTick() + Timeout;

  while ((FLASH->SR & FLASH_SR_CFGBSY) != 0x00U)
  {
    if (HAL_GetTick() >= timeout)
    {
      return HAL_TIMEOUT;
    }
  }

  return HAL_OK;
}

发现其在这里进行报错,然后从下面返回错误码上来。

返回我打印了这个error到底是多少,发现其值为0x80,发现报的错误是 FLASH_SR_PGSERR

然后查看数据手册以及使用手册,发现这个是编程错误。然后继续查找问题,发现G030的一个bank是2K,修改之后发现还是报这个错误。

在详细查看数据手册后,发现G030进行Flash读写是uint64_t进行读写的,如下:

在此情况下,对读写函数进行修改,将数据等改为uint64_t。在将这些修改过后,发现问题没有在flash擦除那里进行报错,而是在FLASH写入那里卡死。

HAL_FLASH_Program(FLASH_TYPEPROGRAM_FAST,addr,Data[i])!=HAL_OK

上述函数错误的地方是 FLASH_TYPEPROGRAM_FAST ,因为其意思是32位写

Fast program a 32 row double-word (64-bit) at a specified address

但是手册给出是64位写,所以这里进行了报错,然后将这里改成下面函数,整个程序的读写就没有问题了,在此问题就得到了解决。

HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD,addr,Data[i])!=HAL_OK

以下则是G030F6P6单片机的FLASH的程序

读:

cpp 复制代码
/*******************************************************************************
* Function Name  : 读取Flash数据
* Description    : Read packed message form flash
* Input          : buff:point to first location of received buffer.length:Maxmum length of reception
* Output         : 
* Return         : reception length
*******************************************************************************/
uint16_t Read_Flash( uint64_t *Data, uint16_t num, uint32_t addr)
{
    uint16_t i=0;
	uint32_t add=0;
	if(num == 0)
	{
		return 0;
	}
	add = addr;
	i=0;
	while((add < FLASH_USER_END_ADDR1) && (i<num))
	{
		Data[i++] = *(__IO uint64_t *)add;
		add = add+8;
	}
	return i;
}

写:

cpp 复制代码
/*******************************************************************************
* Function Name  : Flash写数据
* Description    : Write a group of datas to flash.
* Input          : buff:pointer of first data, length: write length
* Output         : 
* Return         : true/false
*******************************************************************************/
uint16_t Write_Flash( uint64_t *Data , uint16_t num, uint32_t add)
{
	uint16_t i=0;
	uint32_t addr=0;
    FLASH_EraseInitTypeDef EraseInitStruct={0};
    uint32_t PageError=0;//擦除错误地址
    EraseInitStruct.TypeErase = FLASH_TYPEERASE_PAGES;//仅擦除页
    EraseInitStruct.Banks = FLASH_BANK_1;
    EraseInitStruct.Page = 15;		//注:该page为0-15页
    EraseInitStruct.NbPages = 1;	//擦除一页
    
	HAL_FLASH_Unlock();
    HAL_FLASHEx_Erase(&EraseInitStruct,&PageError);
	if(PageError != 0xFFFFFFFF) {
        return 1;
    }
	addr = add;
	i=0;
	while((addr < FLASH_USER_END_ADDR1) && (i<num))
	{
		if(HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD,addr,Data[i])!=HAL_OK)
		{
			addr = addr + 8;
		}else{
			i++;
		}
	}
	HAL_FLASH_Lock();
    return 0;    
}

在这里使用的是uint64_t进行数据的读写,但是如果用在其他程序,就会出现error。因为单片机是32位的出现uint64_t参与的函数就报错。

在这里我写了一个简易的flash内存管理,因为我们在写flash数据的时候,往往很多数据,并不是单一数据,但是每写一次flash则需要进行flash擦除,我这里采用一个数组进行使用,如下:

实现原理:一次性读取一定数量的数据出来,将自己需要的数据修改过后,再讲修改后的数据全部写入,在实际项目中还是比较实用的。

读:

cpp 复制代码
/*
*********************************************************************************************************
*	函 数 名:uint8_t data_read(uint16_t *data,uint16_t datalen,uint16_t save_addr,uint32_t bank_addr)
*	功能说明:数据读取
*	形    参:data 数据   datalen 长度   save_addr数组中保存的地址    bank_addr  地址
*	返 回 值:
*********************************************************************************************************
*/ 
uint8_t data_read(uint16_t *data,uint16_t datalen,uint16_t save_addr,uint32_t bank_addr) 
{
	uint64_t buf[READ_NUM]={0}; //根据实际数据量进行设置
 
	if(datalen == 0)
	{
		return 1;
	}
	Read_Flash(buf,READ_NUM,bank_addr);  //读数据
	
	for(int i=0;i<datalen;i++)   //数据更新
    {
		data[i] = buf[save_addr+i];
    }
	return 0;
}

写:

cpp 复制代码
/*
*********************************************************************************************************
*	函 数 名:uint8_t data_save(uint16_t *data,uint16_t datalen,uint16_t save_addr,uint32_t bank_addr) 
*	功能说明:数据保存
*	形    参:data 数据   datalen 长度   save_addr数组中保存的地址    bank_addr  地址
*	返 回 值:
*********************************************************************************************************
*/ 
uint8_t data_save(uint16_t *data,uint16_t datalen,uint16_t save_addr,uint32_t bank_addr) 
{
	uint64_t buf[READ_NUM]={0}; //根据实际数据量进行设置
	if(datalen == 0)
	{
		return 1;
	}
	Read_Flash(buf,READ_NUM,bank_addr);  //读数据
	if(buf[0]==0xffffffff)
	{
		for(int i=0;i<READ_NUM;i++)
        {
			buf[i] = 1;
        }
		Write_Flash(buf,READ_NUM,bank_addr);  //写数据
	}
	for(int i=0;i<datalen;i++)   //数据更新
    {
		buf[save_addr+i] = data[i];
    }
	
	Write_Flash(buf,READ_NUM,bank_addr);  //写数据
	return 0;
}

其他宏定义相关代码:

cpp 复制代码
//数组大小
#define READ_NUM	30
//地址
#define ADDR_FLASH_PAGE_0  	((uint32_t)0x08000000)   //第一页
#define ADDR_FLASH_PAGE(n) (ADDR_FLASH_PAGE_0 + (uint32_t)(n)*FLASH_PAGE_SIZE)

#define FLASH_USER_PAGE_NUM		1
#define FLASH_USER_START_ADDR1  ADDR_FLASH_PAGE(16-1)
#define FLASH_USER_END_ADDR1    (FLASH_USER_START_ADDR1 + FLASH_USER_PAGE_NUM*FLASH_PAGE_SIZE)

基本上可以实现功能:主要的问题就是那个必须64位读写,不然数据就有问题

相关推荐
scan18 小时前
单片机串口接收状态机STM32
stm32·单片机·串口·51·串口接收
Qingniu018 小时前
【青牛科技】应用方案 | RTC实时时钟芯片D8563和D1302
科技·单片机·嵌入式硬件·实时音视频·安防·工控·储能
Mortal_hhh9 小时前
VScode的C/C++点击转到定义,不是跳转定义而是跳转声明怎么办?(内附详细做法)
ide·vscode·stm32·编辑器
深圳市青牛科技实业有限公司10 小时前
【青牛科技】应用方案|D2587A高压大电流DC-DC
人工智能·科技·单片机·嵌入式硬件·机器人·安防监控
Mr.谢尔比11 小时前
电赛入门之软件stm32keil+cubemx
stm32·单片机·嵌入式硬件·mcu·信息与通信·信号处理
LightningJie11 小时前
STM32中ARR(自动重装寄存器)为什么要减1
stm32·单片机·嵌入式硬件
鹿屿二向箔11 小时前
STM32外设之SPI的介绍
stm32
西瓜籽@11 小时前
STM32——毕设基于单片机的多功能节能窗控制系统
stm32·单片机·课程设计
远翔调光芯片^1382879887214 小时前
远翔升压恒流芯片FP7209X与FP7209M什么区别?做以下应用市场摄影补光灯、便携灯、智能家居(调光)市场、太阳能、车灯、洗墙灯、舞台灯必看!
科技·单片机·智能家居·能源
极客小张14 小时前
基于STM32的智能充电桩:集成RTOS、MQTT与SQLite的先进管理系统设计思路
stm32·单片机·嵌入式硬件·mqtt·sqlite·毕业设计·智能充电桩