51單片機(jī) 普通IO口模擬IIC(I2C)接口通訊的程序代碼
I2C總線是Philips公司提出的一種集成電路IC器件之間相連接的總線協(xié)議,其目的是使電子系統(tǒng)(不只 限于單片機(jī)系統(tǒng))各個(gè)IC器件之間的連線變得容易。因?yàn)槭褂脗鹘y(tǒng)的并行總線在IC器件之間連接,往往會(huì)使得IC之間連線較多,顯得非常復(fù)雜。而I2C總線 則使IC器件之間只需SDA、SCL兩條連線就可以傳送數(shù)據(jù),因而十分方便。由于I2C在印刷體中不容易書寫(需要上標(biāo)),所以實(shí)際書寫時(shí),還常見(jiàn)到 IIC、I2C等書寫方法,本文采用IIC的寫法,敬請(qǐng)注意。關(guān)于IIC總線的知識(shí),請(qǐng)參閱相關(guān)書籍,此處不再做進(jìn)一步介紹。
本文引用地址:http://butianyuan.cn/article/201808/385451.htm下面我們用一個(gè)使用IIC總線連接器件的例子來(lái)簡(jiǎn)單說(shuō)明IIC總線的仿真。
例.EEPROM24C02是采用IIC接口的一種常用2Kbit(256×8bit)的存儲(chǔ)器。編寫程序使用AT89C51的IO口模擬實(shí)現(xiàn)IIC總線協(xié)議進(jìn)行通信,并向24C02存儲(chǔ)器內(nèi)從字節(jié)0到字節(jié)FF寫入數(shù)字0到FF。
51系列單片機(jī)本身沒(méi)有IIC接口,但一些本身具有IIC接口的單片機(jī)往往是高端產(chǎn)品,一方面價(jià)格不菲,另一方面我們的系統(tǒng)也沒(méi)有必要使用之。通常我們就使用軟件通過(guò)51系列單片機(jī)的IO口來(lái)模擬實(shí)現(xiàn)IIC總線通信。
本例事實(shí)上比較簡(jiǎn)單,但需要對(duì)IIC總線時(shí)序有較好的理解。源文件如下圖所示(采用C51語(yǔ)言編寫):
在Keil中編輯好源文件以后,接下來(lái)就可以建立工程文件并生成相應(yīng)的源代碼了,然后我們來(lái)繪制電路圖。
此例的電路圖極其簡(jiǎn)單。只需兩個(gè)IC,即AT89C51和24C02C,和兩個(gè)上拉電阻,而且上拉電阻還可以省略。至于連接,就更為簡(jiǎn)單了。最后得到繪制好的電路圖如下圖所示:
繪制好電路圖,我們就可以將前面剛剛生成的程序源代碼裝入單片機(jī)了,裝入以后,下面我們就可以來(lái)進(jìn)行仿真了。
首先點(diǎn)擊仿真按鈕,系統(tǒng)沒(méi)有什么反映,只有高低電平變化的顏色。我們要想查看結(jié)果,還要用前文中仿真擴(kuò)展 RAM存儲(chǔ)器的方法,先點(diǎn)擊暫停,然后點(diǎn)擊“Debug”菜單下的“I2C Memory Internal Memory – U2”子菜單來(lái)打開(kāi)U2即EEPROM存儲(chǔ)器24C02C的內(nèi)容窗口“I2C Memory Internal Memory – U2”,然后我們就看到了其中的內(nèi)容,也就是我們仿真程序的結(jié)果。如下圖所示:
從圖中我們能清楚地看到我們的仿真結(jié)果,程序完全正確地執(zhí)行了我們的命令。
當(dāng)然,如果你過(guò)早地點(diǎn)擊了暫停按鈕,那么你得到的結(jié)果可能和上圖略有不同,那可能是因?yàn)槌绦蛏形磮?zhí)行完畢。此時(shí)你可以繼續(xù)點(diǎn)擊運(yùn)行按鈕,或者點(diǎn)擊單步按鈕來(lái)仔細(xì)查看程序執(zhí)行過(guò)程中24C02C存儲(chǔ)器內(nèi)容的改變情況。
完整代碼如下:
/*----------------------------------------------------------------
Acess the eeprom--24c04
----------------------------------------------------------------*/
#include
#ifndef INT8U
#define INT8U unsigned char
#endif
#ifndef INT8S
#define INT8S signed char
#endif
#ifndef INT16U
#define INT16U unsigned int
#endif
#define I2C_DELAY; _nop_();_nop_();_nop_();_nop_();_nop_(); // >=4.7uS
//----------------------------------------------------------------
// delay 100us
//----------------------------------------------------------------
void mDelay(INT8U k)
{
INT16U i ;
for(; k>0; k--)
{
for(i=0; i93; i++)
;
}
}
//----------------------------------------------------------------
//OK
//----------------------------------------------------------------
void I2C_Start(void)
{
SDA = 1;
I2C_DELAY;
SCL = 1;
I2C_DELAY;
SDA = 0;
I2C_DELAY;
I2C_DELAY;
}
//----------------------------------------------------------------
//OK
//----------------------------------------------------------------
void I2C_Stop(void)
{
SDA = 0 ;
I2C_DELAY;
SCL = 1 ;
I2C_DELAY;
SDA = 1 ;
I2C_DELAY;
I2C_DELAY;
}
//----------------------------------------------------------------
//
//----------------------------------------------------------------
void sendAck(void)
{
SCL = 0;
I2C_DELAY;
SDA = 0;
I2C_DELAY;
SCL = 1;
I2C_DELAY;
}
//----------------------------------------------------------------
//
//----------------------------------------------------------------
void sendNoAck(void)
{
SCL = 0;
I2C_DELAY;
SDA = 1;
I2C_DELAY;
SCL = 1;
I2C_DELAY;
}
//----------------------------------------------------------------
// 0 = noACK; 1 = ACK ;
//----------------------------------------------------------------
bit checkAck()
{
bit tempbit;
/*發(fā)送完一個(gè)字節(jié)后檢驗(yàn)設(shè)備的應(yīng)答信號(hào)*/
SDA = 1;
I2C_DELAY;
SCL = 0;
I2C_DELAY;
tempbit = SDA;
SCL = 1;
I2C_DELAY;
if(tempbit==1)
{
return 0; //noACK
}
else
{
return 1; //ACK
}
}
//----------------------------------------------------------------
//OK
// a positive clock edge clock a bit into the ROM
//----------------------------------------------------------------
void writeByte(INT8U datum)
{
INT8U bitCnt = 0 ;
for(bitCnt=0; bitCnt8; bitCnt++)
{
SCL = 0 ;
I2C_DELAY;
if ((datum0x80) == 0x80) //if the MSb is 1
SDA = 1 ;
else
SDA = 0 ;
I2C_DELAY;
SCL = 1 ;
I2C_DELAY;
datum=1 ;
}
}
//----------------------------------------------------------------
//OK
//----------------------------------------------------------------
INT8U readByte(void)
{
bit tempbit = 1 ;
INT8U temp = 0 ;
INT8U bitCnt ;
SDA = 1 ; // release the bus,ready to receive byte??????????????
I2C_DELAY;
for(bitCnt=0; bitCnt8; bitCnt++)
{
SCL = 0; //?????????????????????????huan???????????????
I2C_DELAY;
tempbit = SDA ;
if (tempbit)
temp |= 0x01 ;
else
temp = 0xfe ;
SCL = 1 ;
I2C_DELAY;
if(bitCnt7)
temp = 1 ;
}
return(temp) ;
}
/*~~~~~~~~~~~~~~~~~~~~~~~ API ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~*/
/*-----------------------------------------------------------------
write some bytes to sequential address
-----------------------------------------------------------------*/
void writeToROM(INT8U datum[], INT8U address, INT8U num)
{
bit tempbit ;
INT8U i ;
INT8U *datum_P ;
datum_P = datum ;
I2C_Start() ;
writeByte(0xa0) ;
tempbit = checkAck();
writeByte(address) ;
tempbit = checkAck();
for(i=0; i
{
writeByte(*(datum_P+i)) ;
if(!checkAck())
{
I2C_Stop() ;
mDelay(100) ;
}
}
I2C_Stop() ;
}
/*-----------------------------------------------------------------
read some bytes from ROM`s sequential address
-----------------------------------------------------------------*/
void readFromROM(INT8U datum[], INT8U address, INT8U num)
{
bit tempbit ;
INT8U i ;
INT8U *datum_P ;
datum_P = datum;
I2C_Start() ;
writeByte(0xa0) ;
tempbit = checkAck();
writeByte(address) ;
tempbit = checkAck();
I2C_Start() ;
writeByte(0xa1) ;
tempbit = checkAck();
for(i=0; i
{
*(datum_P+i) = readByte() ;
if(i!=num-1)
{
sendAck() ;
}
else
{
sendNoAck() ;
}
}
I2C_Stop() ;
}
/*-----------------------------------------------------------------
wirte one byte to ROM --random write
-----------------------------------------------------------------*/
void writeOneByte(INT8U addr, INT8U datum)
{
bit tempbit ;
/*write a byte to mem*/
I2C_Start();
writeByte(0xa0);
tempbit = checkAck();
writeByte(addr); /*address*/
tempbit = checkAck();
writeByte(datum); /*the data*/
tempbit = checkAck();
I2C_Stop();
mDelay(100) ;
}
/*-----------------------------------------------------------------
read one byte from rom --random read
-----------------------------------------------------------------*/
INT8U readOneByte(INT8U addr)
{
bit tempbit = 1;
INT8U mydata;
/*read a byte from mem*/
I2C_Start();
writeByte(0xa0);
tempbit = checkAck();
writeByte(addr); /*address*/
tempbit = checkAck();
I2C_Start();
writeByte(0xa1);
tempbit = checkAck();
mydata = readByte();
tempbit = checkAck();
return (mydata) ;
I2C_Stop();
}
『本文轉(zhuǎn)載自網(wǎng)絡(luò),版權(quán)歸原作者所有,如有侵權(quán)請(qǐng)聯(lián)系刪除』
評(píng)論