IIC总线原理和编程

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
}


  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
IIC总线和SPI总线是两种不同的通信协议,它们在硬件连接和数据传输方式上有一些区别。 1. 硬件连接: IIC总线使用双线(SCL和SDA)连接,而SPI总线使用三线(SCLK、SDO和SDI)或四线(SCLK、SDO、SDI和SS)连接。SPI总线还可以通过片选信号(SS)连接多个从设备。 2. 数据传输方式: IIC总线是双向的,可以在同一根线上进行数据输入和输出。而SPI总线是全双工的,可以同时进行数据的输入和输出。SPI总线还可以通过多根数据线(MOSI和MISO)实现全双工通信。 3. 适用范围: IIC总线适用于近距离、非常性的数据通信,具有总线仲裁机制,可以实现设备组网。SPI总线适用于多个SPI设备互相连接的场景,可以实现全双工通信。 4. 端口占用: IIC总线在CPU端口占用较少,而SPI总线需要更多的端口资源。 总的来说,IIC总线和SPI总线在硬件连接、数据传输方式和适用范围上有所不同。选择使用哪种总线取决于具体的应用需求和硬件资源的限制。 #### 引用[.reference_title] - *1* [硬件:SPI总线IIC总线区别](https://blog.csdn.net/ilufam1314/article/details/111576995)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [I2C和SPI总线对比](https://blog.csdn.net/skyflying2012/article/details/8237881)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

七 六 伍

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值