1 延长EEPROM寿命的方法
一般应用中,要存储的数据比较单一,EEPROM空间与要存储的数据相比要大得多。为此,可采用一种利用存储器空间延长EEPROM工作寿命的方法。
方法1:不固定数据存放的地址,而是用一个固定的基地址加上EEPROM内的一个单元的内容(即偏移地址)作为真正的地址;若发现存储单元已坏(写入和读出的内容不同),则偏移地址加1,重新写入。如果采用100倍的存储器空间冗余,可将EEPROM的实际寿命延长100倍。
方法2:从第一个存储单元开始存储数据N次,然后转到下一个单元再存N次,依次类推,当最后一个单元存放N次之后,再转到第一个单元重新开始。本文推荐采用该法,因为上一种方法存在一个致命缺陷:当某一个EEPROM单元写坏再用下一个单元时,原先存储的数据就读不出来了。本文后面的内容也依此作根据进行阐释。
2 EEPROM数据储存地址的确认
嵌入式系统中的EEPROM电源消失后,储存的数据依然存在。确认数据所存地址是能够正确将所存数据读出的基本条件,但是掉电后再上电时如何确认数据的地址指针呢?可以采用以下的方法。
2.1 找最大数据法
查找EEPROM中的最大数据,就是掉电前最后一次存储的数据。该法对于单片机来讲,编程时程序量较大,可采用折半查找法或分块查找法降低程序代码量。
2.2用后还原法
循环将当前数据写入EEPROM后,立即将前面的EEPROM空间都清0再上电,查找数据非0的EEPROM空间即可。有的单片机要求在写入数据之前,必须将所有空间都恢复为OxFF,这样查找非0xFF的数据空间即可。
2.3地址指针法
为每一个数据在EEPROM中设一个地址指针EE—ADR[i],这样只要确认EEADR[i]数值即可找到相应数据。对于地址指针法有下述一些技巧。
①一个地址指针对应一个数据。该方法思路简单,假设EEPROM空间可以存放50组数据和50个地址ADR[i],一组数据占用4个字节,一个地址占用1个字节,共250字节,在储存数据时,使50个地址指向50组数据的首地址,且使小于变量i的ADR[i]都清零,或重新赋值为0xFF。查找数据时,找到非O或非0xFF的ADR[i],然后根据ADR[i]的值来确定数据的储存地址即可。
②用两个变量来记录数据储存的地址,ADDR一0LD和ADDILNEW分别标志当前数据读出时的地址和新采集来的数据下一次要存储的地址。
③用一个字节的EEPROM空间作为地址指针。此时,地址指针的数值总是和数据储存的地址值相等,此种情况最具有技巧性,因为此时可以使用最少的地址指针,因而可以最大量的节省。EEPROM空间,从而延长EEP—ROM寿命。此方法的思路是:假设第O字节作为地址指针,当该指针值为1时,数据从第一个地址开始储存;为2时,数据从第二个地址开始储存,依此类推。
3 应用实例
本人在设计智能流量仪表时,使用美国Microchip公司的PIC16F877单片机。该单片机本身自带有256字节的EEPROM,其擦写次数典型值为100万次,主要用其储存累计流量和仪表系数。其中,第201~255字节用于存储仪表系数,因其不需要经常操作,所以对这些数据不再采用变地址动态存储。第0字节储存数据的地址指针,第1~200字节用来储存累计量。
对于累计量,可以采用浮点数储存,只占用3~4个字节,目的在于用固定的字节长度可以保证相同的相对精度。
智能流量仪表常年不停地工作,短时间内的流量相对总的流量而言,可以忽略不计。因此,为了使单片机的EEPROM寿命更长,过一个时间片段(例如10s)储存一次累计量也可以满足精度要求。假若出现停电事故,累计量只有10s的流量误差,相对总量可以满足精度要求。该策略可使EEPROM延长10倍寿命。
因本例采用O字节来记录数据的存储地址,若数据每存储一次就换一个地址,则200字节可以存200/4=50个位置,相当于储存数据的EEPROM空间擦写一次,第O字节就擦写50次,因此,第O字节的寿命是其他的1/50,这显然不可行。故可以采用如下方式:在第一个地址储存50次以后再换下一个地址储存,到最后一个地址也储存50次再转到第一个地址储存时,这样所有的EEPROM空间使用次数都相等。
储存累计量流程图如图1所示。
pass:数据存储之前程序循环的次数,它表征多长时间储存一次数据。
save_adr:数据在EEPROM中的存储地址。
save_time:某数据在同一地址存储的次数。
eeprom_write(eeadr,value):EEPROM写函数。
下面是该流程对应的程序(采用澳大利亚HI—TECH公司的PICC demo编译器)。
if(save_time>50)//存50次后移一位EEPROM地址
{save_time=0;clrwdt();
save_adr+=4;//累计量用4个字节储存
if(save_adr>197)//只用1~200字节存储累计量{save_adr=1;}
eeprom_write(0,save-adr);//save-adr为所存累计量地址首字节
}
eeprom_write(savcl_adr,x[O]);
eeprom_write(save_adr+1,x[1]);
eeprom_write(save_adr+2,x[2]);
eeprom_write(save_adr+3,x[3]);
}
}
4 读出EEPROM中存储的数据
系统中的EEPROM掉电后,再上电时从EEPROM读出数据的程序比较简单。
void Read_Sum_Q(){ //读出掉电前的累计量
y[O]=eeprom_read(save_adr);
y[1]=eeprom_read(1+save_adr);
y[2]_eeprom_read(2+save_adr);
y[3]=eeprom_read(3+save_adr);
p2=(float *)y;
Q=*p2;
}
结语
采用实例所示的方法,10s存一次可使EEPROM寿命延长10倍,采用50倍的EEPROM空间冗余,又使其奉命延长50倍,这样具有1000000次寿命的EEPROM可以不间断工作时间为10×50×1000000/(3600×24×365)=15.85年,基本可以满足各种嵌入式系统的工作要求,大大提高了EEPROM的使用效率。因此,本文介绍的方法在相关领域具有广泛的借鉴意义和实用价值。