尝试延长Flash读写寿命程序

2019-07-21 04:43发布

本程序是在要求记录MCU工作总工作时间的目的下编写(会随时开关机),适用需要保存的数据较少的情况。采用标志头的方式来确定一个扇区内,数据存放的位置。当然这样就要避免保存的数据和标志头一样,不过,修改读数据头的方式,避免这个缺陷。
void UserClock_SpecialSave(void)
{
    WORD buff[4] = {0};
   
    buff[0] = 0xAA55;
    buff[1] = clockDay;
    buff[2] = clockHour;
    buff[3] = clockMin;
    GD_FlashWriteNoErase(buff,sizeof(buff)/2);
}
u16 flashReadBuff[512] = {0};   //扇区1KB
const WORD markStr = 0xAA55;
void UserClock_SpecialRead(void)
{
    s8 i = 0;
    WORD buff[3] = {0};
    GD_FlashRead(GD32_CLOCK_TIME_ADDR,flashReadBuff,sizeof(flashReadBuff)/2);
    for(i=127; i>=0; i--)
    {
        if(memcmp(&flashReadBuff[4*i], &markStr, sizeof(markStr)) == 0)
        {
            memcpy(buff, &flashReadBuff[4*i+1], sizeof(buff));
            break;
        }
    }
    if(i < 0)
    {
        clockDay = 0;
        clockHour = 0;
        clockMin = 0;
    }
    else
    {
        clockDay = buff[0];
        clockHour = buff[1];
        clockMin = buff[2];
    }
}
//不写满一个扇区,不擦除写Flash测试,延长Flash使用寿命
//写入数据最大为1KB
//return 0操作正确
u8 GD_FlashWriteNoErase(u16 *p_buffer,u16 numToWrite)
{
    s8 i = 0;
    s8 a = 0;
fmc_unlock();      //解锁

    GD_FlashRead(GD32_CLOCK_TIME_ADDR, GDFLASH_BUF,sizeof(GDFLASH_BUF)/2);
    for(i=127; i>=0; i--)   //如果在127*8处读到0xAA55,说明整页已经写满
    {
        a = memcmp(&GDFLASH_BUF[4*i], &markStr, sizeof(markStr));
        if(a == 0)
        {
            if(i != 127)
            {
                GD_FlashWriteNoCheck(GD32_CLOCK_TIME_ADDR+8*(i+1),p_buffer,numToWrite);
                break;
            }
            else
            {
                fmc_page_erase(GD32_CLOCK_TIME_ADDR);//擦除这个扇区
                GD_FlashWriteNoCheck(GD32_CLOCK_TIME_ADDR,p_buffer,numToWrite);
                break;
            }
        }
    }
    if(i < 0)
    {
        GD_FlashWriteNoCheck(GD32_CLOCK_TIME_ADDR,p_buffer,numToWrite);
    }         
fmc_lock();//上锁
return 0;
}
//从指定地址开始读出指定长度的数据
//readAddr:起始地址
//pBuffer:数据指针
//NumToWrite:半字(16位)数
void GD_FlashRead(u32 readAddr,u16 *p_buffer,u16 numToRead)   
{
u16 i;
for(i=0;i<numToRead;i++)
{
  p_buffer[i] = GD_FlashReadHalfWord(readAddr);//读取2个字节.
  readAddr+=2;//偏移2个字节.
}
}



友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。