IIC数据传输速率
标准模式(100 kbps)、快速模式(400 kbps)和高速模式(3.4 Mbps),
另外一些变种实现了低速模式(10 kbps)和快速+模式(1 Mbps)
IIC通信方式
IIC通信方式为半双工,只有一根SDA线,同一时间只可以单向通信,485也为半双工,SPI和uart为双工。
I2C总线特征
I2C总线上的每一个设备都可以作为主设备或者从设备,而且每一个设备都会对应一个唯一的地址(地址通过物理接地或者拉高,可以从I2C器件的数据手册得知,如TVP5158芯片,7位地址依次bit6~bit0:x101 1xxx, 最低三位可配,如果全部物理接地,则该设备地址为0x58, 而之所以7bit因为1个bit要代表方向,主向从和从向主),主从设备之间就通过这个地址来确定与哪个器件进行通信,在通常的应用中,我们把CPU带I2C总线接口的模块作为主设备,把挂接在总线上的其他设备都作为从设备。
I2C总线上可挂接的设备数量受总线的最大电容400pF 限制,如果所挂接的是相同型号的器件,则还受器件地址位的限制。
I2C总线数据传输速率在标准模式下可达100kbit/s,快速模式下可达400kbit/s,高速模式下可达3.4Mbit/s。一般通过I2C总线接口可编程时钟来实现传输速率的调整,同时也跟所接的上拉电阻的阻值有关。
I2C总线上的主设备与从设备之间以字节(8位)为单位进行双向的数据传输。
I2C总线协议
I2C协议规定,总线上数据的传输必须以一个起始信号作为开始条件,以一个结束信号作为传输的停止条件。起始和结束信号总是由主设备产生(意味着从设备不可以主动通信?所有的通信都是主设备发起的,主可以发出询问的command,然后等待从设备的通信)。
起始和结束信号产生条件:总线在空闲状态时,SCL和SDA都保持着高电平,当SCL为高电平而SDA由高到低的跳变,表示产生一个起始条件;当SCL为高而SDA由低到高的跳变,表示产生一个停止条件。
在起始条件产生后,总线处于忙状态,由本次数据传输的主从设备独占,其他I2C器件无法访问总线;而在停止条件产生后,本次数据传输的主从设备将释放总线,总线再次处于空闲状态。起始和结束如图所示:
在了解起始条件和停止条件后,我们再来看看在这个过程中数据的传输是如何进行的。前面我们已经提到过,数据传输以字节为单位。主设备在SCL线上产生每个时钟脉冲的过程中将在SDA线上传输一个数据位,当一个字节按数据位从高位到低位的顺序传输完后,紧接着从设备将拉低SDA线,回传给主设备一个应答位, 此时才认为一个字节真正的被传输完成。当然,并不是所有的字节传输都必须有一个应答位,比如:当从设备不能再接收主设备发送的数据时,从设备将回传一个否 定应答位。
数据传输的过程如图所示:
I2C总线上的每一个设备都对应一个唯一的地址,主从设备之间的数据传输是建立在地址的基础上,也就是说,主设备在传输有效数据之前要先指定从设备的地址,地址指定的过程和上面数据传输的过程一样,只不过大多数从设备的地址是7位的,然后协议规定再给地址添加一个最低位用来表示接下来数据传输的方向,0表示主设备向从设备写数据,1表示主设备向从设备读数据。向指定设备发送数据的格式如图所示:(每一最小包数据由9bit组成,8bit内容+1bit ACK, 如果是地址数据,则8bit包含1bit方向)
代码
芯片官方提供的源代码(一直认为 多看源代码 可以提高自己编写程序的质量)
/*
*********************************************************************************************************
*********************************************************************************************************
*/
void I2CWriteCtrl(void)
{
static uint8_t temp_i = 0;
//uint8_t temp_j = 0;
if(I2CbusyState())
{
return;
}
#if IIC_WRITE_ARRAY_EN
if(I2CWriteArray())
{
return;
}
#endif//IIC_WRITE_ARRAY_EN
if(1 == IICwrite.WriteAllTrig)
{
temp_i = 0;
IICwrite.WriteAllTrig = 2;
}
I2CWriteInput();
for( ; temp_i<sizeof(STRUCT_IICM); temp_i++)
{
if((IICM.DataWrite[temp_i] != IICMtemp.DataWrite[temp_i])||(IICwrite.WriteAllTrig))
{
I2CmultiByteWrite(IICM_ADDR_WRITE[temp_i], &IICM.DataWrite[temp_i], 1);
IICMtemp.DataWrite[temp_i] = IICM.DataWrite[temp_i];
temp_i++;
break;
}
}
if((temp_i) >= sizeof(STRUCT_IICM))
{
IICwrite.WriteAllTrig = 0;
temp_i = 0;
}
}
/*
*/
void I2CReadCtrl(void)
{
if(I2CbusyState())
{
return;
}
switch(IICMread.State)
{
case 0:
if(TimeOut(&IICMread.Timer, 10))
{
IICMread.Timer = 0;
IICMread.State ++;
}
break;
case 1:
#if PB_818
I2CmultiByteRead(0X0F, &IICMread.Data[0], 6);
#endif//PB_818
#if PB_817
I2CmultiByteRead(0X09, &IICMread.Data[0], 2);
#endif//PB_817
#if PB_823
I2CmultiByteRead(0X0F, &IICMread.Data[0], 9);
#endif//PB_823
#if PB_816
I2CmultiByteRead(0X0E, &IICMread.Data[0], 3);
#endif//PB_816
#if PB_822
I2CmultiByteRead(0X0E, &IICMread.Data[0], 3);
#endif//PB_822
#if PB_816B
I2CmultiByteRead(0X0E, &IICMread.Data[0], 3);
#endif//PB_816B
IICMread.State ++;
break;
case 2:
#if PB_818
ADCVAL.Vbat = (((IICMread.Data[0]<<2)|(IICMread.Data[1]>>6))+1) * 25;
Solar_Battery_Voltage = ADCVAL.Vbat;
//ADCVAL.Vbat = 4000;
ADCVAL.IbusPower = (((IICMread.Data[2]<<2)|(IICMread.Data[3]>>6))+1)*10;
ADCVAL.Ibat = (((IICMread.Data[4]<<2)|(IICMread.Data[5]>>6))+1)*20;
#endif//PB_818
#if PB_817
ADCVAL.status1 = IICMread.Data[0];
ADCVAL.status2 = IICMread.Data[1];
#endif//PB_817
#if PB_823
ADCVAL.Vbat = IICMread.Data[3] * 25;
ADCVAL.IbusPower = IICMread.Data[8] * 50;
ADCVAL.Ibat = IICMread.Data[7] * 25;
ADCVAL.Iusba1 = IICMread.Data[5] * 50;//I Q1
ADCVAL.Iusba2 = IICMread.Data[6] * 50;//I Q4
ADCVAL.status3 = IICMread.Data[0];
ADCVAL.Vusba2 = IICMread.Data[2] * 25;//V PMID
ADCVAL.Vusba1 = IICMread.Data[1] * 25;//VIN
ADCVAL.VbusPower = IICMread.Data[4] * 50;
#endif//PB_823
#if PB_816
ADCVAL.status1 = IICMread.Data[0];
ADCVAL.status2 = IICMread.Data[1];
ADCVAL.status3 = IICMread.Data[2];
#endif//PB_816
#if PB_822
ADCVAL.status1 = IICMread.Data[0];
ADCVAL.status2 = IICMread.Data[1];
ADCVAL.status3 = IICMread.Data[2];
#endif//PB_822
#if PB_816B
ADCVAL.status1 = IICMread.Data[0];
ADCVAL.status2 = IICMread.Data[1];
ADCVAL.status3 = IICMread.Data[2];
#endif//PB_816B
IICMread.State ++;
break;
case 3:
#if PB_818
I2CmultiByteRead(0X17, &IICMread.Data[0], 1);
#endif//PB_818
#if PB_823
I2CmultiByteRead(0X1B, &IICMread.Data[0], 3);
#endif//PB_823
#if PB_822
I2CmultiByteRead(0X04, &IICMread.Data[0], 3);
#endif//PB_822
#if PB_816B
I2CmultiByteRead(0X016, &IICMread.Data[0], 1);
#endif//PB_816B
IICMread.State ++;
break;
case 4:
#if PB_818
ADCVAL.status1 = IICMread.Data[0];
#endif//PB_818
#if PB_823
ADCVAL.status2 = IICMread.Data[0];
ADCVAL.status1 = IICMread.Data[2];
#endif//PB_823
#if PB_822
ADCVAL.status2 = IICMread.Data[0];
ADCVAL.status1 = IICMread.Data[2];
#endif//PB_823
#if PB_816B
ADCVAL.status4 = IICMread.Data[0];
#endif//PB_816B
IICMread.State ++;
break;
// case 5:
// IICMread.State ++;
// break;
// case 6:
// IICMread.State ++;
// break;
default:
IICMread.State = 0;
break;
}
}
void I2CmultiByteRead(uint8_t addr, uint8_t *parray, uint8_t num)
{
I2CMirq.Addr = addr;
I2CMirq.pAddRead = parray;
I2CMirq.Num = num;
I2CMirq.Busy = 1;
I2CMirq.Mode = 0;
I2CMirq.State = 0;
I2C_Master_Start();
}
void I2CmultiByteWrite(uint8_t addr, uint8_t *parray, uint8_t num)
{
I2CMirq.Addr = addr;
I2CMirq.pAddWrite = parray;
I2CMirq.Num = num;
I2CMirq.Busy = 1;
I2CMirq.Mode = 1;
I2CMirq.State = 0;
I2C_Master_Start();
}
void I2C_Master_Start(void)
{
I2C_M->I2CCTL = 8; //start
}