(学习笔记3)STM32F429库函数之I2C读取EEPROM

首先,I2C总线由两条线——串行数据(SDA)和串行时钟(SCL),这是同步通信,也是半双工通信,不能同时读写。每个器件都有一个唯一的地址识别,当总线空闲时I2C两条线都是高电平,只有当连接到总线的器件的输出级是漏级开路或者集电极开路,唯一的地址能够被正确的器件应答。因为应答信号为0,即低电平,能够拉低总线,而同时其它器件(除主机外)引脚仍表现出高阻态,即可以开始进行通信。
I2C读写EEPROM的通讯过程——
起始信号:在SCL为高电平,SDA由高变低,总线被认为处于忙的状态。调用I2C_GenerateSTART(...)产生起始信号,若起始信号成功发送则产生事件5:SB=1(I2C_SR1的0位置1,表示起始位已发送)
发送7位地址:这7位地址为EEPROM的设备地址,调用I2C_Send7bitAddress(...)发送,这里用的是8位读/写地址,LSB(最低位)为0表示写入,为1时表示读出。输入形参的时候要注意。每发送一个字节,接收器作出应答。若地址发送结束且接收到应答信号则产生事件6:ADDR=1(I2C_SR1的1位置1 ),这里的事件6位发送事件。
发送内存地址:这边的内存地址为EEPROM内部读写的开始地址,调用I2C_SendData(...)发送一个数据,EEPROM会自动识别为内存地址。数据发送成功则产生事件8_2,事件8_2与8的区别为8_2多了BTF(Byte Transfer Finished)标志,表示字节传输也完成了,相同时的是TxE(Data register empty)=1,数据寄存器为空。
if(写一个字节)//(如果要写入多个字节,可以采用页写入的方式,但是02这块芯片每页最多写8个字节,就避免了写一个字节就停止一次,就可以最多写8个字节停止一次了)
{
接着发送一个数据,那么这个数据就会保存到之前发送的内存地址单元,判断是否发送完成。
若发送完成,调用I2C_GenerateSTOP(...)产生停止信号 (写一个字节结束)
}
else(读一个字节)
{
1、重复起始信号,判断事件5
2、发送7位地址即设备地址,这边注意的是,最低位为1表示读,判断事件6(这里的事件6为接收事件)
3、在接收数据之前,先关闭应答位,为的是在数据读完之后产生非应答信号。(这边可以先关闭应答位,产生停止信号,最后发送数据,因为非应答位在数据发送结束才会生成,而停止位只有在非应答信号之后才能生成,大神跟我说这是预配置,我觉得很有道理,能帮助理解逻辑)这里是从机端接收到地址后开始发送数据,不需要调用任何函数,只需要等待事件7(RxNE,数据寄存器不为空)发生就好了。用ReadTemp = I2C_ReceiveData(...)来读取从EEPROM中读来的数据显示出来。
4、停止
}
停止信号:在SCL为高电平,SDA由低变高,总线在停止后的某段时间后,总线进入空闲状态
读写:在SCL为高电平,采样数据,只有SCA为低电平时,SDA可以变换状态
关于在判断事件5中的SB位,在置1后运行下面语句时,由软件序列先读取SR1寄存器后写入DR寄存器清零。事件6中的ADDR置1在I2C_CheckEvent()中读取SR1寄存器后读取SR2寄存器这样的软件序列清零。之后的事件8_2和事件7中的TxE由软件写入DR寄存器清零,BTF、RxNE这两个标志由软件读或写DR寄存器清零。
还有,在传输完一次之后需要等待EEPROM处理内部时序,字节应该还好,当用页写入的话,是一定要等的。。。应该是
妄想在产生起始信号后不写入DR寄存器而想直接通过寄存器操作修改SB位,结果是失败的。
这里用的EEPROM芯片是AT24C02,用的是F429自带的I2C片上外设,没有用软件模拟(表示不会。。。)。
文章纯粹是个人温习用,如有不对的地方,还请大神指出
【(学习笔记3)STM32F429库函数之I2C读取EEPROM】学的是野火,还是得标明出处。

    推荐阅读