首先我们来认识一下i2c通讯协议
i2c总线只需要串行数据SDA线以及串行时钟SCL线,两条线都是双向的。每个从器件都有一个唯一的地址以便识别。
i2c传输过程:start-从机地址-应答/非应答-R/W(1为读/0为写)-数据传输-应答/非应答-stop
数据传输每个字节都需要应答/非应答信号
模拟i2c传输协议:
根据时序图可以知道:start就是拉高SCL,给SDA下降沿;stop是拉高SCL,给SDA上升沿。
当传输数据时即SCL为高电平期间,SDA上的数据必须保持稳定,只有在SCL上的信号为低电平期间,SDA上的高电平或低电平状态才允许变化。
数据传输一般选择8位,在8位数据传输之后,必须进行应答/非应答信号的接受。
应答/非应答:应答的意思是在数据传输之后,接收数据的器件必须传输一个应答信号给主机,如果没有传输数据,那么判断数据传输不成功,需要stop或者reset。软件模拟应答信号就是在数据传输结束之后,需要将SDA拉高释放总线,然后从机会将SDA置低,主机据此判断SDA电平,确定是否收到应答信号(SDA=0为应答,反之则然)。
EEPROM:24系列EEPROM是一种普遍使用的非易失性存储器,24Cxx中xx的单位是kb,例如24C08的存储容量是8kb。
器件地址:一般datasheet中会有给出器件地址。
字节地址:对EEPROM进行读写时,需要选定字节地址,以24C08为例,读写地址在 0x00-0x3ff。
读/写时序:对于24C08来说,内部分为页,每页共可存储16字节数据,即高四位一致的地址。基于此可以知 道,在写数据或者读数据时,要确定是否读/写数据是连续的,而且是不是在同一页中,如果所读/写 数据超过一页,那么则需要进行换页操作。
写一整个字节的数据到EEPROM:
void I2C_EE_ByteWrite(u8* pBuffer,u8 WriteAddr)
{
I2C_GenerateSTART(I2C1,ENABLE); //start while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT)); //设置主机模式(R/W)I2C_Send7bitAddress(I2C1,EEPROM_ADDRESS,I2C_Direction_Transmitter); //向指定的从IIC设备传送地址字,选择为发送方向
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED)); //等待这次选择过程成功
I2C_SendData(I2C1,WriteAddr); //通过外设I2C1发送地址
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //等待字节发送完成I2C_SendData(I2C1,*pBuffer); //写入数据到EEPROM内部 while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED)); //等待字节发送完成
I2C_GenerateSTOP(I2C1,ENABLE); //stop
}
整个函数理解起来很简单,就是按照步骤走的:start-设置主从模式-传送地址到i2c-然后通过i2c总线把地址发送到从机-写数据-stop (这其中的I2C_CheckEvent()函数就是应答/非应答信号)。
假如写多页数据原理上很简单,就是地址递增就好了,这里就不写了。
读取从机某个地址存储的数据:
void I2C_EE_BufferRead(u8* pBuffer,u8 ReadAddr,u16 NumByteToRead)
{
I2C_EE_WaitEepromStandbyState(); //等待EEPROM
while(I2C_GetFlagStatus(I2C1,I2C_FLAG_BUSY)); //检测总线是否忙碌
I2C_GenerateSTART(I2C1,ENABLE);
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT));
I2C_Send7bitAddress(I2C1,EEPROM_ADDRESS,I2C_Direction_Transmitter); //向指定的从iic设备传输地址,选择发送方向
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED));
I2C_Cmd(I2C1,ENABLE);
I2C_SendData(I2C1,ReadAddr); //发送要读取的EEPROM数据的起始地址
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_TRANSMITTED));
I2C_GenerateSTART(I2C1,ENABLE); //再次发送起始条件
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_MODE_SELECT));
I2C_Send7bitAddress(I2C1,EEPROM_ADDRESS,I2C_Direction_Receiver); //向指定的从iic设备传送地址,选择接受方向
while(!I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED));
while(NumByteToRead) //直到读取完成
{
if(NumByteToRead==1)
{
I2C_AcknowledgeConfig(I2C1,DISABLE); //禁止IIC的应答功能
I2C_GenerateSTOP(I2C1,ENABLE);
}
if(I2C_CheckEvent(I2C1,I2C_EVENT_MASTER_BYTE_RECEIVED)) //检测是否接收到数据
{
*pBuffer=I2C_ReceiveData(I2C1); //读取通过I2C1最近接受的数据
pBuffer++;
NumByteToRead--;
}
}
I2C_AcknowledgeConfig(I2C1,ENABLE); //使能I2C的应答功能
}
读数据步骤:检测总线是否空闲-start-发送器件地址和写模式(称之为伪写)-发送读取数据的地址-start-发送7位器件地址和读模式-接受数据
注意:1,在使用i2c时,SCL和SDA需要上拉
2,很多人说硬件i2c存在bug,一般都是用模拟i2c,或者把i2c放在优先级比较高的DMA中使用,暂时还没有定论。
上一篇:STM32F103基于DMA接收不定帧长USART数据
下一篇:STM32的printf函数实现方法
推荐阅读最新更新时间:2024-03-16 15:39