本帖最后由 51黑黑黑 于 2016-2-13 00:26 編輯
C8051F020的Flash可以實現IAP功能(InApplicationProgramming),即程序可對Flash存儲區進行編程,也就是說Flash除了可以存放程序代碼,還可以在剩下的空間內存儲數據,即起到EEPROM的功能,下圖是C8051F020的存儲組織結構圖
可以看到它的結構是64K+128字節的形式,0xfe00~0xffff為工廠保留區,不能進行操作,當64K的Flash用作存儲數據時,先確定程序代碼的位置,以免覆蓋程序導致出錯,還有操作Flash前必須禁止中斷,否則可能會出現無法預知的錯誤,下面是我寫的驅動,經測試沒問題:
void Flash_area_Erase(bit type,uint addr)
{//flash擦除,type為0代表128字節儲存區,為1代表64K存儲區(512字節/扇區)
bit EA_S;
EA_S=EA;
EA=0;
if(!type) PSCTL=0x07;
else PSCTL=0x03;
FLSCL=0x01;//允許寫
*(uchar xdata *)addr=0xff;
PSCTL=0;
FLSCL=0;
EA=EA_S;
}
uchar Flash_area_Read(bit type,uint addr)
{//flash讀取,type為0代表128字節儲存區,為1代表64K存儲區(512字節/扇區)
uchar dat;
bit EA_S;
EA_S=EA;
EA=0;
if(!type) PSCTL=0x04;
else PSCTL=0;
dat=*(uchar code *)addr;
PSCTL=0;
EA=EA_S;
return(dat);
}
void Flash_area_Write(bit type,uint addr,uchar dat)
{//flash寫入,type為0代表128字節儲存區,為1代表64K存儲區(512字節/扇區)
bit EA_S;
EA_S=EA;
EA=0;
if(!type) PSCTL=0x05;
else PSCTL=0x01;
FLSCL=0x01;
*(uchar xdata *)addr=dat;
PSCTL=0;
FLSCL=0;
EA=EA_S;
}
這里要指出的是,在*(uchar code*)addr中,addr轉化為指向ROM的uchar型變量的指針,即它指向的數是uchar型的,但其本身作為一個地址是uint型的。
保存在地址0xFDFE和0xFDFF中的安全鎖定字節集可以保護 FLASH存儲器,使得不能通過JTAG接口讀取或修改其內容。安全鎖定字節中的每一位保護一個8k字節的存儲器塊。將讀鎖定字節中的一位清0可防止通過JTAG接口讀對應的FLASH存儲器塊將寫/擦除鎖定字節中的一位清0可防止通過JTAG接口寫/擦除對應的存儲器塊。當所有其它扇區都被鎖定后128字節的扇區才被鎖定。

使用FLASH訪問限制(FLACL)這一特殊功能寄存器來設置軟件讀限制。所希望的軟件訪問限制地址(設計者希望的軟件訪問保護的最高地址)的高字節被裝入FLACL寄存器。該限制地址的低字節為0x00。如果FLACL寄存器被賦值為0x40,則軟件訪問限制地址為0x4000。所有位于地址0x0000到0x4000(包括0x4000)的存儲區內的代碼將不能被在該地址以上執行的軟件訪問。在FLACL邊界之上執行的程序可以用跳轉或調用指令進入到FLACL邊界以下的保護存儲區。軟件讀限制只對MOVX和MOVC操作起作用。為了防止對FLASH的訪問,應使用FLASH安全字節保護0x4000以下的存儲區或進行整體保護。
這里要提到一個軟件Flash ProgrammingUtilities,在用Keil編譯完成后生成hex文件,需要量產時,可以按下圖操作
即下程序前擦出真個Flash,下完后對代碼進行鎖定,防止被讀寫;對于鎖定了代碼的MCU,需要在FlashErase選項卡中擦除,否則在下面第二個圖顯示的界面中是無法用連接上的
|