本程序是在要求记录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个字节.
}
}
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
一周热门 更多>