实验前须知:
I. IIC中断发生的三种情况:
1>当发出地址信息或接收到一个从机地址并且吻合时
2>当总线仲裁失败时
3>当发送或接收完一个字节的数据(包括响应位)时
II.启动或恢复IIC传输的两种方法:
1>当IICCON[4]即中断状态为0时,通过写IICSTAT寄存器启动IIC操作。
2>
当IICCON[4
]即中断状态为1时
,表示IIC操作被暂停。在这期间设置好其他寄存器之后,向IICCON[4]写入0即可恢复IIC操作。
具体操作可参考s3c2440芯片手册的流程。在这里仅分析IIC主机发送和IIC主机读取。
实验的目的:
从型号为AT24C02C的EEPROM中写数据,然后去读写入的数据。s3c2440中的IIC主控制器作为IIC主设备。
实验的源程序:
at24cxx.rar
实验的问题总结:
I. 我们重点分析一下IIC.C程序,程序如下:
/*
* FILE: i2c.c
* 用于主机发送/接收
*/
#include
#include "s3c24xx.h"
#include "i2c.h"
void Delay(int time);
#define WRDATA (1)
#define RDDATA (2)
typedef struct tI2C {
unsigned char *pData; /* 数据缓冲区 */
volatile int DataCount; /* 等待传输的数据长度 */
volatile int Status; /* 状态 */
volatile int Mode; /* 模式:读/写 */
volatile int Pt; /* pData中待传输数据的位置 */
}tS3C24xx_I2C, *ptS3C24xx_I2C;
static tS3C24xx_I2C g_tS3C24xx_I2C;
/*
* I2C初始化
*/
void i2c_init(void)
{
GPEUP |= 0xc000; // 禁止内部上拉
GPECON |= 0xa0000000; // 选择引脚功能:GPE15:IICSDA, GPE14:IICSCL
INTMSK &= ~(BIT_IIC);
/* bit[7] = 1, 使能ACK
* bit[6] = 0, IICCLK = PCLK/16
* bit[5] = 1, 使能中断
* bit[3:0] = 0xf, Tx clock = IICCLK/16
* PCLK = 50MHz, IICCLK = 3.125MHz, Tx Clock = 0.195MHz
*/
IICCON = (1<<7) | (0<<6) | (1<<5) | (0xf); // 0xaf
IICADD = 0x10; // S3C24xx slave address = [7:1]
IICSTAT = 0x10; // I2C串行输出使能(Rx/Tx)
}
/*
* 主机发送
* slvAddr : 从机地址,buf : 数据存放的缓冲区,len : 数据长度
*/
void i2c_write(unsigned int slvAddr, unsigned char *buf, int len)
{
g_tS3C24xx_I2C.Mode = WRDATA; // 写操作
g_tS3C24xx_I2C.Pt = 0; // 索引值初始为0
g_tS3C24xx_I2C.pData = buf; // 保存缓冲区地址
g_tS3C24xx_I2C.DataCount = len; // 传输长度
IICDS = slvAddr;
IICSTAT = 0xf0; // 主机发送,启动
/* 等待直至数据传输完毕 */
while (g_tS3C24xx_I2C.DataCount != -1);
}
/*
* 主机接收
* slvAddr : 从机地址,buf : 数据存放的缓冲区,len : 数据长度
*/
void i2c_read(unsigned int slvAddr, unsigned char *buf, int len)
{
g_tS3C24xx_I2C.Mode = RDDATA; // 读操作
g_tS3C24xx_I2C.Pt = -1; // 索引值初始化为-1,表示第1个中断时不接收数据(地址中断)
g_tS3C24xx_I2C.pData = buf; // 保存缓冲区地址
g_tS3C24xx_I2C.DataCount = len; // 传输长度
IICDS = slvAddr;
IICSTAT = 0xb0; // 主机接收,启动
/* 等待直至数据传输完毕 */
while (g_tS3C24xx_I2C.DataCount != 0);
}
/*
* I2C中断服务程序
* 根据剩余的数据长度选择继续传输或者结束
*/
void I2CIntHandle(void)
{
unsigned int iicSt,i;
// 清中断
SRCPND = BIT_IIC;
INTPND = BIT_IIC;
iicSt = IICSTAT;
if(iicSt & 0x8){ printf("Bus arbitration failed\n\r"); }
switch (g_tS3C24xx_I2C.Mode)
{
case WRDATA:
{
if((g_tS3C24xx_I2C.DataCount--) == 0)
{
// 下面两行用来恢复I2C操作,发出P信号
IICSTAT = 0xd0;
IICCON = 0xaf;
Delay(10000); // 等待一段时间以便P信号已经发出
break;
}
IICDS = g_tS3C24xx_I2C.pData[g_tS3C24xx_I2C.Pt++];
// 将数据写入IICDS后,需要一段时间才能出现在SDA线上
for (i = 0; i < 10; i++);
IICCON = 0xaf; // 恢复I2C传输
break;
}
case RDDATA:
{
if (g_tS3C24xx_I2C.Pt == -1)
{
// 这次中断是发送I2C设备地址后发生的,没有数据
// 只接收一个数据时,不要发出ACK信号
g_tS3C24xx_I2C.Pt = 0;
if(g_tS3C24xx_I2C.DataCount == 1)
IICCON = 0x2f; // 恢复I2C传输,开始接收数据,接收到数据时不发出ACK
else
IICCON = 0xaf; // 恢复I2C传输,开始接收数据
break;
}
g_tS3C24xx_I2C.pData[g_tS3C24xx_I2C.Pt++] = IICDS;
g_tS3C24xx_I2C.DataCount--;
if (g_tS3C24xx_I2C.DataCount == 0)
{
// 下面两行恢复I2C操作,发出P信号
IICSTAT = 0x90;
IICCON = 0xaf;
Delay(10000); // 等待一段时间以便P信号已经发出
break;
}
else
{
// 接收最后一个数据时,不要发出ACK信号
if(g_tS3C24xx_I2C.DataCount == 1)
IICCON = 0x2f; // 恢复I2C传输,接收到下一数据时无ACK
else
IICCON = 0xaf; // 恢复I2C传输,接收到下一数据时发出ACK
}
break;
}
default:
break;
}
}
/*
* 延时函数
*/
void Delay(int time)
{
for (; time > 0; time--);
}
/****************************************************************************
代码分析
关于IIC的初始化,在这里就不赘述了,我们先进入main.c中看一下,我们会有一个at24cxx_write()函数的调用,
其实质是调用了i2c_write()函数。接着我们进入到i2c_write()函数去看看。在i2c_write()函数中,slvAddr
表示要发送的设备地址,当代码执行IICSTAT = 0xf0这条语句之后,就会发送S信号及设备地址到从机上,
待从机应答之后,从机就会做出一个ACK信号给主机,通过第一张流程图我们可以看到,在响应之后,
就会进入到中断处理函数里边。接着我们进到I2CIntHandle()里边看一下。由于我们的Mode是WRDATA,
所以会进入到case WRDATA里边。在if((g_tS3C24xx_I2C.DataCount--) == 0)这条语句中,
由于我们的g_tS3C24xx_I2C.DataCount的初值是为2的。所以这条if语句不会执行。接下来我们看到:
在IICDS = g_tS3C24xx_I2C.pData[g_tS3C24xx_I2C.Pt++]这条语句中,由于g_tS3C24xx_I2C.pData[0]和g_tS3C24xx_I2C.pData[1]中
存放的数据分别为要写入数据的存储地址和写入的数据,所以第一次在执行这条语句时,是将要写入数据的存储地址赋给IICDS寄存器。
接下来是IICCON = 0xaf这条语句,在执行这条语句之后,IIC传输恢复。此时从机接收到要写入数据的存储地址之后,做出ACK响应。
那么又再一次进入中断。往后分析就大体类似,在此就不赘述了。i2c_read()函数的分析过程相似。
*****************************************************************************/
II. 在IICDS = g_tS3C24xx_I2C.pData[g_tS3C24xx_I2C.Pt++]中,
g_tS3C24xx_I2C.pData[]的长度len是不包括设备地址的。
它表示的是数据长度,在AT24XX中,就是包括word address 和 要传输的数据。你可以查看一下AT24XX芯片手册。
word address 是指AT24XX内部将要写数据的存储地址,而不是AT24XX的设备地址。它其实也是包含在数据那一块。这是根据AT24XX来的。在发送完设备地址之后,需要再发送你要写入数据的存储地址。最后再发送你想写入的数据。
III. 思考:AT24XX中,如何用8位寻遍其所有的地址?
这在设备地址中有用到关于页的选择,所以就达到了8位寻遍所有的地址的要求。
IV. 思考:读数据的时候,是从从机的什么地址开始读的呢?
参考一下AT24XX芯片手册,本程序中使用的是当前地址读,即在你发送完设备地址之后,从设备就传输当前地址的数据给主设备。
关注微信公众号获取更多资讯