1、概述
ME32x系列是內嵌ARM Cortex M0/M3核的32位微控制器。該系列控制器由敏矽微電子有限公司自主開發,并具有自主知識產權。敏矽微電子的微控制器包括有通用MCU和專用SOC系列,具有非常高的性價比,是MCU產品升級換代和國外產品替代的最佳選擇。通用功能有高精度ADC,CAN接口,I2S音頻接口,UART串口,SPI接口,I2C總線接口,看門狗定時器(WDT),通用計數器/定時器。特殊接口包括人機界面控制器(LCD驅動,電容觸摸按鍵)和馬達控制功能模塊。
EEPROM作為比較廉價和方便數據存儲器,被廣泛使用并且習慣思維。而MCU Flash與EEPROM相比,除使用方法略有差異外,作為數據存儲器,所起的效果是一樣的。
2、MCU Flash與EEPROM使用比較
擦除 | 擦除時間 | 編程 | 編程時間 | 硬件接口 | 擦寫壽命 | |
MCU Flash | 扇區為單位擦除,擦除后數據為0xFF | 5ms | 32位word編程 | 7us | 通過寄存器接口設置編程,讀Flash通過指針直接讀 | 10萬次 |
EEPROM | 沒有單獨擦除功能 | - | Byte編程 | - | I2C接口 | 100萬次 |
3、使用MCU Flash存儲數據舉例
以ME32S003系列為例,下面說明如何使用MCU Flash 存儲小量的數據(注意,本例重點在探討實現的一個思路,程序調試請用戶自行解決)。
ME32S003系列有32K Flash,我們拿出1K, 即兩個扇區來存儲數據,在這里約定一個數據存儲單元為64 Bytes(包括標志)。
兩個扇區有16個存儲單元,換句話說,可以存儲10萬x 16 =160次數據,遠遠超過EEPROM的壽命。
所有需要存儲數據放在一個數據結構中,方便存儲和提取數據:
#defineDATA_AREA_ADDRESS62*512//數據扇區起始地址#defineDATA_AREA_SIZE2*512//兩個扇區大小#defineDATA_UINT_SIZE64//每一個存儲單元大小,一定要整除扇區大?。?12)#defineDATA_UINT_FLAG0x5555AAAAtypedefstruct{uint32_tflag;uint32_tdata1;…}data_uint_type;voidflash_erase(uint32_tstartaddr,uint32_tsize){uint32_tendaddr;endaddr=startaddr+size;//erasesectorwhile(startaddrADDR=startaddr;//setupaddrFMC->CMD=0x04;//Triggerprogrammingwhile((FMC->CMD&0x100)!=0);startaddr+=512;}return; }uint8_tflash_word_program(uint32_taddr,uint32_tworddata)//返回一個非0的數據當錯誤發生時{//programwordFMC->ADDR=addr;//setupaddrFMC->DATA=worddata;FMC->CMD=0x02;//Triggerprogrammingwhile((FMC->CMD&0x100)!=0);if(*(uint32_t*)addr==worddata)return0;elsereturn1;}data_uint_type* data_area_init(void)//返回一個指向數據單元的指針,空指針表示沒有數據{data_uint_type*ptr;ptr=get_last_data_uint_ptr();if(((uint32_t)ptr==DATA_AREA_ADDRESS)&&(ptr->flag!==DATA_UINT_FLAG)){flash_erase(DATA_AREA_ADDRESS,DATA_AREA_SIZE);ptr=null;}}data_uint_type* get_last_data_uint_ptr(void)//返回一個指向數據單元的指針{uint32_tstartuintaddr,enduintaddr,temp;startuintaddr=DATA_AREA_ADDRESS/DATA_UINT_SIZE;enduintaddr=startuintaddr+DATA_AREA_SIZE/DATA_UINT_SIZE-1;while(startuintaddr!=enduintaddr){temp=(startuintaddr+enduintaddr)>1;if((data_uint_type*)(temp*DATA_UINT_SIZE)->flag==DATA_UINT_FLAG)startuintaddr=temp;elseenduintaddr=temp}startuintaddr=startuintaddr*DATA_UINT_SIZE;if(((data_uint_type*)startuintaddr)->flag!=DATA_UINT_FLAG)return(data_uint_type*)0;else{ If (startuintaddr< (DATA_AREA_ADDRESS+DATA_AREA_SIZE))?????????????????????????????????{ If (((data_uint_type*)(startuintaddr+ DATA_UINT_SIZE)->flag==DATA_UINT_FLAG)return(data_uint_type*)(startuintaddr+DATA_UINT_SIZE);}elsereturn(data_uint_type*)(startuintaddr);}}uint8_tstore_data_uint(data_uint_type*sur_data_ptr,data_uint_type*dst_data_ptr)//返回一個非0的數據當錯誤發生時{uint32_tn,temp0,temp1,temp2,*dataptr;temp1=sizeof(data_uint_type)>>2;if((temp1<<2)!=?sizeof(data_uint_type))?????temp1++;?????temp2=(uint32_t)?dst_data_ptr;?????dataptr=(uint32_t?*)?sur_data_ptr;?????//erase?sector?????if?((DATA_AREA_ADDRESS==temp2)&&(dst_data_ptr->flag==DATA_UINT_FLAG)){flash_erase(DATA_AREA_ADDRESS,DATA_AREA_SIZE);//erasewholedatasectors}for(n=0;n 系統啟動時,先調用data_area_init()函數,返回當前數據單元指針,你可以使用memory copy 復制數據到你的程序中。如果是空指針,你需要對你的數據賦予初值,并把它存儲到數據區。
全局變量:
data_uint_typemydata;data_uint_type*dataptr;main(void){…dataptr=data_area_init();//初始化if((uint32_t)dataptr==0)//空指針{//對mydata賦予初值…//存儲數據到Flash 數據區data_ptr=(data_uint_type*)DATA_AREA_ADDRESS;store_data_uint(&mydata,data_ptr);}…//任何時候,調用store_data_uint()把mydata數據存儲到Flashdata_ptr++;if((uint32_t)data_ptr==DATA_AREA_ADDRESS)data_ptr =(data_uint_type *) DATA_AREA_ADDRESS;store_data_uint(&mydata,data_ptr);…//data_ptr永遠指向當前Flash數據…}注意事項:
由于mydata的地址是編譯器自動設定的,如果發生mydata地址不是word對齊地址,需要手動設置。總之,要確保mydata地址是word對齊的。
4、Revision History
Revision Description Date 1.0 Initial Release. July 2019 審核編輯:湯梓紅
-
mcu
+關注
關注
146文章
17316瀏覽量
352227 -
FlaSh
+關注
關注
10文章
1642瀏覽量
148430 -
存儲器
+關注
關注
38文章
7528瀏覽量
164182 -
EEPROM
+關注
關注
9文章
1033瀏覽量
81845
發布評論請先 登錄
相關推薦
評論