当前位置:首页 > 单片机 > 单片机
[导读]I2C总线是Philips公司提出的一种集成电路IC器件之间相连接的总线协议,其目的是使电子系统(不只 限于单片机系统)各个IC器件之间的连线变得容易。因为使用传统的并行总线在IC器件之间连接,往往会使得IC之间连线较多,显得非常复杂。而I2C总线 则使IC器件之间只需SDA、SCL两条连线就可以传送数据,因而十分方便。由于I2C在印刷体中不容易书写(需要上标),所以实际书写时,还常见到 IIC、I2C等书写方法,本文采用IIC的写法,敬请注意。关于IIC总线的知识,请参阅相关书籍,此处不再做进一步介绍。

 I2C总线是Philips公司提出的一种集成电路IC器件之间相连接的总线协议,其目的是使电子系统(不只 限于单片机系统)各个IC器件之间的连线变得容易。因为使用传统的并行总线在IC器件之间连接,往往会使得IC之间连线较多,显得非常复杂。而I2C总线 则使IC器件之间只需SDA、SCL两条连线就可以传送数据,因而十分方便。由于I2C在印刷体中不容易书写(需要上标),所以实际书写时,还常见到 IIC、I2C等书写方法,本文采用IIC的写法,敬请注意。关于IIC总线的知识,请参阅相关书籍,此处不再做进一步介绍。

下面我们用一个使用IIC总线连接器件的例子来简单说明IIC总线的仿真。

例.EEPROM24C02是采用IIC接口的一种常用2Kbit(256×8bit)的存储器。编写程序使用AT89C51的IO口模拟实现IIC总线协议进行通信,并向24C02存储器内从字节0到字节FF写入数字0到FF。

51系列单片机本身没有IIC接口,但一些本身具有IIC接口的单片机往往是高端产品,一方面价格不菲,另一方面我们的系统也没有必要使用之。通常我们就使用软件通过51系列单片机的IO口来模拟实现IIC总线通信。

本例事实上比较简单,但需要对IIC总线时序有较好的理解。源文件如下图所示(采用C51语言编写):

 


 

 


 

 


 

在Keil中编辑好源文件以后,接下来就可以建立工程文件并生成相应的源代码了,然后我们来绘制电路图。

此例的电路图极其简单。只需两个IC,即AT89C51和24C02C,和两个上拉电阻,而且上拉电阻还可以省略。至于连接,就更为简单了。最后得到绘制好的电路图如下图所示:

 


 

绘制好电路图,我们就可以将前面刚刚生成的程序源代码装入单片机了,装入以后,下面我们就可以来进行仿真了。

首先点击仿真按钮,系统没有什么反映,只有高低电平变化的颜色。我们要想查看结果,还要用前文中仿真扩展 RAM存储器的方法,先点击暂停,然后点击“Debug”菜单下的“I2C Memory Internal Memory – U2”子菜单来打开U2即EEPROM存储器24C02C的内容窗口“I2C Memory Internal Memory – U2”,然后我们就看到了其中的内容,也就是我们仿真程序的结果。如下图所示:

 


 

从图中我们能清楚地看到我们的仿真结果,程序完全正确地执行了我们的命令。

当然,如果你过早地点击了暂停按钮,那么你得到的结果可能和上图略有不同,那可能是因为程序尚未执行完毕。此时你可以继续点击运行按钮,或者点击单步按钮来仔细查看程序执行过程中24C02C存储器内容的改变情况。

完整代码如下:

/*----------------------------------------------------------------

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; i<93; 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;

/*发送完一个字节后检验设备的应答信号*/

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; bitCnt<8; bitCnt++)

{

SCL = 0 ;

I2C_DELAY;

if ((datum&0x80) == 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; bitCnt<8; bitCnt++)

{

SCL = 0; //?????????????????????????huan???????????????

I2C_DELAY;

tempbit = SDA ;

if (tempbit)

temp |= 0x01 ;

else

temp &= 0xfe ;

SCL = 1 ;

I2C_DELAY;

if(bitCnt<7)

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();

}

『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

STM32与51单片机之间有什么差异呢?两者可以说是一场科技与性能的较量了。在科技飞速发展的今天,微控制器(MCU)已广泛应用于各类电子设备和系统中,发挥着举足轻重的作用。其中,STM32和51单片机作为两种常见的微控制...

关键字: STM32 51单片机 MCU

51单片机将是下述内容的主要介绍对象,通过这篇文章,小编希望大家可以对51单片机的相关情况以及信息有所认识和了解,详细内容如下。

关键字: 单片机 51单片机

在嵌入式系统开发中,单片机是不可或缺的重要组成部分。其中,STM32单片机和51单片机是两种常见的单片机芯片。本文将对比分析这两种单片机的区别,并探讨STM32单片机的优势。

关键字: stm32单片 51单片机

51单片机是指由美国INTEL公司生产的一系列单片机的总称,这一系列单片机包括了许多品种,如8031,8051,8751,8032,8052,8752等,其中8051是最早最典型的产品,该系列其它单片机都是在8051的基...

关键字: 51单片机 串行通信

51单片机是一种常见的微控制器,它具有串行通信接口(Serial Communication Interface,SCI)。通过串口通信接口,51单片机可以与其他设备或系统进行串行通信,实现数据传输和控制。

关键字: 51单片机 串口通信

Pic单片机和51单片机是两种应用广泛的微控制器,它们各自具有不同的特点和优势,选择哪种单片机取决于具体应用需求。下面将对Pic单片机和51单片机进行详细的介绍和对比。

关键字: PIC单片机 51单片机 单片机

随着电子信息技术的快速发展,串行通信在各种应用中得到了广泛的应用。其中,51单片机作为一种常见的嵌入式系统,具有低功耗、高性能、易于编程等特点,常用于各种自动化控制、数据采集等系统中。而PC机具有强大的数据处理能力和丰富...

关键字: 51单片机 PC机 串行通信

51 单片机内部有一个全双工串行接口。什么叫全双工串口呢?一般来说,只能接受或只能发送的称为单工串行;既可接收又可发送,但不能同时进行的称为半双工;能同时接收和发送的串行口称为全双工串行口。

关键字: 51单片机 串口通信 半双工

步进电机是一种无刷电机,可将电脉冲转换为机械旋转。顾名思义,它根据输入脉冲逐步旋转,是现代数字程序控制系统中的主要执行元件。

关键字: 51单片机 步进电机 无刷电机

1.ROM:只读存储器,单片机的只读存储区大多用于存储程序固又称程序存储器。

关键字: 51单片机 工作原理 电路
关闭
关闭