一.简介:
二.IIC的特征:
1.IIC物理层![](https://i-blog.csdnimg.cn/blog_migrate/fa248b46b25dc9814eb1ad5c00b28d96.png)
(i)只有两条总线线路: (一条双向串行数据线SDA, 一条双向串行时钟线SCL)------>意味着它只可以将数据一位一位的发送出去,同时也不可以进行两边同时通。所以采用的是半双工通信。总线可以支持多个IIC设备。
(ii)半双工通信:同一时间只可以进行单向通信。
(iii)总线通过上拉电阻,使得IIC设备在空闲状态下为高阻态模式(目的:不干扰别的正在正常进行通信的设备),当所以设备都处于高阻态的时候,上拉电阻会把总线也拉成高电平模式。
(iv)每个IIC设备都有一个自己独一无二的地址,主机和不同设备的识别靠的就是这个机制,当主机通过SDA总线发送设备地址后,相同地址的设备会进行回应,从而进行相互识别,进而实现通信。
(v)多个设备同时需要进行通信的时候,总线就会进行仲裁,一般是通过低电平的方式
详细请看:I2C总线仲裁机制_艾特号的博客-CSDN博客_i2c仲裁机制
(Vi) 串行的8 位双向数据传输位速率在标准模式下可达100kbit/s, 快速模式下可达400kbit/s, 高速模式下可达3.4Mbit/s;
2.IIC协议层
IIC的读写过程
(i)主机写数据到从机
S表示的是起始信号,这个时候所有在IIC总线上的设备都可以收到,紧接着主机会通过SDA发送设备地址,当发送的设备地址和某个设备匹配了,这个设备就呗选择为从机,从而进行通信,而其余的设备输出高阻态,断开与总线的连接。
发送了设备地址后,紧接着回发送一位来,来确定数据传输的方向(主机读数据还是主机发数据),然后从机会产生一个应答信号或者非应答信号来告诉主机当前的是否继续发送数据,
(ii)主机读数据到从机
和主机写数据到从机不同的是,这次是从机发送数据,主机接受,但是开始的第一个字节,仍然是主机发送起始信号和设备广播地址,来和某个设备进行匹配,但是数据传输位要设置为“读”,以及终止信号也由主机发送
(iii)起始信号和终止信号
![](https://i-blog.csdnimg.cn/blog_migrate/bb5607cbe6527b7d5463145890bf76a1.png)
SDA数据线在SCL时钟线的一个时钟周期传输一位数据
当时钟电平为低电平的时候可以允许数据的变化
当时钟电平拉高的时候,数据线的数据为无效数据,
(iv)应答/非应答信号
当传递完一个字节(8bit)后,主机会释放对SDA的控制【加入第八位数据位0,此时SDA是低电平,这样如果不释放控制权,结果就是只可以产生应答信号,所以需要释放控制权交到从机手里】,由从机进行控制,紧接着在第九个时钟周期从机从机会向主机发送应答或者非应答信号,然后主机接收后进行判断是否还要进行传输数据。若为应答信号,则会继续传输下一个数据;反之,若为非应答信号,则会让主机发送终止信号,来结束传输。
传输数据的顺序为:起始信号+第一个字节(寻址信号和判断读写模式)+你需要传输的N个字节以及应答位+停止信号。
软件模拟iiC
寄存器代码解释:
找到你对应需要进行软件模拟IIC的接口
//初始化IIC
void IIC_Init(void)
{
RCC->APB2ENR|=1<<4;//先使能外设IO PORTC时钟
GPIOC->CRH&=0XFFF00FFF;//PC11/12 推挽输出
GPIOC->CRH|=0X00033000;
GPIOC->ODR|=3<<11; //PC11,12 输出高
}
起始和终止信号
}
//产生IIC起始信号
void IIC_Start(void)
{
SDA_OUT(); //sda线输出
IIC_SDA=1;
IIC_SCL=1;
delay_us(4);
IIC_SDA=0;//START:when CLK is high,DATA change form high to low
delay_us(4);
IIC_SCL=0;//钳住I2C总线,准备发送或接收数据
}
//产生IIC停止信号
void IIC_Stop(void)
{
SDA_OUT();//sda线输出
IIC_SCL=0;
IIC_SDA=0;
delay_us(4);
IIC_SCL=1;//STOP:when CLK is high DATA change form low to high
delay_us(4);
IIC_SDA=1;//发送I2C总线结束信号
}
应答和非应答信号
//产生ACK应答
void IIC_Ack(void)
{
IIC_SCL=0;
SDA_OUT();
IIC_SDA=0; //数据位拉低
delay_us(2);
IIC_SCL=1;
delay_us(2);
IIC_SCL=0;
}
//不产生ACK应答
void IIC_NAck(void)
{
IIC_SCL=0;
SDA_OUT();
IIC_SDA=1; //数据位拉高
delay_us(2);
IIC_SCL=1;
delay_us(2);
IIC_SCL=0;
}
主机等待应答信号的接收,当数据线被拉低的时候就代表接收了应答信号,让时钟线拉低继续进行传输,当SDA == 1,证明接收到了非应答信号,就会让主机发送停止信号,结束传输。
u8 IIC_Wait_Ack(void)
{
u8 ucErrTime=0;
SDA_IN(); //SDA设置为输入
IIC_SDA=1;delay_us(1);
IIC_SCL=1;delay_us(1);
while(READ_SDA)
{
ucErrTime++;
if(ucErrTime>250)
{
IIC_Stop();
return 1;
}
}
IIC_SCL=0;//时钟输出0
return 0;
}
读取字节和传输字节
传入的数据txd会和0X80(1000000)进行与运算,然后结果右移七位,换句话说就是把传进来的字节的最高位单拿出来,赋值给SDA,以便于传输,然后将字节左移一位,就会把第二位变为最高位,以此类推,将每一位都传入SDA。
void IIC_Send_Byte(u8 txd)
{
u8 t;
SDA_OUT();
IIC_SCL=0;//拉低时钟开始数据传输
for(t=0;t<8;t++)
{
IIC_SDA=(txd&0x80)>>7;
txd<<=1;
delay_us(2); //对TEA5767这三个延时都是必须的
IIC_SCL=1;
delay_us(2);
IIC_SCL=0;
delay_us(2);
}
}
//读1个字节,ack=1时,发送ACK,ack=0,发送nACK
u8 IIC_Read_Byte(unsigned char ack)
{
unsigned char i,receive=0;
SDA_IN();//SDA设置为输入
for(i=0;i<8;i++ )
{
IIC_SCL=0;
delay_us(2);
IIC_SCL=1;
receive<<=1;
if(READ_SDA)receive++;
delay_us(1);
}
if (!ack)
IIC_NAck();//发送nACK
else
IIC_Ack(); //发送ACK
return receive;
}