一般,在單片機中的Flash存儲器用于存放程序代碼,屬于只讀型存儲器。而在MSP430些列的單片機中,都可以通過內置的Flash控制器擦除或改寫任何一段的內容。另外,msp430的單片機內部還專門留有一段Flash區(qū)域(information memory),用于存放掉電后需要永久保存的數(shù)據(jù)。利用430內部的Flash控制器,可以完成較大容量的數(shù)據(jù)記錄、用戶設置參數(shù)在掉電后的保存等功能。
硬件介紹:
要對Flash讀寫,首先要了解MSP430的存儲器組織。430單片機的存儲器組織結構采用馮諾依曼結構,RAM和ROM統(tǒng)一編址在同一尋址空間中,沒有代碼空間和數(shù)據(jù)空間之分。
一般430的單片機都統(tǒng)一編址在0-64k地址范圍中,只有少數(shù)高端的型號才能突破64k(如:FG461x系列)。絕大多數(shù)的msp430單片機都編址在64kB范圍內。地址的大概編碼方式如下:

這是msp430f425的存儲器分配圖,其他在64k范圍內的存儲器的單片機編址方式與此類似:低256B是寄存器區(qū),然后是RAM;空白;1000H到10FFH是信息Flash區(qū);大于1100H-0FFFFH是主存儲器區(qū)(從0FFFFH開始往低地址有單片機的主Flash,多余的部分空白)。
MSP430F14x的Flash分布:

MSP430F16x的Flash分布:

主Flash部分和信息Flash部分如下(60kB Flash對應的單片機,如msp430f149、msp430f149):
主Flash分為以512B為段的單位,0段是單片機中斷向量等程序入口地址,使用時不要擦除此段或改寫此段,若要擦除或是改寫,請先保存內容到RAM或其他段;主Flash各段內容均要避免寫入或擦除,以免造成不可預料的后果。
信息Flash分為兩段:段A和段B,每段128B;可以保存用戶自己的內容(主Flash也可以但是要避免與程序代碼區(qū)沖突);這里就把信息Flash的兩段稱為InfoA(1080H-10FFh)和InfoB(1000H-10FFH)。
Flash的操作包括:字或字節(jié)寫入;塊寫入;段擦除;主Flash擦除;全部擦除。任何的Flash操作都可以從Flash或從RAM中運行。
Flash操作時需要時序發(fā)生器,F(xiàn)lash控制器內部含有時序發(fā)生器用以產生所需的Flash時鐘,F(xiàn)lash時鐘的范圍必須在257kHz到476kHz之間。時序發(fā)生器的框圖如下:

時序發(fā)生器可以選擇ACLK、MCLK、SMCLK作為時鐘源,通過分頻獲得所需的257kHz到476kHz之間的Flash操作時鐘。如果時鐘頻率不再這個范圍內,將會產生不可預料的結果。
擦除:擦除之后,存儲器中的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位來設置擦除的模式:段擦除,主Flash擦除,全部擦除。
對要擦除段內的一個地址空寫入啟動擦出周期:空寫入可以啟動時序發(fā)生器和擦除操作??諏懭牒驜USY位立即變高直到擦除周期結束,這一位變?yōu)榈?0)。BUSY, MERAS和 ERASE位在擦除周期結束后會自動復位。擦除周期的時間和要擦出的Flash大小無關,每次擦除的時間對于MSP430F1xx系系列單片機來說,所需時間是一樣的。擦除的時序如下:

當空寫入到的地址不在要擦除的段地址范圍內的時候,空寫入無效,直接被忽略。在擦除周期內,應該關中斷,直到擦除完成,重新開中斷,擦除期間的中斷已經置標志位,開中斷后立即響應。
從Flash中啟動的擦除操作:擦除操作可以從Flash中啟動或是從RAM中啟動。當操作是從Flash中啟動的時候,F(xiàn)lash控制器控制了操作時序,CPU運行被暫停直到擦除結束。擦除周期結束后,CPU繼續(xù)執(zhí)行,從空寫入之后的指令開始運行。當從Flash中啟動擦除操作時,可以擦除即將運行的程序所在的段,如果擦除了即將運行的程序所在的Flash段時,擦除結束后,CPU的運行不可預料。
從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中運行接下來的程序。CPU可以訪問任何Flash地址之前,必須檢查BUSY位以確定擦除周期結束。如果BUSY = 1訪問Flash,這是一個訪問沖突,這時ACCVIFG將被設置,而擦除的結果將是不可預測的的。
從RAM中啟動擦除操作時,過程如下:

要在擦除之前確認沒有訪問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é)寫操作時的兩倍快,因為在塊寫入完成之前,變成電壓一直維持直到塊寫入完成。同一個位置不能在擦除周期之前寫入兩次或以上,否則將發(fā)生數(shù)據(jù)損壞。寫操作時,BUSY位被置1,寫入完成后,BUSY被自動清零。如果寫操作是從RAM發(fā)起的,在BUSY=1時,程序不能訪問Flash,否則會發(fā)生訪問沖突,置位ACCVIFG,F(xiàn)lash寫入操作不可以預料。
字或字節(jié)寫入:字或字節(jié)寫入可以從Flash內部發(fā)起,也可以從RAM中發(fā)起。如果是從Flash中啟動的寫操作,時序將由Flash控制,在寫入完成之前CPU運行將被暫停。寫入完成后CPU將繼續(xù)運行。
操作時序如下:

