實驗16:EEPROM-IIC
主程序
/**************************************************************************************
* EEPROM-IIC實驗 *
實現現象:下載程序後數碼管後4位顯示0,按K1保存顯示的數據,按K2讀取上次保存的數據,
按K3顯示數據加一,按K4顯示數據清零。最大能寫入的數據是255.
注意事項:由於P3.2口跟紅外線共用,所以做按鍵實驗時爲了不讓紅外線影響實驗效果,最好把紅外線先取下來。
***************************************************************************************/
#include "reg52.h" //此文件中定義了單片機的一些特殊功能寄存器
#include "i2c.h"
typedef unsigned int u16; //對數據類型進行聲明定義
typedef unsigned char u8;
sbit LSA=P2^2;
sbit LSB=P2^3;
sbit LSC=P2^4;
sbit k1=P3^1;
sbit k2=P3^0;
sbit k3=P3^2;
sbit k4=P3^3; //定義按鍵端口
char num=0;
u8 disp[4];
u8 code smgduan[10]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f};
/*******************************************************************************
* 函 數 名 : delay
* 函數功能 : 延時函數,i=1時,大約延時10us
*******************************************************************************/
void delay(u16 i)
{
while(i--);
}
/*******************************************************************************
* 函數名 :Keypros()
* 函數功能 :按鍵處理函數
* 輸入 : 無
* 輸出 : 無
*******************************************************************************/
void Keypros()
{
if(k1==0)
{
delay(1000); //消抖處理
if(k1==0)
{
At24c02Write(1,num); //在地址1內寫入數據num
}
while(!k1);
}
if(k2==0)
{
delay(1000); //消抖處理
if(k2==0)
{
num=At24c02Read(1); //讀取EEPROM地址1內的數據保存在num中
}
while(!k2);
}
if(k3==0)
{
delay(100); //消抖處理
if(k3==0)
{
num++; //數據加1
if(num>255)num=0;
}
while(!k3);
}
if(k4==0)
{
delay(1000); //消抖處理
if(k4==0)
{
num=0; //數據清零
}
while(!k4);
}
}
/*******************************************************************************
* 函數名 :datapros()
* 函數功能 :數據處理函數
* 輸入 : 無
* 輸出 : 無
*******************************************************************************/
void datapros()
{
disp[0]=smgduan[num/1000];//千位
disp[1]=smgduan[num%1000/100];//百位
disp[2]=smgduan[num%1000%100/10];//個位
disp[3]=smgduan[num%1000%100%10];
}
/*******************************************************************************
* 函數名 :DigDisplay()
* 函數功能 :數碼管顯示函數
* 輸入 : 無
* 輸出 : 無
*******************************************************************************/
void DigDisplay()
{
u8 i;
for(i=0;i<4;i++)
{
switch(i) //位選,選擇點亮的數碼管,
{
case(0):
LSA=0;LSB=0;LSC=0; break;//顯示第0位
case(1):
LSA=1;LSB=0;LSC=0; break;//顯示第1位
case(2):
LSA=0;LSB=1;LSC=0; break;//顯示第2位
case(3):
LSA=1;LSB=1;LSC=0; break;//顯示第3位
}
P0=disp[3-i];//發送數據
delay(100); //間隔一段時間掃描
P0=0x00;//消隱
}
}
/*******************************************************************************
* 函 數 名 : main
* 函數功能 : 主函數
* 輸 入 : 無
* 輸 出 : 無
*******************************************************************************/
void main()
{
while(1)
{
Keypros(); //按鍵處理函數
datapros(); //數據處理函數
DigDisplay();//數碼管顯示函數
}
}
i2c程序
#include"i2c.h"
/*******************************************************************************
* 函數名 : Delay10us()
* 函數功能 : 延時10us
* 輸入 : 無
* 輸出 : 無
*******************************************************************************/
void Delay10us()
{
unsigned char a,b;
for(b=1;b>0;b--)
for(a=2;a>0;a--);
}
/*******************************************************************************
* 函數名 : I2cStart()
* 函數功能 : 起始信號:在SCL時鐘信號在高電平期間SDA信號產生一個下降沿
* 輸入 : 無
* 輸出 : 無
* 備註 : 起始之後SDA和SCL都爲0
*******************************************************************************/
void I2cStart()
{
SDA=1;
Delay10us();
SCL=1;
Delay10us();//建立時間是SDA保持時間>4.7us
SDA=0;
Delay10us();//保持時間是>4us
SCL=0;
Delay10us();
}
/*******************************************************************************
* 函數名 : I2cStop()
* 函數功能 : 終止信號:在SCL時鐘信號高電平期間SDA信號產生一個上升沿
* 輸入 : 無
* 輸出 : 無
* 備註 : 結束之後保持SDA和SCL都爲1;表示總線空閒
*******************************************************************************/
void I2cStop()
{
SDA=0;
Delay10us();
SCL=1;
Delay10us();//建立時間大於4.7us
SDA=1;
Delay10us();
}
/*******************************************************************************
* 函數名 : I2cSendByte(unsigned char dat)
* 函數功能 : 通過I2C發送一個字節。在SCL時鐘信號高電平期間,保持發送信號SDA保持穩定
* 輸入 : num
* 輸出 : 0或1。發送成功返回1,發送失敗返回0
* 備註 : 發送完一個字節SCL=0,SDA=1
*******************************************************************************/
unsigned char I2cSendByte(unsigned char dat)
{
unsigned char a=0,b=0;//最大255,一個機器週期爲1us,最大延時255us。
for(a=0;a<8;a++)//要發送8位,從最高位開始
{
SDA=dat>>7; //起始信號之後SCL=0,所以可以直接改變SDA信號
dat=dat<<1;
Delay10us();
SCL=1;
Delay10us();//建立時間>4.7us
SCL=0;
Delay10us();//時間大於4us
}
SDA=1;
Delay10us();
SCL=1;
while(SDA)//等待應答,也就是等待從設備把SDA拉低
{
b++;
if(b>200) //如果超過2000us沒有應答發送失敗,或者爲非應答,表示接收結束
{
SCL=0;
Delay10us();
return 0;
}
}
SCL=0;
Delay10us();
return 1;
}
/*******************************************************************************
* 函數名 : I2cReadByte()
* 函數功能 : 使用I2c讀取一個字節
* 輸入 : 無
* 輸出 : dat
* 備註 : 接收完一個字節SCL=0,SDA=1.
*******************************************************************************/
unsigned char I2cReadByte()
{
unsigned char a=0,dat=0;
SDA=1; //起始和發送一個字節之後SCL都是0
Delay10us();
for(a=0;a<8;a++)//接收8個字節
{
SCL=1;
Delay10us();
dat<<=1;
dat|=SDA;
Delay10us();
SCL=0;
Delay10us();
}
return dat;
}
/*******************************************************************************
* 函數名 : void At24c02Write(unsigned char addr,unsigned char dat)
* 函數功能 : 往24c02的一個地址寫入一個數據
* 輸入 : 無
* 輸出 : 無
*******************************************************************************/
void At24c02Write(unsigned char addr,unsigned char dat)
{
I2cStart();
I2cSendByte(0xa0);//發送寫器件地址
I2cSendByte(addr);//發送要寫入內存地址
I2cSendByte(dat); //發送數據
I2cStop();
}
/*******************************************************************************
* 函數名 : unsigned char At24c02Read(unsigned char addr)
* 函數功能 : 讀取24c02的一個地址的一個數據
* 輸入 : 無
* 輸出 : 無
*******************************************************************************/
unsigned char At24c02Read(unsigned char addr)
{
unsigned char num;
I2cStart();
I2cSendByte(0xa0); //發送寫器件地址
I2cSendByte(addr); //發送要讀取的地址
I2cStart();
I2cSendByte(0xa1); //發送讀器件地址
num=I2cReadByte(); //讀取數據
I2cStop();
return num;
}