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_ADDRESS 62*512 //數據扇區起始地址 #defineDATA_AREA_SIZE 2*512 //兩個扇區大小 #defineDATA_UINT_SIZE 64 //每一個存儲單元大小,一定要整除扇區大小(512) #defineDATA_UINT_FLAG 0x5555AAAA typedef struct { uint32_t flag; uint32_t data1; … } data_uint_type; voidflash_erase(uint32_t startaddr, uint32_t size) { uint32_t endaddr; endaddr=startaddr+size; //erase sector while(startaddrADDR = startaddr; // setup addr FMC->CMD = 0x04; //Triggerprogramming while ((FMC->CMD 0x100)!=0); startaddr+=512; } return; } uint8_t flash_word_program(uint32_taddr, uint32_t worddata) //返回一個非0的數據當錯誤發生時 { //program word FMC->ADDR = addr; // set upaddr FMC->DATA =worddata; FMC->CMD = 0x02; //Triggerprogramming while ((FMC->CMD 0x100)!=0); if (*(uint32_t *)addr== worddata) return 0; else return 1; } 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; else enduintaddr= 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) ; }else return (data_uint_type*)(startuintaddr) ; } } uint8_tstore_data_uint(data_uint_type* sur_data_ptr, data_uint_type* dst_data_ptr) //返回一個非0的數據當錯誤發生時 { uint32_t n,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);//erase whole data sectors } 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數據存儲到Flash data_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
來源:敏矽MCU
免責聲明:本文為轉載文章,轉載此文目的在于傳遞更多信息,版權歸原作者所有。本文所用視頻、圖片、文字如涉及作品版權問題,請聯系小編進行處理
審核編輯 黃宇
;n++)>)>
-
mcu
+關注
關注
146文章
17173瀏覽量
351656 -
控制器
+關注
關注
112文章
16396瀏覽量
178512 -
FlaSh
+關注
關注
10文章
1638瀏覽量
148182 -
存儲器
+關注
關注
38文章
7513瀏覽量
163987 -
EEPROM
+關注
關注
9文章
1022瀏覽量
81686
發布評論請先 登錄
相關推薦
評論