若是從RAM中啟動寫Flash,程序將繼續(xù)從RAM中運行。CPU再次訪問Flash之前必須確認BUSY位已經清零,否則會發(fā)生訪問沖突,置位ACCVIFG,寫入的結果將不可預料。
字或字節(jié)寫入模式下,內部產生的編程電壓時適用于完整的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
塊寫入:當需要寫入連續(xù)的多個字或字節(jié)時,塊寫入能夠能夠提高Flash訪問速度。塊寫入時,內部產生的編程電壓一直存在在64個字節(jié)的塊寫入期間。塊寫入不能有Flash存儲器內啟動,必須從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
當任何寫或擦除操作是從RAM啟動,而BUSY = 1,CPU不能讀取或寫入或從任何Flash位置。否則,發(fā)生訪問沖突,ACCVIFG設置,結果是不可預知的。此外,如果閃存寫入讓WRT= 0,ACCVIFG中斷標志設置,F(xiàn)lash不受影響。
如果寫入或擦除操作時從Flash啟動的,CPU訪問下一條指令時(從Flash讀取指令),F(xiàn)lash控制器返回03FFFH給CPU;03FFFH是指令JMP PC,這讓CPU一直循環(huán)直到Flash操作完成。Flash寫入或擦除操作完成后,允許CPU繼續(xù)訪問接下來的指令。
當BUSY=1時,F(xiàn)lash訪問時:

在開始Flash操作之前,需要停止所有的中斷源。如果在Flash操作期間有中斷響應,讀中斷服務程序的地址時,將收到03FFFH作為中斷服務程序的地址。如果BUSY=1;CPU將一直執(zhí)行難IMP PC指令;Flash操作完成后,將從03FFFH執(zhí)行中斷服務程序而不是正確的中斷程序的地址。
停止寫入或擦除:任何寫入和擦除操作都可以在正常完成之前,通過設置緊急退出位EMEX退出操作。設置EMEX時,立即停止當前活動的操作,停止Flash控制器;所有的Flash操作停止,F(xiàn)lash返回可讀模式,F(xiàn)CTL1的所有位復位;操作的結果不可預料。
設置和訪問Flash控制器:FCTLx是16位的、密碼保護的、可讀寫的寄存器。寫入這些寄存器都必須在高位包含密碼0A5H,如果寫入的不是0A5H,將會引起復位。讀寄存器時高位讀出的是96H。
在擦除或寫入字或字節(jié)時寫FCTL1寄存器將會引起訪問沖突,置位ACCVIFG.塊寫入時,WAIT=1時可以寫FCTL1寄存器,當WAIT=0時寫FCTL1寄存器是訪問沖突,置位ACCVIFG。BUSY=1時,所有寫入FCTL2寄存器都是訪問沖突。BUSY=1時,所有的FCTLx都可以讀操作,不會引起訪問沖突。
Flash的中斷:Flash控制器有兩個中斷源:KEYV, 和ACCVIFG。ACCVIFG在訪問沖突的時候被置位。當ACCVIE在Flash操作完成后被重新使能后ACCVIFG會引起中斷請求。ACCIFG和NMI同樣的中斷向量,所以這個中斷不需要GIE位允許即可產生中斷請求。必須通過軟件檢測ACCVIFG位,以確定發(fā)生了訪問沖突;ACCVIFG位必須軟件復位。KEYV是關鍵值錯誤當寫Flash的寄存器時沒有寫正確的高位密碼時被置位,這是會立刻引起PUC信號復位整個硬件。
編程Flash的硬件:編程430的Flash內容有三種選擇,通過JTAG、通過BSL和用戶定制。用戶定制即是通過單片機的程序訪問自己的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提供的用戶指南。
程序實現(xiàn):
首先設置Flash的時鐘,初始化Flash控制器:
void FlashInit()
{
FCTL2 = FWKEY + FSSEL_2 + FN1; // 默認 SMCLK/3 =333KHz
}
這個函數(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標志變回零即操作完成。
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ù)和前面硬件介紹部分的程序流程相同,這里不再詳細說明。
使用示例:
使用方法和之前的一樣,工程中加入C文件,源代碼文件中文件包含H文件,即可使用,具體參考示例項目:
演示主要程序主要如下:
#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中運行的,沒有演示RAM中運行的程序。如果在RAM運行程序,則需要先把程序從Flash中移到RAM中,然后跳轉到RAM中運行。
調試截圖如下:
調試時,view-Memory菜單,調出存儲器窗口;觀察Flash內容。
這里寫入的是Info Flash部分,觀察這部分的結果,和寫入的結果同。
Flash程序控制器的程序就到這兒了。Flash可以用于存儲長期保存的數(shù)據(jù)。
編輯:hfy
-
單片機
+關注
關注
6074文章
45322瀏覽量
662996 -
FlaSh
+關注
關注
10文章
1715瀏覽量
154637 -
Flash控制器
+關注
關注
0文章
7瀏覽量
8330
發(fā)布評論請先 登錄
MSP430常用內聯(lián)函數(shù)說明
MSP-FET430閃存仿真工具(FET)手冊
用msp430單片機如何設計軟件濾波?
單片機MSP430引腳直接接7883的引腳進行控制,ADS7883無數(shù)據(jù)輸出是什么原因?
用MSP430單片機SPI與TLV5638通訊,TLV5638輸出都是0是什么原因?
請問有沒有辦法對MSP430的固件升級?
請問ADS112C04作為一個模數(shù)轉化器,怎么和msp430單片機相連,進行編程呢?
MSP430電容式觸摸電源設計器GUI用戶指南
適用于MSP430和Simplelink? MSP432?? MCU的MSP-BSL引導加載程序
MSP430微控制器串行編程適配器用戶指南

一文了解msp430單片機的存儲器組織
評論