DSP28335I2C接口应用 下载本文

内容发布更新时间 : 2024/12/24 7:05:23星期一 下面是文章的全部内容请认真阅读。

实用文档

DSP I2C 应用说明

1.示例程序中几种状态

第一次看i2c_eeprom示例程序,对程序中的MsgStatus信息状态切换非常懵懂,为什么要有这几个状态?状态切换顺序如何安排?一大堆的状态,让人有些摸不着头脑。先把程序中的头文件

涉及的7种状态分析一下。

// I2C Message Commands for I2CMSG struct #define I2C_MSGSTAT_INACTIVE 0x0000 //未激活状态:一般成功发送数据或者//接受数据后可以设置信息状态为此状态,告诉用户可进行下一次的写数据或读数据。 #define I2C_MSGSTAT_SEND_WITHSTOP 0x0010 //发送带停止位数据:这是为写数据而设///的状态,写入地址和数据之后发个停止位告诉存储器数据写入完毕。

#define I2C_MSGSTAT_WRITE_BUSY 0x0011 //写数据忙状态:在将待写的数据放入//缓存后,就可以使能IIC传输数据了,然后把信息状态设为该状态,意在告诉用户:数据//已经在传送过程中。当然是否传送完毕,还需要通过查询SCD位来判断。 #define I2C_MSGSTAT_SEND_NOSTOP 0x0020//发送无停止位数据:这个状态是为了读//取数据而设的,有查阅过AT24C1024EEPROM存储器使用手册的读者知道,在读数据之前//要发送数据的地址,发完地址不能产生停止位,这是存储器硬件设计决定的。设为这个状//态意在告诉读者,可以发送要读取的数据的地址了。 #define I2C_MSGSTAT_SEND_NOSTOP_BUSY 0x0021//发送无停止位数据忙状态:这个状态是//为了读取数据而设的,似于I2C_MSGSTAT_WRITE_BUSY,说明地址数据已经在传送过程中。//传送是否成功,还要看ARDY的状态。 #define I2C_MSGSTAT_RESTART 0x0022//重发开始位状态:这个状态也是为读取 ////数据而设。我们知道,读取存储器数据主要分两个步骤:第一,发送START位+设备地址

//+数据地址+无停止位。第二,再发START位+设备地址,紧接着存储器发送数据到IIC接收

//缓存器(I2CDRR),接收到设定好的数据数量(I2CCNT值)时输出停止位STOP.

//值得注意的是:理论上写完数据就能马上读取 数据 ,但事实上EEPROM存储器仍需要一 ////定延时来存储数据,约有2ms左右。通过示波器可以观察到,写完数据后,并不能马上 //成功读取数据,也就是说读数据的第一步骤要重复好几次(总线为50K时,大约要重复 //8次)才能成功。 #define I2C_MSGSTAT_READ_BUSY 0x0023//读取数据忙状态:这个状态是为读取数 //据而设。在读数据的第二步骤中,发完START位+设备地址后,就设为这一状态。意在说//明IIC开始等待接收固定数量(I2CCNT值)的数据。可以通过查询ARDY位判断。

//头文件中的其他定义应该没什么大问题了!

2.AT24C1024 EEPROM读写数据格式

(1)AT24C1024设备地址: 1 0 1 0 0 A1 P0 R/W

文案大全

实用文档

(2)单字节写入:START -> 发送从设备地址(写控制码R/W=0) -> 处理Ack -> 发送字节地址 -> 处理Ack [-> 发送1字节数据 -> 处理Ack] -> STOP。如下图:

(3)按页写入:START -> 发送从设备地址(写控制码R/W=0) -> 处理Ack -> 发送字节地址 -> 处理Ack [-> 发送1字节数据 -> 处理Ack-> 发送第2字节数据 -> 处理Ack-> 发送第3字节数据 -> 处理Ack……直到发完X字节] -> STOP。如下图。注意,连续写入的数据字节数不能超过每页所能容下的总量。如果写入的数据超过一页的长度,将发生回卷,即从EEPROM的0地址处进行数据覆盖。比如,AT24C1024有512页,每页最大容纳256字节。超过这个长度,地址指针将从每页首地址重新开始。

(4)随机单字节读取:第一步:发START -> 发送从设备地址(写控制码R/W=0) -> 处理Ack -> 发送字节地址高位 -> 处理Ack -> 发送字节地址低位 -> 处理Ack ->第二步:发START -> 发送器件地址(读控制码R/W=1) -> 处理Ack -> 接收1字节数据 -> STOP。

(5)随机连续读取:在随机单字节读取操作的STOP信号发送之前,加入若干个 [-> 发送Ack -> 接收1字节数据] 即可实现。

文案大全

实用文档

