STM32:IIC详解

发布于:2024-11-03 ⋅ 阅读:(45) ⋅ 点赞:(0)

1.IIC历史

2.IIC优势

3.IIC时序

3.1 初始化IIC

//初始化IIC
void PCA9557_Init(void)
{					     
 	RCC->APB2ENR |= 1<<4;//先使能外设IO PORTC时钟 							 
	GPIOC->CRH &= 0XFFFF00FF;//PC10/11 推挽输出
	GPIOC->CRH |= 0X00003300;	   
	GPIOC->ODR |= 1<<10;     //PC10 输出高
	GPIOC->ODR |= 1<<11;     //PC11 输出高
}

3.2 产生IIC起始信号

//产生IIC起始信号
void PCA9557_Start(void)
{
	PCA9557_SDA_OUT(); //sda线输出
	PCA9557_SDA=1;
	PCA9557_SCL=1;
	delay_us(4);
	PCA9557_SDA=0;//START:when CLK is high,DATA change form high to low
	delay_us(4);
	PCA9557_SCL=0;//钳住I2C总线,准备发送或接收数据
}

3.3 产生IIC停止信号

//产生IIC停止信号
void PCA9557_Stop(void)
{
	PCA9557_SDA_OUT();//sda线输出
	PCA9557_SCL=0;
	PCA9557_SDA=0;//STOP:when CLK is high DATA change form low to high
	delay_us(4);
	PCA9557_SCL=1;
	PCA9557_SDA=1;//发送I2C总线结束信号
	delay_us(4);
}

3.4 等待ACK

//等待应答信号到来
//返回值:1,接收应答失败
// 0,接收应答成功
uint8_t PCA9557_Wait_Ack(void)
{
	uint8_t ucErrTime=0;
	PCA9557_SDA_IN(); //SDA设置为输入
	PCA9557_SDA=1;delay_us(1);
	PCA9557_SCL=1;delay_us(1);
	while(PCA9557_READ_SDA)
	{
		ucErrTime++;
		if(ucErrTime>250)
		{
			PCA9557_Stop();
			return 1;
		}
	}
	PCA9557_SCL=0;//时钟输出0
	return 0;
}

3.5 产生ACK / 不产生ACK

//产生ACK应答
void PCA9557_ACK(void)
{
	PCA9557_SCL=0;
	PCA9557_SDA_OUT();
	PCA9557_SDA=0;
	delay_us(2);
	PCA9557_SCL=1;
	delay_us(2);
	PCA9557_SCL=0;
}

//不产生ACK应答
void PCA9557_NAck(void)
{
	PCA9557_SCL=0;
	PCA9557_SDA_OUT();
	PCA9557_SDA=1;
	delay_us(2);
	PCA9557_SCL=1;
	delay_us(2);
	PCA9557_SCL=0;
}

3.6 IIC发送一个字节

//IIC发送一个字节
//返回从机有无应答
//1,有应答
//0,无应答
uint8_t PCA9557_Send_Byte(uint8_t txd)
{
	uint8_t t;
	PCA9557_SDA_OUT();
	PCA9557_SCL=0;//拉低时钟开始数据传输
	for(t=0;t<8;t++)
	{
		if (txd&0x80) PCA9557_SDA = 1;
		else PCA9557_SDA = 0;
		txd<<=1;
		delay_us(2); //对TEA5767这三个延时都是必须的
		PCA9557_SCL=1;
		delay_us(2);
		PCA9557_SCL=0;
		delay_us(2);
	}
	t=PCA9557_Wait_Ack();  //这里固定只要写字节就需要等待应答
	return (t);
}

3.7 IIC读取一个字节

//读1个字节,ack=1时,发送ACK,ack=0,发送nACK
uint8_t PCA9557_Read_Byte(uint8_t ack)
{
	uint8_t i,receive=0;
	PCA9557_SDA_IN();//SDA设置为输入
	for(i=0;i<8;i++ )
	{
		PCA9557_SCL=0;
		delay_us(2);
		PCA9557_SCL=1;
		receive<<=1;
		if(PCA9557_READ_SDA)receive |= 1;
		delay_us(1);
	}
	if (!ack)
	PCA9557_NAck();//发送nACK
	else
	PCA9557_ACK(); //发送ACK
	return receive;
}

3.8 IIC发送多字节

uint8_t PCA9557Write(uint8_t addr, uint8_t len, uint8_t *pData)
{
	uint8_t i;
	PCA9557_Start();
	if(PCA9557_Send_Byte(PCA9557_ADDR_WRITE)==1)
	{
		PCA9557_Stop();
		return 0;
	}
	if(PCA9557_Send_Byte(addr)==1)
	{
		PCA9557_Stop();
		return 0;
	}
	for(i = 0; i < len; i++)
	{
		if(PCA9557_Send_Byte(pData[i])==1)
		{
			PCA9557_Stop();
			return 0;
		}
	}
	PCA9557_Stop();
	
	return 1;
}

3.9 IIC读取多字节

uint8_t PCA9557Read(uint8_t addr,uint8_t len,uint8_t *pData)
{
	uint8_t i;
	PCA9557_Start();
	if(PCA9557_Send_Byte(PCA9557_ADDR_WRITE)==1)
	{
		PCA9557_Stop();
		return 0;
	}
	if(PCA9557_Send_Byte(addr)==1)
	{
		PCA9557_Stop();
		return 0;
	}
	PCA9557_Start();
	if(PCA9557_Send_Byte(PCA9557_ADDR_READ)==1)
	{
		PCA9557_Stop();
		return 0;
	}
	for(i=0; i<len-1; i++)
	{
		pData[i] = PCA9557_Read_Byte(1);
	}
	pData[len-1] = PCA9557_Read_Byte(0);
	PCA9557_Stop();
	
	return 1;
}

4 IIC问题

4.1 STM32 HAL库的IIC设备地址没有左移

#define I2C_7BIT_ADD_WRITE(__ADDRESS__)                    ((uint8_t)((__ADDRESS__) & (uint8_t)(~I2C_OAR1_ADD0)))
#define I2C_7BIT_ADD_READ(__ADDRESS__)                     ((uint8_t)((__ADDRESS__) | I2C_OAR1_ADD0))

底层中并没有把设备的地址左移,而是直接把最低位改为“0”或“1”,需要自己把器件地址的左移了一位。

	stat = HAL_I2C_Mem_Read(&hi2c1,(PCA9557_I2C_BUS_ADDR << 1),PCA9557_INPUT_REG,1,&io_statu,1,0xFF);
	
	if(stat != 0 )
	{
	  LOG_INFO("io %d Read Error STAT %d\r\n",io,stat);
	}
	

今日签到

点亮在社区的每一天
去签到