I2C 只是用兩條雙向的線,一條 Serial Data Line (SDA) ,另一條Serial Clock (SCL)。
SCL:上升沿將數據輸入到每個EEPROM器件中;下降沿驅動EEPROM器件輸出數據。(邊沿觸發)
SDA:雙向數據線,為OD門,與其它任意數量的OD與OC門成“線與”關系。
為了加深對I2C總線的理解,用C語言模擬IIC總線,邊看源代碼邊讀波形:
如下圖所示的寫操作的時序圖:
讀時序的理解同理。對于時序不理解的朋友請參考“I2C總線概述及時序總結”
完整的程序如下:
#include
#define uchar unsigned char
#define uint unsigned int
#define read_ADD 0xa1
uchar a;
sbit SDA=P2^0;
sbit SCL=P2^1;
void SomeNop(); //短延時
void init(); //初始化
void check_ACK(void);
void I2CStart(void);
void I2cStop(void);
void write_byte(uchar dat);//寫字節
void delay(uint z);
uchar read_byte(); //讀字節
void write(uchar addr,uchar dat); //指定地址寫
uchar read(uchar addr); //指定地址讀
bit flag; //應答標志位
void main()
{
init();
write_add(5,0xaa); //向地址5寫入0xaa
delay(10); //延時,否則被坑呀!!!
P1=read_add(5); //讀取地址5的值
while(1);
}
//***************************************************************************
void delay()//簡單延時函數
{ ;; }
//***************************************************************************
void start() //開始信號 SCL在高電平期間,SDA一個下降沿則表示啟動信號
{
sda=1; //釋放SDA總線
delay();
scl=1;
delay();
sda=0;
delay();
}
//***************************************************************************
void stop() //停止 SCL在高電平期間,SDA一個上升沿則表示停止信號
{
sda=0;
delay();
scl=1;
delay();
sda=1;
delay();
}
//***************************************************************************
void respons() //應答 SCL在高電平期間,SDA被從設備拉為低電平表示應答
{
uchar i;
scl=1;
delay();
while((sda==1)&&(i《250))i++;
scl=0;
delay();
}
//***************************************************************************
void init()//總線初始化 將總線都拉高一釋放總線 發送啟動信號前,要先初始化總線。即總有檢測到總線空閑才開始發送啟動信號
{
sda=1;
delay();
scl=1;
delay();
}
//***************************************************************************
void write_byte(uchar date) //寫一個字節
{
uchar i,temp;
temp=date;
for(i=0;i《8;i++)
{
temp=temp《《1;
scl=0;//拉低SCL,因為只有在時鐘信號為低電平期間按數據線上的高低電平狀態才允許變化;并在此時和上一個循環的scl=1一起形成一個上升沿
delay();
sda=CY;
delay();
scl=1;//拉高SCL,此時SDA上的數據穩定
delay();
}
scl=0;//拉低SCL,為下次數據傳輸做好準備
delay();
sda=1;//釋放SDA總線,接下來由從設備控制,比如從設備接收完數據后,在SCL為高時,拉低SDA作為應答信號
delay();
}
//***************************************************************************
uchar read_byte()//讀一個字節
{
uchar i,k;
scl=0;
delay();
sda=1;
delay();
for(i=0;i《8;i++)
{
scl=1;//上升沿時,IIC設備將數據放在sda線上,并在高電平期間數據已經穩定,可以接收啦
delay();
k=(k《《1)|sda;
scl=0;//拉低SCL,使發送端可以把數據放在SDA上
delay();
}
return k;
}
//***************************************************************************
void write_add(uchar address,uchar date)//任意地址寫一個字節
{
start();//啟動
write_byte(0xa0);//發送從設備地址
respons();//等待從設備的響應
write_byte(address);//發出芯片內地址
respons();//等待從設備的響應
write_byte(date);//發送數據
respons();//等待從設備的響應
stop();//停止
}
//***************************************************************************
uchar read_add(uchar address)//讀取一個字節
{
uchar date;
start();//啟動
write_byte(0xa0);//發送發送從設備地址 寫操作
respons();//等待從設備的響應
write_byte(address);//發送芯片內地址
respons();//等待從設備的響應
start();//啟動
write_byte(0xa1);//發送發送從設備地址 讀操作
respons();//等待從設備的響應
date=read_byte();//獲取數據
stop();//停止
return date;//返回數據
}
?
評論
查看更多