目标:EEPROM-IIC总线理解(三)
代码编写(一):
//延时函数,延时10us
void I2CDelay() //误差 0us
{
unsigned char a,b;
for(b=1;b>0;b--)
for(a=2;a>0;a--);
}
//起始信号:SCL为高电平时SDA有个从高到底的跳变动作,之后才能进行数据传输
void I2CStart() //产生总线起始信号
{
I2C_SDA = 1; //首先确保SDA、SCL都是高电平
I2C_SCL = 1;
I2CDelay();
I2C_SDA = 0; //先拉低SDA
I2CDelay();
I2C_SCL = 0; //再拉低SCL
}
//终止信号:SCL为高电平时SDA有个从低到高的跳变动作,之后停止数据传输
void I2CStop() //产生总线停止信号
{
I2C_SCL = 0; //首先确保SDA、SCL都是低电平
I2C_SDA = 0;
I2CDelay();
I2C_SCL = 1; //先拉高SCL
I2CDelay();
I2C_SDA = 1; //再拉高SDA
I2CDelay();
}
//注意:在SCL是高电平时不要乱改变SDA的状态防止是停止信号使运行失败
//数据发送(写):只有在SCL是高电平时SDA上的数据是有效的。编写单片机向总线发送数据时,可以在SCL是低电平时把数据传输到SDA中(SCL在低电平时SDA才能改变数据),
//等一个字节传输完后,拉高SCL电平延时一段时间后再拉低SCL电平(在延时时间内即SCL是高电平这段时间内SDA不能发生变化)来使SDA中的而数据发送到从机
unsigned char I2cSendByte(unsigned char dat)
{
unsigned char a=0;b=0; //最大255,一个机器周期为1us,最大延时255us。
for(a=0;a<8;a++){
SDA = dat>>7; //起始信号后SCL=0,所以可以直接改变SDA,因为数据是先按高字节发送的所以向右移7位把最到位高位放到最低位的地方,此时数据虽然给力SDA但并每没有发送到从机
dat = dat<<1; //把次高位变为最高位
I2CDelay(); //上面几行是数据变化的阶段
SCL=1; //发送数据时SCL保持高电平,此时数据稳定
I2CDelay();
SCL=0; //再拉低SCL,完成一个位周期,scl一个位周期然后数据发送出去了,准确的说是在SCL是高电平的这段时间里发送出去了
I2CDelay();//此上到SCL=1;是数据发送的条件
}
SDA=1;//8位数据发送完后,主机释放SDA,以检测从机应答
I2CDelay();
SCL=1;//拉高SCL,来读取SDA的状态
while(SDA) //等待应答,也就是等待从设备把SDA拉低
{
b++;
if(b>200){ //如果超过2000us没有应答发送失败,或者为非应答,表示接收结束
SCL=0;
I2CDelay();
return 0; //未响应
}
}
SCL=0; //时钟线拉低
I2CDelay();
return 1; //响应了,说明b++还没累加到200
}
FC总线每接收完一个字节(8个位)后,在第9个时钟信号时,会在SDA上回应一个低电平的ACK应答信号,以此表明当前受控的器件已接收完一个字节,可以开始下一个字节的传送了。
编程时可以在传送完一个字节后,把连接SDA的I/O口线设置回读数据状态,如使用51系列的单片机时就要把I/O口置高电平,然后在SCL.脚设置一个脉冲,在SCL为高电平时读取SDA,如不为低电平就说明器件状态不空闲或出错。需要注意,因为SDA是双向的I/O,无论是接收还是器件接收,每个字节完成后,接收方都可以发送一个ACK回应给发送方。
//数据读取:
unsigned char I2cReadByte()
{
unsigned char a=0,dat=0; //dat用于接受单片机接受到的数据
SDA=1; //起始和发送一个字节之后SCL都是0
I2CDelay();
for(a=0;a<8;a++){
SCL=1; //只有在SCL电平是高电平时SDA数据才有效,所以读的时候先把SCL拉高再去读SDA中的数据
I2CDelay();
dat<<=1;
dat|=SDA; //dat来接受SDA中的数据
I2CDelay();
SCL=0; //拉低SCL
I2CDelay();
}
return dat; //返回接收到的数据
}