(6)当前位置单字节读取:START -> 发送从设备地址(读控制码) -> 处理Ack -> 发送字节地址 -> 处理Ack -> 接收1字节数据 -> STOP。当前指的是之前进行过读取操作但是没有发送STOP信号,EEPROM芯片内部指针所在的位置即为当前位置。

(7)当前位置连续读取:在当前位置单节读取操作的STOP信号发送之前,加入若干个 [-> 发送Ack -> 接收1字节数据] 即可实现。

备注:部分内容引用百度文库中的《I2C读写流程》文档

3.主程序说明及流程图

详细的注释可以参见程序附录,,这里主要解释一下设备地址取0x50的缘由。从AT24C1024的数据手册可知,设备地址为

。而在程序

中设置的地址是0x50,即0B0101 0000。看似不妥,其实,这个例子中设备地址采用7位地址模式,这样只取0x50的低7位作为设备地址的高七位,设备地址的最低位R/W则由寄存器I2CMDR中的TRX位决定。这样读数据时设备地址为0B1010 0001,写数据时设备地址改为0B1010 0000。通过示波器可以验证这些数据。

问题1:每次检测到SCD中断时,也就是顺利发完指定数量的数据或者接收指定数量的数据了,理论上I2caRegs.I2CCNT==0,但是本人设了几个中断点去观察该寄存器的值,发现总是不为零。通过观察I2caRegs.I2CFFTX.bit.TXFFST和I2caRegs.I2CFFRX.bit.RXFFST,可以发现这两个寄存器为零时,I2caRegs.I2CCNT也不为零。由此可知,I2CCNT不能实时反映发送/接收存储器中数据的数量,不过可以用TXFFST/RXFFST来代替。

图1 主程序流程图,图2为写数据程序流程图,图3为读数据程序流程图,图4为中断函数流程图。

文案大全

实用文档

main主程序各模块初始化FOR循环I2cMsgOut1状态文案大全==I2C_MSGSTAT_SEND_WITHSTOPN?Y调用I2CA_WriteData函数; 令ERR=该函数返回值ERR==I2C_SUCCESS?NYa.更改当前指针为:I2cMsgOut1b.令I2cMsgOut1状态=I2C_MSGSTAT_WRITE_BUSYI2cMsgOut1状态==I2C_MSGSTAT_INACTIVE?NYI2cMsgIn1状态==I2C_MSGSTAT_SEND_NOSTOPN?Y调用I2CA_ReadData函数; 返回值N==I2C_SUCCESS?Ya.更改当前指针为:I2cMsgIn1b.令I2cMsgOut1状态=I2C_MSGSTAT_SEND_NOSTOP_BUSYI2cMsgIn1状态== I2C_MSGSTAT_RESTARTN?Y调用I2CA_ReadData函数; ERR==I2C_SUCCESS?NYa.更改当前指针为:I2cMsgIn1b.令I2cMsgIn1状态=I2C_MSGSTAT_READ_BUSY

实用文档

图1 主程序流程图

I2CA_ReadData函数NSTP==1?I2CA_WriteData函数NY返回:I2C_STP_NOT_READY_ERROR配置从设备地址STP==1?Y返回:I2C_STP_NOT_READY_ERROR配置从设备地址I2cMsgIn1状态==I2C_MSGSTAT_SEND_NOSTOP?YN总线忙否?(BB==1?)Y返回I2C_BUS_BUSY_ERRORN总线忙否?(BB==1?)Y返回I2C_BUS_BUSY_ERRORNI2cMsgIn1状态==I2C_MSGSTAT_RESTART?Y配置I2CCNTN配置I2CCNT;地址数据放入缓存;信息数据放入缓存;配置I2CMDR配置I2CCNT;地址数据放入缓存;配置I2CMDR配置I2CMDR返回I2C_SUCCESS返回I2C_SUCCESS

图2写数据程序流程图 图3读数据程序流程图

中断i2c_int1a_isr获取中断源:IntSource中断源==I2C_SCD_ISRC?YN当前状态==I2C_MSGSTAT_WRITE_BUSY?YNI2cMsgIn1当前状态==I2C_MSGSTAT_SEND_NOSTOP_BUSY?Y令I2cMsgIn1当前状态=I2C_MSGSTAT_SEND_NOSTOPI2cMsgIn1当前状态==I2C_MSGSTAT_READ_BUSY?Y令I2cMsgIn1当前状态=I2C_MSGSTAT_INACTIVE从接收缓存器I2CDRR读取数据数据处理:统计正确次数和错误次数N当前状态==I2C_MSGSTAT_SEND_NOSTOP_BUSY?Y令当前状态=I2C_MSGSTAT_RESTARTNNACK==1?Y设置IIC产生STOP位;清除NACK;N令I2cMsgOut1当前状态=I2C_MSGSTAT_INACTIVE中断源==I2C_ARDY_ISR?YN停止仿真NNPassCount==I2C_NUMBYTES?Y进入fail死循环进入Pass死循环中断返回 文案大全