近日研究STC12C2051的EEPROM的應用,發現官方只給出其匯編語言版本的程序。只有其公司最新推出的STC11/10xx系列的單片機有EEPROM的C語言測試程序。在網上搜索未果后,便自己動手修改,并將STC12C2052系列和STC11/10xx系列單片機的EEPROM測試程序整合起來,制作了一個適用性更好的程序。經在STC12C2052單片機硬件上測試正常,特此與大家分享!
/*********************************************************************************************
程序名:STC系列單片機內部EEPROM 測試程序
編寫人:杜洋
時間:2009-5-17
修改日志:
硬件電路:
P1口接8個LED到VCC。
適用硬件:
STC12C2052系列(杜洋改進部分)
STC12C5AxxAD系列單片機
STC12C52xxAD系列單片機
STC11xx系列單片機
STC10xx系列單片機
使用說明:
1,程序先別P1口高4位和低4位分別點亮一次。
2,檢查EEPROM中對應地址內的值是否與用戶測試值相同。
3,如果相同則P1.7上的LED亮,然后在P1口顯示EEPROM中的值。
4,如果不同則P1.3上的LED亮,然后全片擦除EEPROM并寫入用戶測試值到指定地址。
注意:
# 在第一次下載時為寫入(第4步),復位后才會測試。
# 當供電電源低于一定值時將不能寫入EEPROM,詳見數據手冊。
聲明:
本程序基于宏晶公司STC11/10xx系列單片機EEPROM測試程序之上編寫,詳見官方數據手冊。
宏晶STC官網:www.mcu-memory.com
/*********************************************************************************************/
#include <reg51.H>
#include <intrins.H>
/*********************************************************************************************/
typedef unsigned char INT8U;
typedef unsigned int INT16U;
/*********************************************************************************************/
//用于STC12C2052系列單片機時選擇//
sfr IAP_DATA = 0xE2; //STC12C2052系列單片機的EEPRON操作地址是0xe2(以下類推)
sfr IAP_ADDRH = 0xE3;
sfr IAP_ADDRL = 0xE4;
sfr IAP_CMD = 0xE5;
sfr IAP_TRIG = 0xE6;
sfr IAP_CONTR = 0xE7;
#define WD1 0x46 //使用STC12C2052系列單片機時,先寫入0x46,然寫入0xb9
#define WD2 0xb9
/*********************************************************************************************
//用于STC11/10xx系列單片機時選擇//
sfr IAP_DATA = 0xC2; //STC11xx系列單片機的EEPRON操作地址是0xc2(以下類推)
sfr IAP_ADDRH = 0xC3;
sfr IAP_ADDRL = 0xC4;
sfr IAP_CMD = 0xC5;
sfr IAP_TRIG = 0xC6;
sfr IAP_CONTR = 0xC7;
#define WD1 0x5a //使用STC11xx系列單片機時,先寫入0x5a,然寫入0xa5
#define WD2 0xa5
/*********************************************************************************************/
//定義Flash 操作等待時間及允許IAP/ISP/EEPROM 操作的常數//
//#define ENABLE_ISP 0x80 //系統工作時鐘<30MHz 時,對IAP_CONTR 寄存器設置此值
//#define ENABLE_ISP 0x81 //系統工作時鐘<24MHz 時,對IAP_CONTR 寄存器設置此值
#define ENABLE_ISP 0x82 //系統工作時鐘<20MHz 時,對IAP_CONTR 寄存器設置此值
//#define ENABLE_ISP 0x83 //系統工作時鐘<12MHz 時,對IAP_CONTR 寄存器設置此值
//#define ENABLE_ISP 0x84 //系統工作時鐘<6MHz 時,對IAP_CONTR 寄存器設置此值
//#define ENABLE_ISP 0x85 //系統工作時鐘<3MHz 時,對IAP_CONTR 寄存器設置此值
//#define ENABLE_ISP 0x86 //系統工作時鐘<2MHz 時,對IAP_CONTR 寄存器設置此值
//#define ENABLE_ISP 0x87 //系統工作時鐘<1MHz 時,對IAP_CONTR 寄存器設置此值
/*********************************************************************************************/
#define DEBUG_DATA 0x55 //存儲在 EEPROM 單元的數值(用戶可修改測試)
#define DATA_FLASH_START_ADDRESS 0x00 //EEPROM存入地址(用戶可修改測試)
/*********************************************************************************************/
union union_temp16
{
INT16U un_temp16;
INT8U un_temp8[2];
}my_unTemp16;
INT8U Byte_Read(INT16U add); //讀一字節,調用前需打開IAP 功能
void Byte_Program(INT16U add, INT8U ch); //字節編程,調用前需打開IAP 功能
void Sector_Erase(INT16U add); //擦除扇區
void IAP_Disable(); //關閉IAP 功能
void Delay();
/*********************************************************************************************/
void main (void)
{
INT16U eeprom_address;
INT8U read_eeprom;
P1 = 0xF0; //演示程序開始,讓 P1[3:0] 控制的燈亮
Delay(); //延時
P1 = 0x0F; //演示程序開始,讓 P1[7:4] 控制的燈亮
Delay() ; //延時
//將EEPROM 測試起始地址單元的內容讀出
eeprom_address = DATA_FLASH_START_ADDRESS; //將測試起始地址送eeprom_address
read_eeprom = Byte_Read(eeprom_address); //讀EEPROM的值,存到read_eeprom
if (DEBUG_DATA == read_eeprom)
{ //數據是對的,亮 P1.7 控制的燈,然后在 P1 口上將 EEPROM 的數據顯示出來
P1 = ~0x80;
Delay() ; //延時
P1 = ~read_eeprom;
}
else
{ //數據是錯的,亮 P1.3 控制的燈,然后在 P1 口上將 EEPROM 的數據顯示出來
//再將該EEPROM所在的扇區整個擦除,將正確的數據寫入后,亮 P1.5 控制的燈
P1 = ~0x08;
Delay() ; //延時
P1 = ~read_eeprom;
Delay() ; //延時
Sector_Erase(eeprom_address); //擦除整個扇區
Byte_Program(eeprom_address, DEBUG_DATA);//將 DEBUG_DATA 寫入 EEPROM
P1 = ~0x20; //熄滅 P1.3 控制的燈,亮 P1.5 控制的燈
}
while (1); //CPU 在此無限循環執行此句
}
/*********************************************************************************************/
//讀一字節,調用前需打開IAP 功能,入口:DPTR = 字節地址,返回:A = 讀出字節
INT8U Byte_Read(INT16U add)
{
IAP_DATA = 0x00;
IAP_CONTR = ENABLE_ISP; //打開IAP 功能, 設置Flash 操作等待時間
IAP_CMD = 0x01; //IAP/ISP/EEPROM 字節讀命令
my_unTemp16.un_temp16 = add;
IAP_ADDRH = my_unTemp16.un_temp8[0]; //設置目標單元地址的高8 位地址
IAP_ADDRL = my_unTemp16.un_temp8[1]; //設置目標單元地址的低8 位地址
//EA = 0;
IAP_TRIG = WD1; //先送 WD1,再送WD2 到ISP/IAP 觸發寄存器,每次都需如此
IAP_TRIG = WD2; //送完WD2 后,ISP/IAP 命令立即被觸發起動
_nop_();
//EA = 1;
IAP_Disable(); //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
//一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
return (IAP_DATA);
}
/*********************************************************************************************/
//字節編程,調用前需打開IAP 功能,入口:DPTR = 字節地址, A= 須編程字節的數據
void Byte_Program(INT16U add, INT8U ch)
{
IAP_CONTR = ENABLE_ISP; //打開 IAP 功能, 設置Flash 操作等待時間
IAP_CMD = 0x02; //IAP/ISP/EEPROM 字節編程命令
my_unTemp16.un_temp16 = add;
IAP_ADDRH = my_unTemp16.un_temp8[0]; //設置目標單元地址的高8 位地址
IAP_ADDRL = my_unTemp16.un_temp8[1]; //設置目標單元地址的低8 位地址
IAP_DATA = ch; //要編程的數據先送進IAP_DATA 寄存器
//EA = 0;
IAP_TRIG = WD1; //先送 WD1,再送WD2 到ISP/IAP 觸發寄存器,每次都需如此
IAP_TRIG = WD2; //送完WD2 后,ISP/IAP 命令立即被觸發起動
_nop_();
//EA = 1;
IAP_Disable(); //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
//一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
}
/*********************************************************************************************/
//擦除扇區, 入口:DPTR = 扇區地址
void Sector_Erase(INT16U add)
{
IAP_CONTR = ENABLE_ISP; //打開IAP 功能, 設置Flash 操作等待時間
IAP_CMD = 0x03; //IAP/ISP/EEPROM 扇區擦除命令
my_unTemp16.un_temp16 = add;
IAP_ADDRH = my_unTemp16.un_temp8[0]; //設置目標單元地址的高8 位地址
IAP_ADDRL = my_unTemp16.un_temp8[1]; //設置目標單元地址的低8 位地址
//EA = 0;
IAP_TRIG = WD1; //先送 WD1,再送WD2 到ISP/IAP 觸發寄存器,每次都需如此
IAP_TRIG = WD2; //送完WD2 后,ISP/IAP 命令立即被觸發起動
_nop_();
//EA = 1;
IAP_Disable(); //關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
//一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
}
/*********************************************************************************************/
void IAP_Disable()
{
//關閉IAP 功能, 清相關的特殊功能寄存器,使CPU 處于安全狀態,
//一次連續的IAP 操作完成之后建議關閉IAP 功能,不需要每次都關
IAP_CONTR = 0; //關閉IAP 功能
IAP_CMD = 0; //清命令寄存器,使命令寄存器無命令,此句可不用
IAP_TRIG = 0; //清命令觸發寄存器,使命令觸發寄存器無觸發,此句可不用
IAP_ADDRH = 0;
IAP_ADDRL = 0;
}
/*********************************************************************************************/
void Delay() //延時程序
{
INT8U i;
INT16U d=5000;
while (d--)
{
i=255;
while (i--);
}
}