一般,在單片機(jī)中的Flash存儲器用于存放程序代碼,屬于只讀型存儲器。而在MSP430些列的單片機(jī)中,都可以通過內(nèi)置的Flash控制器擦除或改寫任何一段的內(nèi)容。另外,msp430的單片機(jī)內(nèi)部還專門留有一段Flash區(qū)域(information memory),用于存放掉電后需要永久保存的數(shù)據(jù)。利用430內(nèi)部的Flash控制器,可以完成較大容量的數(shù)據(jù)記錄、用戶設(shè)置參數(shù)在掉電后的保存等功能。
硬件介紹:
要對Flash讀寫,首先要了解MSP430的存儲器組織。430單片機(jī)的存儲器組織結(jié)構(gòu)采用馮諾依曼結(jié)構(gòu),RAM和ROM統(tǒng)一編址在同一尋址空間中,沒有代碼空間和數(shù)據(jù)空間之分。
一般430的單片機(jī)都統(tǒng)一編址在0-64k地址范圍中,只有少數(shù)高端的型號才能突破64k(如:FG461x系列)。絕大多數(shù)的msp430單片機(jī)都編址在64kB范圍內(nèi)。地址的大概編碼方式如下:
這是msp430f425的存儲器分配圖,其他在64k范圍內(nèi)的存儲器的單片機(jī)編址方式與此類似:低256B是寄存器區(qū),然后是RAM;空白;1000H到10FFH是信息Flash區(qū);大于1100H-0FFFFH是主存儲器區(qū)(從0FFFFH開始往低地址有單片機(jī)的主Flash,多余的部分空白)。
MSP430F14x的Flash分布:
MSP430F16x的Flash分布:
主Flash部分和信息Flash部分如下(60kB Flash對應(yīng)的單片機(jī),如msp430f149、msp430f149):
主Flash分為以512B為段的單位,0段是單片機(jī)中斷向量等程序入口地址,使用時不要擦除此段或改寫此段,若要擦除或是改寫,請先保存內(nèi)容到RAM或其他段;主Flash各段內(nèi)容均要避免寫入或擦除,以免造成不可預(yù)料的后果。
信息Flash分為兩段:段A和段B,每段128B;可以保存用戶自己的內(nèi)容(主Flash也可以但是要避免與程序代碼區(qū)沖突);這里就把信息Flash的兩段稱為InfoA(1080H-10FFh)和InfoB(1000H-10FFH)。
Flash的操作包括:字或字節(jié)寫入;塊寫入;段擦除;主Flash擦除;全部擦除。任何的Flash操作都可以從Flash或從RAM中運(yùn)行。
Flash操作時需要時序發(fā)生器,F(xiàn)lash控制器內(nèi)部含有時序發(fā)生器用以產(chǎn)生所需的Flash時鐘,F(xiàn)lash時鐘的范圍必須在257kHz到476kHz之間。時序發(fā)生器的框圖如下:
時序發(fā)生器可以選擇ACLK、MCLK、SMCLK作為時鐘源,通過分頻獲得所需的257kHz到476kHz之間的Flash操作時鐘。如果時鐘頻率不再這個范圍內(nèi),將會產(chǎn)生不可預(yù)料的結(jié)果。
擦除:擦除之后,存儲器中的bit都變?yōu)?;Flash中的每一位都可以通過編程寫入有1到0,但是要想由0變?yōu)?,必須通過擦除周期。擦除的最小單位是段。有三種擦除模式:
MERAS ERASE Erase Mode 0 1 Segment erase 1 0 Mass erase (all main memory segments) 1 1 Erase all flash memory (main and information .segments)
可以通過MERAS、ERASE位來設(shè)置擦除的模式:段擦除,主Flash擦除,全部擦除。
對要擦除段內(nèi)的一個地址空寫入啟動擦出周期:空寫入可以啟動時序發(fā)生器和擦除操作??諏懭牒驜USY位立即變高直到擦除周期結(jié)束,這一位變?yōu)榈?0)。BUSY, MERAS和 ERASE位在擦除周期結(jié)束后會自動復(fù)位。擦除周期的時間和要擦出的Flash大小無關(guān),每次擦除的時間對于MSP430F1xx系系列單片機(jī)來說,所需時間是一樣的。擦除的時序如下:
當(dāng)空寫入到的地址不在要擦除的段地址范圍內(nèi)的時候,空寫入無效,直接被忽略。在擦除周期內(nèi),應(yīng)該關(guān)中斷,直到擦除完成,重新開中斷,擦除期間的中斷已經(jīng)置標(biāo)志位,開中斷后立即響應(yīng)。
從Flash中啟動的擦除操作:擦除操作可以從Flash中啟動或是從RAM中啟動。當(dāng)操作是從Flash中啟動的時候,F(xiàn)lash控制器控制了操作時序,CPU運(yùn)行被暫停直到擦除結(jié)束。擦除周期結(jié)束后,CPU繼續(xù)執(zhí)行,從空寫入之后的指令開始運(yùn)行。當(dāng)從Flash中啟動擦除操作時,可以擦除即將運(yùn)行的程序所在的段,如果擦除了即將運(yùn)行的程序所在的Flash段時,擦除結(jié)束后,CPU的運(yùn)行不可預(yù)料。
從Flash啟動時擦除周期如下:
用戶指南里面的示例匯編程序如下:
; Segment Erase from flash. 514 kHz < SMCLK < 952 kHz ; Assumes ACCVIE = NMIIE = OFIE = 0. MOV #WDTPW+WDTHOLD,&WDTCTL ; Disable WDT DINT //; Disable interrupts MOV #FWKEY+FSSEL1+FN0,&FCTL2 ; SMCLK/2 MOV #FWKEY,&FCTL3 ; Clear LOCK MOV #FWKEY+ERASE,&FCTL1 ; Enable segment erase CLR &0FC10h ; Dummy write, erase S1 MOV #FWKEY+LOCK,&FCTL3 ; Done, set LOCK ... ; Re-enable WDT? EINT ; Enable interrupts
從RAM中啟動擦除操作:任意擦除周期都可以從RAM啟動,這時CPU不再暫停而是繼續(xù)從RAM中運(yùn)行接下來的程序。CPU可以訪問任何Flash地址之前,必須檢查BUSY位以確定擦除周期結(jié)束。如果BUSY = 1訪問Flash,這是一個訪問沖突,這時ACCVIFG將被設(shè)置,而擦除的結(jié)果將是不可預(yù)測的的。
從RAM中啟動擦除操作時,過程如下:
要在擦除之前確認(rèn)沒有訪問Flash,然后擦除完成之前不允許訪問Flash。
; Segment Erase from RAM. 514 kHz < SMCLK < 952 kHz ; Assumes ACCVIE = NMIIE = OFIE = 0. MOV #WDTPW+WDTHOLD,&WDTCTL ; Disable WDT DINT ; Disable interrupts L1 BIT #BUSY,&FCTL3 ; Test BUSY JNZ L1 ; Loop while busy MOV #FWKEY+FSSEL1+FN0,&FCTL2; SMCLK/2 MOV #FWKEY,&FCTL3 ; Clear LOCK MOV #FWKEY+ERASE,&FCTL1 ; Enable erase CLR &0FC10h ; Dummy write, erase S1 L2 BIT #BUSY,&FCTL3 ; Test BUSY JNZ L2 ; Loop while busy MOV #FWKEY+LOCK,&FCTL3 ; Done, set LOCK ... ; Re-enable WDT? EINT ; Enable interrupts
寫Flash操作:寫入的模式由WRT和BLKWRT位來確定:
BLKWRT WRT Write Mode 0 1 Byte/word write 1 1 Block write
這兩種模式中塊寫入大約是字或字節(jié)寫操作時的兩倍快,因?yàn)樵趬K寫入完成之前,變成電壓一直維持直到塊寫入完成。同一個位置不能在擦除周期之前寫入兩次或以上,否則將發(fā)生數(shù)據(jù)損壞。寫操作時,BUSY位被置1,寫入完成后,BUSY被自動清零。如果寫操作是從RAM發(fā)起的,在BUSY=1時,程序不能訪問Flash,否則會發(fā)生訪問沖突,置位ACCVIFG,F(xiàn)lash寫入操作不可以預(yù)料。
字或字節(jié)寫入:字或字節(jié)寫入可以從Flash內(nèi)部發(fā)起,也可以從RAM中發(fā)起。如果是從Flash中啟動的寫操作,時序?qū)⒂蒄lash控制,在寫入完成之前CPU運(yùn)行將被暫停。寫入完成后CPU將繼續(xù)運(yùn)行。
操作時序如下:
若是從RAM中啟動寫Flash,程序?qū)⒗^續(xù)從RAM中運(yùn)行。CPU再次訪問Flash之前必須確認(rèn)BUSY位已經(jīng)清零,否則會發(fā)生訪問沖突,置位ACCVIFG,寫入的結(jié)果將不可預(yù)料。
字或字節(jié)寫入模式下,內(nèi)部產(chǎn)生的編程電壓時適用于完整的64個字節(jié)塊的寫入 In byte/word mode, the internally-generated programming voltage is applied to the complete 64-byte block, each time a byte or word is written, for 32 of the 35 fFTG cycles. With each byte or word write, the amount of time the block is subjected to the programming voltage accumulates. The cumulative programming time, tCPT, must not be exceeded for any block. If the cumulative programming time is met, the block must be erased before performing any further writes to any address within the block.
從Flash發(fā)起寫字節(jié)或字時:
; Byte/word write from flash. 514 kHz < SMCLK < 952 kHz ; Assumes 0FF1Eh is already erased ; Assumes ACCVIE = NMIIE = OFIE = 0. MOV #WDTPW+WDTHOLD,&WDTCTL ; Disable WDT DINT ; Disable interrupts MOV #FWKEY+FSSEL1+FN0,&FCTL2 ; SMCLK/2 MOV #FWKEY,&FCTL3 ; Clear LOCK MOV #FWKEY+WRT,&FCTL1 ; Enable write MOV #0123h,&0FF1Eh ; 0123h ?> 0FF1Eh MOV #FWKEY,&FCTL1 ; Done. Clear WRT MOV #FWKEY+LOCK,&FCTL3 ; Set LOCK ... ; Re-enable WDT? EINT ; Enable interrupts
從RAM中啟動寫入操作時:
; Byte/word write from RAM. 514 kHz < SMCLK < 952 kHz ; Assumes 0FF1Eh is already erased ; Assumes ACCVIE = NMIIE = OFIE = 0. MOV #WDTPW+WDTHOLD,&WDTCTL ; Disable WDT DINT ; Disable interrupts L1 BIT #BUSY,&FCTL3 ; Test BUSY JNZ L1 ; Loop while busy MOV #FWKEY+FSSEL1+FN0,&FCTL2 ; SMCLK/2 MOV #FWKEY,&FCTL3 ; Clear LOCK MOV #FWKEY+WRT,&FCTL1 ; Enable write MOV #0123h,&0FF1Eh ; 0123h ?> 0FF1Eh L2 BIT #BUSY,&FCTL3 ; Test BUSY JNZ L2 ; Loop while busy MOV #FWKEY,&FCTL1 ; Clear WRT MOV #FWKEY+LOCK,&FCTL3 ; Set LOCK ... ; Re-enable WDT? EINT ; Enable interrupts
塊寫入:當(dāng)需要寫入連續(xù)的多個字或字節(jié)時,塊寫入能夠能夠提高Flash訪問速度。塊寫入時,內(nèi)部產(chǎn)生的編程電壓一直存在在64個字節(jié)的塊寫入期間。塊寫入不能有Flash存儲器內(nèi)啟動,必須從RAM中發(fā)起塊寫入操作。塊寫入期間,BUSY位被置位。在寫入每個字節(jié)或字時必須檢測WAIT位。下一個字或字節(jié)能夠被寫入時,WAIT位被置位。
塊寫入的過程如下:
; Write one block starting at 0F000h. ; Must be executed from RAM, Assumes Flash is already erased. ; 514 kHz < SMCLK < 952 kHz ; Assumes ACCVIE = NMIIE = OFIE = 0. MOV #32,R5 ; Use as write counter MOV #0F000h,R6 ; Write pointer MOV #WDTPW+WDTHOLD,&WDTCTL ; Disable WDT DINT ; Disable interrupts L1 BIT #BUSY,&FCTL3 ; Test BUSY JNZ L1 ; Loop while busy MOV #FWKEY+FSSEL1+FN0,&FCTL2 ; SMCLK/2 MOV #FWKEY,&FCTL3 ; Clear LOCK MOV #FWKEY+BLKWRT+WRT,&FCTL1 ; Enable block write L2 MOV Write_Value,0(R6) ; Write location L3 BIT #WAIT,&FCTL3 ; Test WAIT JZ L3 ; Loop while WAIT=0 INCD R6 ; Point to next word DEC R5 ; Decrement write counter JNZ L2 ; End of block? MOV #FWKEY,&FCTL1 ; Clear WRT,BLKWRT L4 BIT #BUSY,&FCTL3 ; Test BUSY JNZ L4 ; Loop while busy MOV #FWKEY+LOCK,&FCTL3 ; Set LOCK ... ; Re-enable WDT if needed EINT ; Enable interrupts
當(dāng)任何寫或擦除操作是從RAM啟動,而BUSY = 1,CPU不能讀取或?qū)懭牖驈娜魏蜦lash位置。否則,發(fā)生訪問沖突,ACCVIFG設(shè)置,結(jié)果是不可預(yù)知的。此外,如果閃存寫入讓W(xué)RT= 0,ACCVIFG中斷標(biāo)志設(shè)置,F(xiàn)lash不受影響。
如果寫入或擦除操作時從Flash啟動的,CPU訪問下一條指令時(從Flash讀取指令),F(xiàn)lash控制器返回03FFFH給CPU;03FFFH是指令JMP PC,這讓CPU一直循環(huán)直到Flash操作完成。Flash寫入或擦除操作完成后,允許CPU繼續(xù)訪問接下來的指令。
當(dāng)BUSY=1時,F(xiàn)lash訪問時:
在開始Flash操作之前,需要停止所有的中斷源。如果在Flash操作期間有中斷響應(yīng),讀中斷服務(wù)程序的地址時,將收到03FFFH作為中斷服務(wù)程序的地址。如果BUSY=1;CPU將一直執(zhí)行難IMP PC指令;Flash操作完成后,將從03FFFH執(zhí)行中斷服務(wù)程序而不是正確的中斷程序的地址。
停止寫入或擦除:任何寫入和擦除操作都可以在正常完成之前,通過設(shè)置緊急退出位EMEX退出操作。設(shè)置EMEX時,立即停止當(dāng)前活動的操作,停止Flash控制器;所有的Flash操作停止,F(xiàn)lash返回可讀模式,F(xiàn)CTL1的所有位復(fù)位;操作的結(jié)果不可預(yù)料。
設(shè)置和訪問Flash控制器:FCTLx是16位的、密碼保護(hù)的、可讀寫的寄存器。寫入這些寄存器都必須在高位包含密碼0A5H,如果寫入的不是0A5H,將會引起復(fù)位。讀寄存器時高位讀出的是96H。
在擦除或?qū)懭胱只蜃止?jié)時寫FCTL1寄存器將會引起訪問沖突,置位ACCVIFG.塊寫入時,WAIT=1時可以寫FCTL1寄存器,當(dāng)WAIT=0時寫FCTL1寄存器是訪問沖突,置位ACCVIFG。BUSY=1時,所有寫入FCTL2寄存器都是訪問沖突。BUSY=1時,所有的FCTLx都可以讀操作,不會引起訪問沖突。
Flash的中斷:Flash控制器有兩個中斷源:KEYV, 和ACCVIFG。ACCVIFG在訪問沖突的時候被置位。當(dāng)ACCVIE在Flash操作完成后被重新使能后ACCVIFG會引起中斷請求。ACCIFG和NMI同樣的中斷向量,所以這個中斷不需要GIE位允許即可產(chǎn)生中斷請求。必須通過軟件檢測ACCVIFG位,以確定發(fā)生了訪問沖突;ACCVIFG位必須軟件復(fù)位。KEYV是關(guān)鍵值錯誤當(dāng)寫Flash的寄存器時沒有寫正確的高位密碼時被置位,這是會立刻引起PUC信號復(fù)位整個硬件。
編程Flash的硬件:編程430的Flash內(nèi)容有三種選擇,通過JTAG、通過BSL和用戶定制。用戶定制即是通過單片機(jī)的程序訪問自己的Flash。
Flash的寄存器列表如下:
Register Short Form Register Type Address Initial State Flash memory control register 1 FCTL1 Read/write 0128h 09600h with PUC Flash memory control register 2 FCTL2 Read/write 012Ah 09642h with PUC Flash memory control register 3 FCTL3 Read/write 012Ch 09618h with PUC Interrupt Enable 1 IE1 Read/write 000h Reset with PUC
Flash的硬件部分就介紹這么多了,有什么不大懂的地方請參考TI提供的用戶指南。
程序?qū)崿F(xiàn):
首先設(shè)置Flash的時鐘,初始化Flash控制器:
void FlashInit() { FCTL2 = FWKEY + FSSEL_2 + FN1; // 默認(rèn) SMCLK/3 =333KHz }
這個函數(shù)僅僅設(shè)置了時鐘。
擦除函數(shù):
void FlashErase(unsigned int Addr) { char *FlashPtr; FlashPtr = (char *)Addr; FCTL1 = FWKEY + ERASE; // Set Erase bit FCTL3 = FWKEY; // Clear Lock bit DINT; *FlashPtr = 0; // Dummy write to erase Flash segment B WaitForEnable(); //Busy EINT; FCTL1 = FWKEY; // Lock FCTL3 = FWKEY + LOCK; // Set Lock bit }
這個和上面給出的流程一樣,參數(shù)是要被擦除的段的首地址。WaitForEnable函數(shù)等等待BUSY標(biāo)志變回零即操作完成。
void WaitForEnable() { while((FCTL3 & BUSY) == BUSY); //Busy }
寫入字節(jié):
void FlashWriteChar(unsigned int addr,char Data) { char *FlashPtr = (char *)addr; // Segment A pointer FCTL1 = FWKEY + WRT; // Set WRT bit for write operation FCTL3 = FWKEY; // Clear Lock bit DINT; *FlashPtr = Data; // Save Data WaitForEnable(); //Busy EINT; FCTL1 = FWKEY; // Clear WRT bit FCTL3 = FWKEY + LOCK; // Set LOCK bit }
寫入字:
void FlashWriteWord(unsigned int addr,unsigned int Data) { unsigned int *FlashPtr = (unsigned int *)addr; FCTL1 = FWKEY + WRT; // Set WRT bit for write operation FCTL3 = FWKEY; // Clear Lock bit DINT; *FlashPtr = Data; // Save Data WaitForEnable(); //Busy EINT; FCTL1 = FWKEY; // Clear WRT bit FCTL3 = FWKEY + LOCK; // Set LOCK bit }
寫入字或字節(jié)兩個函數(shù)差別僅僅是指針類型不同。
讀取字或字節(jié):
char FlashReadChar(unsigned int Addr) { char Data; char *FlashPtr = (char *) Addr; Data = *FlashPtr; return(Data); } unsigned int FlashReadWord(unsigned int Addr) { unsigned int Data; unsigned int *FlashPtr = (unsigned int *)Addr; Data = *FlashPtr; return(Data); } 這兩個函數(shù)的差別也是僅僅指針類型不同。
這些函數(shù)和前面硬件介紹部分的程序流程相同,這里不再詳細(xì)說明。
使用示例:
使用方法和之前的一樣,工程中加入C文件,源代碼文件中文件包含H文件,即可使用,具體參考示例項(xiàng)目:
演示主要程序主要如下:
#include#include "Flash.h" int a; void main( void ) { // Stop watchdog timer to prevent time out reset WDTCTL = WDTPW + WDTHOLD; ClkInit(); FlashInit(); FlashWriteChar(InfoB,0x25); a=FlashReadChar (InfoB); //InfoB在H文件中有宏定義 FlashWriteWord(InfoB+2,0x5669); a = FlashReadWord(InfoB+2); FlashErase(InfoB); LPM0; }這里向InfoB(1000h)首地址開始寫數(shù)據(jù),先寫一個字節(jié) 再寫入一個字(注意寫入字時,必須是偶數(shù)地址,奇數(shù)地址會寫在這個地址所在的前一個偶數(shù)地址),讀出,然后擦除。這里的程序都是在Flash中運(yùn)行的,沒有演示RAM中運(yùn)行的程序。如果在RAM運(yùn)行程序,則需要先把程序從Flash中移到RAM中,然后跳轉(zhuǎn)到RAM中運(yùn)行。
調(diào)試截圖如下:
調(diào)試時,view-Memory菜單,調(diào)出存儲器窗口;觀察Flash內(nèi)容。
這里寫入的是Info Flash部分,觀察這部分的結(jié)果,和寫入的結(jié)果同。
Flash程序控制器的程序就到這兒了。Flash可以用于存儲長期保存的數(shù)據(jù)。
編輯:hfy
-
單片機(jī)
+關(guān)注
關(guān)注
6035文章
44554瀏覽量
634660 -
FlaSh
+關(guān)注
關(guān)注
10文章
1633瀏覽量
147944 -
Flash控制器
+關(guān)注
關(guān)注
0文章
6瀏覽量
8059
發(fā)布評論請先 登錄
相關(guān)推薦
評論