一、概述
1.背景
I2C(IIC,Inter-Integrated Circuit)总线是由Philips公司开发的一种简单、双向二线制同步串行总线。它只需要两根线即可在连接于总线上的器件之间传送信息。一种串行、半双工总线,主要用于近距离、低速芯片之间的通信。IIC总线有两根双向信号线,一根数据线SDA用于收发数据,一根时钟线SCL(SCK)用于通信双方时钟的同步;IIC总线硬件结构简单,成本较低,应用广泛。
主器件用于启动总线传送数据,并产生时钟以开放传送的器件,此时任何被寻址的器件均被认为是从器件。
I2C总线简化了硬件电路PCB布线,降低了系统成本,提高了系统可靠性。因为I2C芯片(如mpu6050、ft5x06等)除了这两根线和少量中断线,与系统再没有连接的线,用户常用IC可以很容易形成标准化和模块化,便于重复利用。
2.传输方向
在总线上主和从、发和收的关系不是恒定的,而取决于此时数据传送方向。
如果主机要发送数据给从器件,则主机首先寻址从器件,然后主动发送数据至从器件,最后由主机终止数据传送。
如果主机要接收从器件的数据,首先由主器件寻址从器件.然后主机接收从器件发送的数据,最后由主机终止接收过程。在这种情况下,主机负责产生定时时钟和终止数据传送。
3.速度
连接到相同总线上的IC数量只受总线最大电容的限制,串行的8位双向数据传输位速率在标准模式下可达100Kbit/s,快速模式下可达400Kbit/s,高速模式下可达3.4Mbit/s。
总线具有极低的电流消耗.抗高噪声干扰,增加总线驱动器可以使总线电容扩大10倍,传输距离达到15m;兼容不同电压等级的器件,工作温度范围宽。
4.地址
I2C总线上的每一个设备都可以作为主设备或者从设备,而且每一个设备都会对应一个唯一的地址(地址通过物理接地或者拉高,可以从I2C器件的数据手册得知,如AT24C02芯片,7位地址依次1010xxx, 最低三位可配,如果全部物理接地,则该设备地址为0x50),主从设备之间就通过这个地址来确定与哪个器件进行通信,在通常的应用中,我们把STM32作为主设备,把挂接在总线上的其他设备都作为从设备。
二、IIC通信过程
空闲时SCL和SDA为高电平,发数据时SCL低电平,收数据SCL高电平
1.I2C通信过程中,存在4种信号
1.起始信号(条件):表示双方做好通信准备
2.主机发送1字节数据,指明从机地址和后续字节的传送方向
3.应答信号:有应答信号和无应答信号。有应答信号SDA为低电平,无应答信号为高电平
4.停止信号(条件):告诉从机通信结束,释放总线
2.起始信号和停止信号
起始信号:当SCL为高电平,SDA从高电平变为低电平
终止信号:当SCL为高电平,SDA从低电平变为高电平
3.同步信号
时钟线SCL为低电平时发送器向数据线SDA上发送一位数据,此期间信号允许发送变化
时钟线SCL为高电平时接收器从数据线SDA上读取一位数据,此期间信号不允许发送变化,必须保持稳定
SCL时钟线作用:告诉发送器和接收器对数据收发的时机
4.数据收传输与应答
先传输最高位,后传输低位,发送完1字节后接收器必须发送1位应答位来回应发送器,所以一帧共有9位
总结:
IIC协议整个通信流程 = 起始信号 + 1字节(从机地址(确定目标7bit)和传输方向(0/1))+ 应答信号(有(SDA 0)/无(SDA 1) + 数据传输(发送/接收) + 停止信号
三、典型I2C时序
A表示应答,A非表示非应答,S表示起始信号,P表示终止信号
主机向从机发送数据: 数据发送过程:主机发送启动信号,主机发送1字节数据后确定数据传输方向,从机应答,主机再发送数据,从机应答; 数据停止传输过程的两种过程:一是主机发送数据后从机应答,主机主动发送停止信号;二是主机发送数据后从机不应答,主机再发送停止信号 从机向主机发送数据: 数据发送过程:主机发送启动信号,主机发送1字节数据后确定数据传输方向,从机应答,从机再发送数据,主机应答; 数据停止传输过程:从机向主机发送数据,主机不应答,主机再发送停止信号 主机先向从机发送数据,然后从机再向主机发送数据 发送过程: 主机发送启动信号,主机发送1字节数据后确定数据传输方向,从机应答,主机再发送数据,从机应答,从机数据接收完后,主机再发送启动信号(防止总线被抢占),主机发送1字节数据后确定数据传输方向,从机应答,从机发送数据,主机应答; 数据停止传输过程:从机向主机发送数据,主机不应答,主机再发送停止信号
主机向从机发送数据:
从机向主机发送数据:
四、应用领域
摄像头控制、触摸屏、无人机、计步器(加速度/角速度/陀螺仪传感器)、心率(心率传感器)、激光测距、FM收音机(FM调频收音模块)
五、EEPROM(AT24C02)test
1.特点
总容量256 (2k/8)个字节
接口:I2C
2.地址
A0-A2接地 000
1字节:写0xA0、读0xA1
3.读写时序图
向目标地址写入数据
DEVICE ADDRESS = 0xA0
WORD ADDRESS 写入EEPROM的地址(0~255)
DATA 写入数据 先发送要读的地址,再接收数据
4.test
参考
int mian(void)
{
USART1_init(115200);
printf("hello world\r\n");
AT24C02_Init();
if(!AT24C02_Check())
printf("check success!\r\n");
AT24C02_WriteOneByte(0,0xFF);
uint8_t rd_data = AT24C02_ReadOneByte(0);
printf("read data = %x\r\n",rd_data);
}
#define SDA_IN() {GPIOB->MODER &= ~(3<<(9*2)); GPIOB->MODER |= 0<<(9*2);}
#define SDA_OUT() {GPIOB->MODER &= ~(3<<(9*2)); GPIOB->MODER |= 1<<(9*2);}
#define IIC_SCL PBout(8)
#define IIC_SDA PBout(9)
#define READ_SDA PBin(9)
void delay_ms(uint32_t n)
{
while(n--)
{
SysTick->CTRL = 0;
SysTick->LOAD = 168000-1;
SysTick->VAL = 0;
SysTick->CTRL = 5;
while ((SysTick->CTRL & 0x00010000)==0);
}
SysTick->CTRL = 0;
}
void delay_us(uint32_t n)
{
while(n--)
{
SysTick->CTRL = 0;
SysTick->LOAD = 168-1;
SysTick->VAL = 0;
SysTick->CTRL = 5;
while ((SysTick->CTRL & 0x00010000)==0);
}
SysTick->CTRL = 0;
}
void IIC_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB,ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOB, &GPIO_InitStructure);
IIC_SCL = 1;
IIC_SDA = 1;
}
void IIC_Start(void)
{
SDA_OUT();
IIC_SCL = 1;
IIC_SDA = 1;
delay_us(4);
IIC_SDA = 0;
delay_us(4);
IIC_SCL=0;
}
void IIC_Stop(void)
{
SDA_OUT();
IIC_SCL = 0;
IIC_SDA = 0;
delay_us(4);
IIC_SCL = 1;
IIC_SDA = 1;
delay_us(4);
}
void IIC_Ack(void)
{
IIC_SCL=0;
SDA_OUT();
IIC_SDA=0;
delay_us(2);
IIC_SCL=1;
delay_us(2);
IIC_SCL=0;
}
void IIC_NAck(void)
{
IIC_SCL=0;
SDA_OUT();
IIC_SDA=1;
delay_us(2);
IIC_SCL=1;
delay_us(2);
IIC_SCL=0;
}
uint8_t IIC_Wait_Ack(void)
{
uint8_t ucErrTime;
SDA_IN();
while(READ_SDA)
{
ucErrTime++;
if(ucErrTime>200)
{
IIC_Stop();
return 1;
break;
}
}
IIC_SCL = 0;
return 0;
}
void IIC_Send_Byte(uint8_t txd)
{
int tmp;
IIC_SCL = 0;
SDA_OUT();
for(tmp = 7;tmp>=0;tmp--)
{
if(txd & (1<<tmp))
IIC_SDA = 1;
else
IIC_SDA = 0;
delay_us(2);
IIC_SCL = 1;
delay_us(2);
IIC_SCL = 0;
delay_us(2);
}
}
uint8_t IIC_Read_Byte(uint8_t ack)
{
int i;
uint8_t receive=0;
SDA_IN();
for(i=7;i>=0;i--)
{
IIC_SCL = 0;
delay_us(2);
IIC_SCL = 1;
if(IIC_SDA)
receive |= (1<<i);
delay_us(1);
}
if (!ack)
IIC_NAck();
else
IIC_Ack();
return receive;
}
void AT24C02_Init(void)
{
IIC_Init();
}
void AT24C02_WriteOneByte(uint16_t WriteAddr,uint8_t DataToWrite)
{
IIC_Start();
IIC_Send_Byte(0XA0);
IIC_Wait_Ack();
IIC_Send_Byte(WriteAddr%256);
IIC_Wait_Ack();
IIC_Send_Byte(DataToWrite);
IIC_Wait_Ack();
IIC_Stop();
delay_ms(5);
}
uint8_t AT24C02_ReadOneByte(uint16_t ReadAddr)
{
uint8_t temp=0;
IIC_Start();
IIC_Send_Byte(0XA0);
IIC_Wait_Ack();
IIC_Send_Byte(ReadAddr%256);
IIC_Wait_Ack();
IIC_Start();
IIC_Send_Byte(0XA1);
IIC_Wait_Ack();
temp=IIC_Read_Byte(0);
IIC_Stop();
return temp;
}
uint8_t AT24C02_Check(void)
{
uint8_t temp;
temp=AT24C02_ReadOneByte(255);
if(temp==0XFF)return 0;
else
{
AT24C02_WriteOneByte(255,0XFF);
temp=AT24C02_ReadOneByte(255);
if(temp==0X55)return 0;
}
return 1;
}
|