/**************************************************
EEPROM讀寫操作
晶振:內(nèi)部 8M
***************************************************/
#include
#include
#define uchar unsigned char
#define uint unsigned int
#define ulong unsigned long
uchar syserr;
void delay_EEPROM(uchar n) //延時
{
unsigned int a,b;
for(a=0;a
;
}
//------在此設(shè)定芯片型號------
#define e2prom 256// <---在此設(shè)定芯片型號, 1代表24C01; 16代表24C16; 512代表24C512
#if e2prom==1
#define PAGE_SIZE 8
#define SIZE 0x007f
#elif e2prom==2
#define PAGE_SIZE 8
#define SIZE 0x00ff
#elif e2prom==4
#define PAGE_SIZE 16
#define SIZE 0x01ff
#elif e2prom==8
#define PAGE_SIZE 16
#define SIZE 0x03ff
#elif e2prom==16
#define PAGE_SIZE 16
#define SIZE 0x07ff
#elif e2prom==32
#define PAGE_SIZE 32
#define SIZE 0x0fff
#elif e2prom==64
#define PAGE_SIZE 32
#define SIZE 0x1fff
#elif e2prom==128
#define PAGE_SIZE 64
#define SIZE 0x3fff
#elif e2prom==256
#define PAGE_SIZE 64
#define SIZE 0x7fff
#elif e2prom==512
#define PAGE_SIZE 128
#define SIZE 0xffff
#endif
//--------------------------
//--------在此設(shè)定芯片地址-------
#define W_ADD_COM 0xa0//寫字節(jié)命令及器件地址(根據(jù)地址實際情況改變), 1010 A2 A1 A0 0
#define R_ADD_COM 0xa1//讀命令字節(jié)及器件地址(根據(jù)地址實際情況改變), 1010 A2 A1 A0 1
//-------------------------------
#define SLAW0x18//SLA_W 已正常發(fā)送代碼,判斷器件是否正常應(yīng)答的常量.
//-----在此改變預(yù)置錯誤號-----
#define ERR_SLAW10//寫字節(jié)命令及器件地址錯, 在此也就是讀寫器件錯!!
//---------------------------
//-----------4個I2總線公用函數(shù), 可供其它I2總線器件的程序調(diào)用--------------
void i2cstart(void);//總線上起動開始條件
uchar i2cwt(uchar a);//把一個字節(jié)數(shù)據(jù)輸入器件, 返回TWI狀態(tài)
uchar i2crd(void);//i2c讀要調(diào)用的函數(shù)
void i2cstop(void);//總線上起動停止條件
//------------------------------------------------------------------------
uchar * wt24c_fc(uchar *p, uint ad, uchar n);//向24Cxx寫入數(shù)據(jù)wt24c_h()所要調(diào)用的函數(shù)
//向24Cxx寫入數(shù)據(jù)
//參數(shù): *p_rsc要輸出數(shù)據(jù)的主機內(nèi)存地址指針; ad_dst要寫入數(shù)據(jù)的i2c的地址(雙字節(jié)); num數(shù)據(jù)個數(shù)
//參數(shù)條件: ad_dst: ad_dst+(num-1)不能大于器件的最高地址; num必須>0;
void wt24c(uchar *p_rsc, uint ad_dst, uint num)
{ uint n;
n=ad_dst/PAGE_SIZE;//確定地址與塊地址的差
if(n)
n=(ulong)PAGE_SIZE*(n+1)-ad_dst;
else
n=PAGE_SIZE-ad_dst;
if(n>=num)//如果ad_dst所在的數(shù)據(jù)塊的末尾地址 >= ad_dst + num, 就直接寫入num個數(shù)據(jù)
{
wt24c_fc(p_rsc, ad_dst, num);
if(syserr!=0)
return;
}
else//如果ad_dst所在的數(shù)據(jù)塊末尾地址 < ad_dst + num, 就先寫入ad_dst所在的數(shù)據(jù)塊末尾地址與 ad_dst 之差個數(shù)據(jù)
{
p_rsc=wt24c_fc(p_rsc, ad_dst, n);
if(syserr!=0)
return;
num-=n; //更新剩下數(shù)據(jù)個數(shù)
ad_dst+=n;//更新剩下數(shù)據(jù)的起始地址
//把剩下數(shù)據(jù)寫入器件
while(num>=PAGE_SIZE)//先按PAGE_SIZE為長度一頁一頁的寫入
{p_rsc=wt24c_fc(p_rsc, ad_dst, PAGE_SIZE);
if(syserr!=0) return;
num-=PAGE_SIZE;//更新剩余數(shù)據(jù)個數(shù)
ad_dst+=PAGE_SIZE;//更新剩下數(shù)據(jù)的起始地址
}
if(num)//把最后剩下的小于一個PAGE_SIZE長度的數(shù)據(jù)寫入器件
wt24c_fc(p_rsc, ad_dst, num);
}
}
//從24cxx讀出數(shù)據(jù)
//參數(shù): *p_dst要讀入數(shù)據(jù)的主機內(nèi)存地址指針; ad_rsc要輸出數(shù)據(jù)的i2c的地址(整形); num數(shù)據(jù)個數(shù)(整形)
//參數(shù)條件: ad_dst+(num-1)不能大于器件的最高地址; num必須>0;
void port_init(void)
{
PORTC = 0xFF;
DDRC &= 0xfc;
}
//TWI initialize
// bit rate:100
void twi_init(void)
{
TWCR= 0x00; //disable twi
TWBR= 0x20; //set bit rate
TWSR= 0x00; //set prescale
TWAR= 0x00; //set slave address
TWCR= 0x04; //enable twi
}
void rd24c(uchar *p_dst, uint ad_rsc, uint num)
{
uchar t=0;
#if e2prom<32
t=ad_rsc>>8;
t<<=1;
#endif
i2cstart();//發(fā)送起始信號
if(i2cwt(W_ADD_COM+t)==SLAW)//發(fā)送SLA_W, 寫字節(jié)命令及器件地址
{
#if e2prom>=32
i2cwt(ad_rsc>>8);//ad_rsc的高位, 發(fā)送要讀出數(shù)據(jù)的地址
#endif
i2cwt(ad_rsc);//ad_rsc的低位
i2cstart();//再發(fā)送起始信號
i2cwt(R_ADD_COM+t);//發(fā)送SLA_R, 讀命令字節(jié)及器件地址
for(;num>0;num--)
{
*p_dst=i2crd();//從器件讀出一個字節(jié)
p_dst++;
}
}
else
syserr=ERR_SLAW;//寫字節(jié)命令及器件地址錯或?qū)Ψ綗o應(yīng)答
i2cstop();
}
//向24Cxx寫入數(shù)據(jù)wt24c_h()所要調(diào)用的函數(shù)
//返回寫入n個字節(jié)后的主機內(nèi)存指針
uchar * wt24c_fc(uchar *p, uint ad, uchar n)
{
uchar t=0;
#if e2prom<32
t=ad>>8;
t<<=1;
#endif
i2cstart();//發(fā)送起始信號
if(i2cwt(W_ADD_COM+t)==SLAW)//發(fā)送SLA_W, 寫字節(jié)命令及器件地址
{
#if e2prom>=32
i2cwt(ad>>8);//ad_dst的高位到器件
#endif
i2cwt(ad);//ad_dst的低位到器件
for(;n>0;n--)//發(fā)送要寫入的數(shù)據(jù)
{ i2cwt(*p);
p++;
}
}
else syserr=ERR_SLAW;//寫字節(jié)命令及器件地址錯
i2cstop();
delay_EEPROM(10);//延時6ms
return(p);
}
//-------------------------------以下為其它I2總線器件可調(diào)用的函數(shù)--------------------------
//總線上起動開始條件
void i2cstart(void)
{
TWCR= BIT(TWINT) | BIT(TWSTA) | BIT(TWEN);
while (!(TWCR & BIT(TWINT)));
}
//把一個字節(jié)數(shù)據(jù)輸入器件, 返回TWI狀態(tài)
uchar i2cwt(uchar a)
{
TWDR = a;
TWCR = BIT(TWINT) | BIT(TWEN);
while (!(TWCR & BIT(TWINT)));
_NOP();
return(TWSR&0b11111000);
}
//i2c讀要調(diào)用的函數(shù)
//從器件讀出一個字節(jié)
uchar i2crd(void)
{
TWCR= BIT(TWINT) | BIT(TWEA) | BIT(TWEN);
while (!(TWCR & BIT(TWINT)));
return(TWDR);
}
//總線上起動停止條件
void i2cstop(void)
{
TWCR = BIT(TWINT) | BIT(TWSTO) | BIT(TWEN);
}