大部分的单片机flash都是只支持块擦写,所以都需要一块一块的擦除和写入,为了应用的方便,可以保证随时随地的改写任意地址的任意长度数据,而不影响地址所在块其他位置的数据,所以单独写一个函数,实现也比较简单,记录下来方便今后直接使用,实现原理就是查找当前地址所在块,读取出来改块信息,修改相应位置信息后擦除原位置块重新写入,同时用循环保证数据位置跨块或数据长度超过块长度是的分次写入,下面直接贴代码。
void ChangeFlashAny(uint32_t addr,unsigned char *pdata, int length)
{
uint8_t flash_data_temp[512];
uint32_t block_addr_start = (addr / 512) * 512;
uint16_t offset = addr % 512;
uint16_t block_len = 0;
while(length > 0)
{
for(uint16_t i = 0;i < 512;i++)
{
flash_data_temp[i] = ReadFlash(block_addr_start +i);
}
FlashErase(block_addr_start);
block_len = (offset + length) > 512 ? (512 - offset) : length;
for(uint16_t i = 0;i < block_len;i++)
{
flash_data_temp[offset+i] = *(pdata+i);
}
length = length - block_len;
pdata = pdata + block_len;
WriteFlash(block_addr_start,flash_data_temp,512);
block_addr_start = block_addr_start + 512;
offset = 0;
}
}
程序中的擦除,写入和读取的代码和块大小视实际单片机而定,我是在MSP430单片机环境下测试的,430的flash以512字节为一块,使用时只要传入flash任意位置地址,修改的数据指针和数据长度就可以了。