熱度 10
上一篇主要說(shuō)明NVM區(qū)操作注意事項(xiàng),本文針對(duì)上篇提到的NVM區(qū)數(shù)據(jù)備份方法進(jìn)行補(bǔ)充講解。
NVM區(qū)主要特性是寫(xiě)入數(shù)據(jù)掉電不丟失,可以永久的保存數(shù)據(jù),一般用作存放不經(jīng)常修改的數(shù)據(jù),此功能類似FLASH。向NVM區(qū)寫(xiě)入數(shù)據(jù)可分為3步:第一步,將目標(biāo)扇區(qū)內(nèi)原有數(shù)據(jù)讀出到RAM中;第二步,擦除NVM目標(biāo)扇區(qū)內(nèi)數(shù)據(jù);第三步,將新數(shù)據(jù)和RAM中的舊數(shù)據(jù)寫(xiě)入到該扇區(qū)中。基于以上寫(xiě)操作的特點(diǎn)可以看出,若執(zhí)行寫(xiě)NVM區(qū)操作的第二步或第三步時(shí)芯片斷電了,就會(huì)造成NVM區(qū)內(nèi)原有數(shù)據(jù)丟失,而新數(shù)據(jù)寫(xiě)入失敗,表現(xiàn)出NVM區(qū)內(nèi)數(shù)據(jù)錯(cuò)亂的現(xiàn)象。雖然這只是小概率事件,但出于穩(wěn)定性和產(chǎn)品健壯性考慮也要加入數(shù)據(jù)備份機(jī)制。
數(shù)據(jù)備份有兩套基本方案,分別適用于小數(shù)據(jù)量和大數(shù)據(jù)量的應(yīng)用場(chǎng)景。本期介紹小數(shù)據(jù)量備份方案,即要求每次更新數(shù)據(jù)量小于一個(gè)扇區(qū)大小減1,本例扇區(qū)大小256字節(jié),因此更新數(shù)據(jù)量最大255字節(jié)。(注意:不同型號(hào)芯片NVM扇區(qū)大小會(huì)有差異,可以聯(lián)系凌科技術(shù)確認(rèn))。實(shí)現(xiàn)思路是:從NVM區(qū)選取兩塊區(qū)域,每個(gè)區(qū)域的最后一字節(jié)記錄該區(qū)域的操作次數(shù)。以該標(biāo)志決定更新數(shù)據(jù)應(yīng)選用哪個(gè)區(qū)域。依次循環(huán)操作兩個(gè)區(qū)域,操作區(qū)域1的時(shí)候,區(qū)域2就充當(dāng)了備份作用,反之同理。
第一次寫(xiě)入數(shù)據(jù)
第一步:將NVM區(qū)地址0x0000~0x00FF定義為Block1,將0x0200~0x02FF定義為Block2。
#define Block1 0x0000
#define Block2 0x0200
第二步:定義一個(gè)256字節(jié)的數(shù)組databuf和一個(gè)標(biāo)志位變量cnt,并將cnt賦值0。
unsigned char databuf[256];
Unsigned char cnt = 0;
第三步:將目標(biāo)數(shù)據(jù)(待寫(xiě)入數(shù)據(jù))拷貝到databuf數(shù)組中,并將cnt+1拷貝到databuf+255的位置。
databuf[255] = cnt+1;
第四步:調(diào)用WriteNVM函數(shù)向Block1寫(xiě)入數(shù)據(jù)databuf,寫(xiě)入長(zhǎng)度256字節(jié)。調(diào)用WriteNVM函數(shù)向Block2+255位置寫(xiě)入cnt,寫(xiě)入長(zhǎng)度1字節(jié)
WriteNVM(Block1,databuf,256);
WriteNVM(Block2+255,&cnt,1);
更新數(shù)據(jù)
第一步:調(diào)用ReadNVM函數(shù)分別讀取Block1和Block2的標(biāo)志位(最后一字節(jié))。
unsigned char flag1,flag2;
ReadNVM(Block1+255,&flag1,1);
ReadNVM(Block2+255,&flag2,1);
第二步:判斷兩個(gè)Block的標(biāo)志位大小,選取標(biāo)志位數(shù)值小的Block進(jìn)行更新。將目標(biāo)數(shù)據(jù)(待寫(xiě)入數(shù)據(jù))拷貝到databuf數(shù)組中,并將標(biāo)志位加2后拷貝到databuf+255的位置。
if(flag1 > flag2)
{
cnt = flag2 + 2;
databuf[255] = cnt;
WriteNVM(Block2,databuf,256);
}else{
cnt = flag1 +2;
databuf[255] = cnt;
WriteNVM(Block1,databuf,256);
}
讀取數(shù)據(jù)
第一步:讀取Block1和Block2的標(biāo)志位數(shù)據(jù)。
ReadNVM(Block1+255,&flag1,1);
ReadNVM(Block2+255,&flag2,1);
第二步:判斷標(biāo)志位大小,數(shù)值大的Block內(nèi)為新數(shù)據(jù),數(shù)值小的Block內(nèi)為舊數(shù)據(jù)。
if(flag1 > flag2)
{
ReadNVM(Block1,databuf,255);
}else{
ReadNVM(Block2,databuf,255);
}