24CXX讀寫程序(適用PIC和51,適用24C01~24C2048)
//24C02,24C04,24C1024測試通過
//-------------------讀寫串行EEPROM-------------
//作者:蘭天白云
//功能描述:讀寫串行EEPROM(適用24C01~24C2048)
//輸入:MCU地址,EEP地址,讀寫字節(jié)數(shù),24的控制字
//輸出:錯(cuò)誤標(biāo)志,0--正確,1--錯(cuò)誤
//版本說明:V1.1
//注:(1)寫數(shù)據(jù)時(shí)不用考慮跨頁問題(已有處理)
// (2)控制字與24的硬件連接有關(guān)
// 例:24的1,2,3,4腳都與地連接
// 則:寫24的控制字=0xa0,讀24的控制字=0xa1
//----------------------------------------------
//根據(jù)使用的24確定
#define PAGESIZE 8 //頁大小 24C02=8 24C512=128
#define EEPALLBYTE 256 //總字節(jié)數(shù) 24C02=256 24C512=65536
static bit EEP_flag; //0--運(yùn)行正確,1--運(yùn)行錯(cuò)誤
//根據(jù)使用的單片機(jī)確定(PIC)
#include
#define SDA_DIR TRISC2
#define SCL_DIR TRISC3
#define SDA RC2 //RC2---數(shù)據(jù)線
#define SCL RC3 //RC3---時(shí)鐘線
#define SDAinput() { SDA_DIR=1; } //設(shè)端口為輸入
#define SDAoutput() { SDA_DIR=0; } //輸出
#define SCLoutput() { SCL_DIR=0; }
//根據(jù)使用的單片機(jī)確定(51)
/*
#include
//#define SDA_DIR TRISC2
//#define SCL_DIR TRISC3
#define SDA P1^0 //數(shù)據(jù)線
#define SCL P1^1 //時(shí)鐘線
#define SDAinput() { SDA=1; } //設(shè)端口為輸入
#define SDAoutput() { _nop_(); } //為兼容PIC而保留
#define SCLoutput() { _nop_(); } //為兼容PIC而保留
*/
//指明由外部函數(shù)調(diào)用
extern bit RW24CXX(unsigned char *p_data,unsigned int eep_addr,
unsigned int n,unsigned char Control);
extern bit copy(unsigned int SSAddres,unsigned int DSAddres,unsigned int n);
extern bit insert(unsigned char *p_data,unsigned int eep_addr,
unsigned int n,unsigned char Control);
extern bit delete(unsigned int eep_addr,unsigned int n,unsigned char Control);
//IO口配置
static void EEP_IO(void)
{ SCLoutput();
SDAoutput();
}
//為了兼容大多數(shù)24CXX,至少延時(shí)2uS 與晶振有關(guān),定義成靜態(tài)函數(shù)
static void delayus(void)
{ unsigned char t1=0x01;
while(--t1);
}
////延時(shí)10mS 與晶振有關(guān),定義成靜態(tài)函數(shù)
static void delay10ms(void)
{ unsigned int t1=0xffff;
while(--t1);
}
//不管IO口以前怎么用
//從現(xiàn)在起用作IIC總線
static void I2C_Start(void) //啟動總線
{ EEP_IO();
SCL=0; //確保時(shí)鐘=低
delayus();
SDA=1;
delayus();
SCL=1;
delayus();
SDA=0; //啟動總線
delayus();
SCL=0;
delayus();
}
//停止時(shí),數(shù)據(jù)手冊要求時(shí)鐘和數(shù)據(jù)線都為1
//但本人認(rèn)為時(shí)鐘線設(shè)為0抗干擾更好
static void I2C_Stop(void) //停止總線
{ SCL=0;
delayus();
SDA=0;
delayus();
SCL=1;
delayus();
SDA=1;
delayus();
SCL=0; //設(shè)為0抗干擾更好
delayus();
}
static void I2C_RecAck(void) //讀應(yīng)答信號,用于寫
{
SCL=0;
SDAinput(); //設(shè)數(shù)據(jù)線為輸入
delayus();
SCL=1;
delayus();
EEP_flag=SDA;
SCL=0;
delayus();
SDAoutput();
delayus();
}
static void I2C_SendAck(void) //發(fā)送應(yīng)答信號,用于連續(xù)讀
{
SDA=0;
delayus();
SCL=1;
delayus();
SCL=0;
delayus();
}
static void I2C_NoAck(void) //不發(fā)送應(yīng)答信號,用于停止讀
{
SDA=1;
delayus();
SCL=1;
delayus();
SCL=0;
delayus();
}
//寫一字節(jié)到EEPROM
static void I2C_wbyte(unsigned char wbyte)
{unsigned char i=8;
//SDAoutput();
delayus();
for(;i>0;i--)
{SCL=0;
if(wbyte&0x80){SDA=1;}
else {SDA=0;}
delayus();
SCL=1;
wbyte<<=1;
delayus();
}
}
//從EEPROM讀1字節(jié)數(shù)據(jù)返回
static unsigned char I2C_rbyte(void)
{unsigned char i=8;
unsigned char rbyte;
SDAinput(); //設(shè)數(shù)據(jù)線為輸入
while(i--)
{SCL=1;
delayus();
rbyte=(rbyte<<1)|SDA;
SCL=0;
delayus();
}
SDAoutput();
return(rbyte);
}
//讀寫n字節(jié)數(shù)據(jù)
//數(shù)據(jù)地址由p_data確定,EEPROM地址由eep_addr確定
//數(shù)據(jù)數(shù)量由n確定,讀寫由Control確定
bit RW24CXX(unsigned char *p_data,unsigned int eep_addr,
unsigned int n,unsigned char Control)
{ unsigned int i=0,j=3;
while(1) //
{
I2C_Start();
I2C_wbyte(Control&0xfe);
I2C_RecAck();
if(EEP_flag)
break;
if(EEPALLBYTE>256) //有16位地址嗎?
{I2C_wbyte(eep_addr/256); //16位地址
I2C_RecAck();
if(EEP_flag)
break;
}
I2C_wbyte(eep_addr); //8位地址
I2C_RecAck();
if(EEP_flag)
break;
if(!(Control&0x01)) //寫----------
{while(n--)
{
I2C_wbyte(*p_data);
I2C_RecAck();
if(EEP_flag)
break;
p_data++;
eep_addr++;
if((eep_addr%PAGESIZE==0)) //跨頁
{i2c_stop();
delay10ms();
break;
}
}
if(n==0)
{i2c_stop();
delay10ms();
break;
}
else if(EEP_flag)
{if(j--==0) //允許重試3次
break;
}
}
else //讀-----------
{I2C_Start();
I2C_wbyte(Control);
I2C_RecAck();
if(EEP_flag)
break;
while(n--)
{*p_data++=I2C_rbyte();
if(n>0)
I2C_SendAck();
else
{I2C_NoAck();
I2C_Stop();
break;
}
}
}
}
return(EEP_flag);
}
//用數(shù)據(jù)FData填充從eep_addr開始的區(qū)域
bit fill(unsigned char FData,unsigned int eep_addr,
unsigned int n,unsigned char Control)
{unsigned char j=3;
while(1) //
{
I2C_Start();
I2C_wbyte(Control&0xfe);
I2C_RecAck();
if(EEP_flag)
break;
if(EEPALLBYTE>256) //有16位地址嗎?
{I2C_wbyte(eep_addr/256); //16位地址
I2C_RecAck();
if(EEP_flag)
break;
}
I2C_wbyte(eep_addr); //8位地址
I2C_RecAck();
if(EEP_flag)
break;
while(n--)
{
I2C_wbyte(FData);
I2C_RecAck();
if(EEP_flag)
break;
eep_addr++;
if((eep_addr%PAGESIZE==0)) //跨頁
{i2c_stop();
delay10ms();
break;
}
}
if(n==0)
{i2c_stop();
delay10ms();
break;
}
else if(EEP_flag)
{if(j--==0) //允許重試3次
break;
}
}
return(EEP_flag);
}
//先把原來的數(shù)據(jù)搬走(用copy),再插入新數(shù)據(jù)
bit insert(unsigned char *p_data,unsigned int eep_addr,
unsigned int n,unsigned char Control)
{
}
//先刪除,再把數(shù)據(jù)搬過來(用copy)
bit delete(unsigned int eep_addr,unsigned int n,unsigned char Control)
{fill(0xff,eep_addr,n,Control);
}
//復(fù)制
bit copy(unsigned int SSAddres,unsigned int DSAddres,unsigned int n)
{unsigned char buff[8]; //每次復(fù)制的數(shù)量由數(shù)組決定
unsigned char i=8;
while(n>0)
{if(n>=8)
{i=8;
n-=8;
}
else
{i=n;
n=0;
}
RW24CXX(buffer,SSAddres,i,Control|0x01);
RW24CXX(buffer,DSAddres,i,Control&0xfe);
if(EEP_flag)
break;
SSAddres+=8;
DSAddres+=8;
}
return(EEP_flag);
}