Dialog的sdk没有提供类似fds的这种文件操作系统,DA14585 的话基本上也用不着。目前原生的dialog sdk中,仅提供了一个叫做 spi_flash的库函数,用于基本的flash操作。我在我们的一个产品中,设计数据存储结构的时候,分配了50页(50*4096Byte)用来存储不间断实时记录的温度数据,当存储的数据量超过50页的记录上限后,就会擦除第一页,在这一页继续记录,第一页满了之后,再去第二页。如此这般的操作,空间不够的时候就擦除较早一些时候保存的数据,保证永远有空间可以用来记录新的数据。这里的话,有一个需求是开机的时候要找到当前写到哪个地方了,下一个数据需要保存到哪个地址?第一个有效的记录在哪个位置?
这里有多种方式来实现我们的目的。这里的话,我最初设计了一个两层的记录-查找结构。由于不想遍历50页的数据(太多了),这里我新增了一页,用于单独记录index数据,每新加一个实际的温度记录,就会在这一页中同步新增一个index记录,查由于温度记录的格式都是统一的,所以查找实际记录的时候,可以根据index的值来直接计算出对应的位置。记录index的页写满之后,直接擦除掉,从头开始写。
这里的话,这个项目其实是我转固件做的第一个项目,没啥经验。设计这个 实际记录 + index 记录的 数据存储方式的时候,没有想到要将二分法应用到flash的遍历中。当时的想法是,遍历一页的数据比遍历50页的数据速度快多了。实际上,如果使用二分法的话,多少页的数据都能快速遍历完,是不需要额外设计一页来记录index数据的。二分法是后边灵机一动,应用上的,但是数据结构已经设计好了,就没有必要再修改掉。
所以目前这里写的二分法来遍历,其实只是遍历了一页flash,当然如果有需要的话,算法的原理不变,随意扩充多少页都是可以的。这里我使用过两种方式,一种是常规的逐个4字节遍历,一种是使用二分查找的思想遍历,当然这个和二分法查找还是有一点区别的。这里将两种方法都分享一下。
为了保存遍历的结果,我这里设计了一个header,结构如下:
struct index_header{
uint32_t last_temp_addr; //the address of last record.
uint32_t last_temp_index; //end of the last index.
uint32_t last_index_addr; //current write address of index page.
uint32_t start_temp_index; //the valid start temp index.
uint32_t start_temp_addr; //the valid start temp addr.
uint32_t temp_size; //size of all temps.
};
每个温度数据的完整记录都以一个不断递增的唯一index开始,uint32类型,占用4个字节。当时分层设计的一页flash,专门保存header信息,每存储一个温度记录,就会在这里同步新增一个4个字节的对应的index。遍历的时候,就是找这个index,如果是0xFFFFFFFF,表示这个位置没有记录,如果找到了,可以直接根据index计算出实际的位置。这个header包含了第一个和最后一个有效温度数据的存储地址和index。
对于flash而言,要在大量的记录中找到目标,是需要花费很多时间的,使用二分法去查找,会节约大量的时间。由于内存有限,这里每次只会读取4个字节出来,但是如果内存足够多,可以一次读取几十或者上百个字节,这样的话不使用二分法也能快速找到目标的。
先上代码:
/**view the index page,get index and address info.
*
* @return RES_OK,RES_READ_ERROR,RES_OTHER,
**/
//check first address, and then use a binary search.
int8_t check_index_binary(void) {
//set default value.
memset(&g_index_header,0,sizeof(struct index_header));
g_index_header.last_index_addr=INDEX_PAGE_START;//very important,otherwise write_index(...) will get bugs.
g_index_header.last_temp_addr=TEMP_PAGE_START;
g_index_header.start_temp_addr=TEMP_PAGE_START;
const uint8_t INDEX_SIZE=4;
uint8_t check_read_size=INDEX_SIZE*2;
uint8_t data[check_read_size];//cache read data.
memset(data,0xFF,check_read_size);
uint8_t is_find=0;
uint32_t find_addr;
//read first address.
uint32_t read_size=strict_read(data,INDEX_PAGE_START,INDEX_SIZE);
if(read_size==INDEX_SIZE){
if(!is_valid_data(data,0,INDEX_SIZE)){
return RES_OK;
}else{
uint32_t sum_num=PAGE_SIZE/INDEX_SIZE;
uint32_t min_position=0;
uint32_t max_position=sum_num-1;
uint32_t mid_position;
uint32_t mid_addr;
while(!is_find&&min_position<=max_position) {
mid_position=(min_position+max_position)/2;
mid_addr=INDEX_PAGE_START+mid_position*INDEX_SIZE;
uint32_t read_size=strict_read(data,mid_addr,check_read_size);
// if(read_size==check_read_size) {//read success.
uint8_t is_pre_writed=0;
uint8_t is_pos_writed=0;
is_pre_writed=is_valid_data(data,0,INDEX_SIZE);
if(is_pre_writed) {
if(mid_position==sum_num-1) { //find at the end.
is_find=1;
break;
}
is_pos_writed=is_valid_data(data,INDEX_SIZE,INDEX_SIZE);
if(is_pos_writed) { //search pos position.
min_position=mid_position+1;
mid_position=(min_position+max_position)/2;
} else { //finded.
is_find=1;
}
} else { //search pre position;
max_position=mid_position-1;
mid_position=(min_position+max_position)/2;
}
// }
}
if(is_find){
find_addr=mid_addr;
}
}
}
if(is_find) {
memcpy(&g_index_header.last_temp_index,&data[0],INDEX_SIZE);
g_index_header.last_temp_addr=get_temp_addr_by_index(g_index_header.last_temp_index);
g_index_header.last_index_addr=find_addr;
if(g_index_header.last_temp_index
这里的话,程序开始会先读一次,看看是否是完全没有记录,这一段其实没什么必要,因为就算没有数据,二分法查找的次数是有上限的,不用考虑花费的时间。
然后开始遍历,这里对比二分法,也是根据位置来找,但是这里的位置是以4个字节(uint32)为单位的,这里的position表示完整状态下,一页flash中所存储的index数据的序号,实际的偏移地址需要对应乘以4。因为要找到最后一个不为0xFFFFFFFF的值,遍历的时候,每次读取8个字节总共2个index。
初始情况下,min position就是0的位置,max position为最后一个位置,mid position的值永远是 (min+max)/2 。
然后需要分情况讨论,读取的2个index:如果两个index都有值,则需要继续往后边找,min position变为 mid position +1,max position 不变;如果第一个位置没有值(由于是顺序写入的,这个时候第二个位置也必然没有值),则需要继续往前找,min position不变,max position变为 mid position -1;如果第一个index有值,第二个index没有值,则找到了最后的写入index的位置;若最后一个position也有值,则表示写满了,该位置即为最后的index记录;
一旦找到了目标index,就可以退出循环,根据对应位置index的值来直接计算出实际温度记录在flash中的位置。
下面附上最开始不使用二分法的遍历方式:
/**view the index page,get index and address info.
*
* @return RES_OK,RES_READ_ERROR,RES_OTHER,
**/
int8_t check_index(void) {
//set default value.
set_def_index_value();
const uint8_t READ_SIZE=4;//just read one record to avoid error read.
uint32_t address=INDEX_PAGE_START+PAGE_SIZE*INDEX_PAGE_NUM-READ_SIZE;//the last 4 bytes of index page address.
uint8_t data[READ_SIZE];//cache read data.
memset(data,0xFF,READ_SIZE);
do {//reverse read 64 bytes every time from flash.
uint32_t read_size=strict_read(data,address,READ_SIZE);
if(read_size==READ_SIZE) {
//for(i=READ_SIZE-4; i>=0; i=i-4) {//reverse read 4 bytes every time from data.
if(data[0]!=INVALID_VALUE||data[1]!=INVALID_VALUE||data[2]!=INVALID_VALUE||data[3]!=INVALID_VALUE) {//find the last valid data.
memcpy(&g_index_header.last_temp_index,&data[0],4);
g_index_header.last_temp_addr=get_temp_addr_by_index(g_index_header.last_temp_index);
g_index_header.last_index_addr=address;
if(g_index_header.last_temp_index=0;i=i-4)**/
} else { /**end of:if(read_size==READ_SIZE)**/
return RES_READ_ERROR;
}
address-=READ_SIZE;
} while(address>=INDEX_PAGE_START);
return RES_OK;
}
以上所有代码均通过大量Unit 测试,可靠性有充分保证。