基于mini2440的IIC读写(裸机)

mini2440开发板提供的测试代码过于复杂,让人很难理解,而且有些错误,如GPE14-15不能设置上拉电阻,可是代码里却设置了,虽然无关紧要。为了方便学习,我在闲暇之时我研究了一下。IIC的原理是比较简单的,可是在实际编程中却遇到很多困难。不知道哪里出了错误,有些逻辑上认为正确的,但在硬件上却不能实现。如下两行代码:

 

rGPECON &= ~(0x0f<<28);
rGPECON |= (0xa0<<28);


是为设置GPE14、GPE15位SCL和SDA的,在加上第一句代码后烧写程序就运行失败了。现在还不知道为什么加上第一行代码后就失败了,求高手解释。有些问题不实践真的不知道什么原因,所谓实践出真知嘛。希望对有兴趣的朋友有所帮助。

 

我们来看程序编写过程:

s3c2440的IIC七位地址时的数据接收和发送格式:

基于mini2440的IIC读写(裸机)_第1张图片

基于mini2440的IIC读写(裸机)_第2张图片

灰色的是从设备发送的。


IIC的具体时序:

基于mini2440的IIC读写(裸机)_第3张图片

由此可知SCL为高位时SDA从高位跃迁到低位时表示开始、SCL为高位时SDA从低位跃迁到高位时表示结束。我们可以设置IICSTAT寄存器来发送开始或结束信号。在第九个时钟时主设备或从设备拉低SDA,表示应答ACK。应答后产出中断,此时可以将数据写入IICDS或读出,然后清除中断标志位IICCON[4]恢复操作。具体的操作参考芯片手册或源代码。


mini2440的IIC接口连接一个EEPROM即AT24C08A,我选择页写和连续读来测试IIC接口。具体时序如下:

基于mini2440的IIC读写(裸机)_第4张图片

基于mini2440的IIC读写(裸机)_第5张图片

其中页写的时许很明确。但是连读读的时序前面不太明确,在数据手册中我们可以得知连续读是当前地址读和随机读的一种,所以前面的时序我采取随机读时许,即组合起来。要是还不太清楚请看源代码。随机读时序如下:

基于mini2440的IIC读写(裸机)_第6张图片

AT24C08A接法在硬件电路图中可知如下:

基于mini2440的IIC读写(裸机)_第7张图片

查数据手册可以其地址为0xa0或0xa1。最低位为读写标志。在s3c2440中根据IIC模式自动选择,所以地址我们只需填写0xa0即可。

值得注意的是EEPROM读写速度不是很快,所以每次读写一个字节都要加一定延时,这点十分关键。这往往是程序读写失败的原因。当然我们也需要设置好SCL的频率。在IICCON里设置这里不多说。

在接收模式下最后一个字节数据不发送ACK这点也需要注意。

测试代码中我们采用将数据写入EEPROM中然后读取出来输出到串口来检验,程序可以采用中断或轮询,代码如下:

mian.c部分

 

#define  GLOBAL_CLK 1

#include "def.h"
#include "option.h"
#include "2440addr.h"
#include "profile.h"
#include "mmu.h"

typedef unsigned char uchar;
typedef unsigned int uint;

extern void AT24C08_wirte(uchar waddr, uchar *dat,  int num);
extern void AT24C08_read(uchar waddr, uchar *rev,  int num);
extern void IIC_init(void);
extern void delay(int time);

uchar dat[]={"0123456789abcdef"};
uchar rev[50]={0};

void Main(void)
{

	rGPBCON=(1<<0);//关闭蜂鸣器
    rGPBDAT=0x00;
	
	IIC_init();

	AT24C08_wirte(0x00, dat, 16);
	Uart_Printf("\n\n");
	AT24C08_read(0x00, rev, 10);
	
	Uart_Printf("\n%s\n",rev);
}


IIC.c部分

 

 

#include "def.h"
#include "option.h"
#include "2440addr.h"
#include "profile.h"
#include "mmu.h"

typedef unsigned char uchar;
typedef unsigned int uint;
void __irq IIC_INT(void);

int flag=1;//中断标志

void delay(int time)
{
	int i,j,k;
	for(i=0;i<100;++i)
		for(k=0;k<100;++k)
			for(j=0;j<time;j++);
}

//IIC初始化
void IIC_init(void)
{
	//设置GPE15、GPE14为SDA、SCL
	rGPECON |= (0xa0<<28);
	
	//允许ACK、允许中断、发送频率Khz
	rINTMSK &= ~(1<<27);
	rIICCON |= (1<<7 | 0<<6 | 1<<5 | 0xf);
	rIICSTAT |= 1<<4;//IICDS可写
	
	MMU_Init();//映射地址
	pISR_IIC = (unsigned)IIC_INT;//中断入口
}

//AT24C08页写
void AT24C08_wirte(uchar waddr, uchar *dat, int num)
{
	int i=0;

	rIICDS = 0xa0;//AT24C08地址
	rIICSTAT =0xf0;//发送模式、发送开始信号
	while(flag == 1)
		delay(100);//等待中断
	flag = 1;
	
	rIICDS = waddr;//起始地址
	rIICCON = 0xaf;//清除中断标志位
	while(flag == 1)
		delay(100);//等待中断
	flag = 1;
		
	for(i=0;i<num;++i){
		rIICDS = *(dat + i);
		rIICCON = 0xaf;//清除中断标志位
		while(flag == 1)
			delay(100);//等待中断
		flag = 1;
	}
	
	rIICSTAT =0xd0;//发送停止位
	rIICCON = 0xaf;//清除中断标志位
	delay(100);
}

//AT24C08连续读		
void AT24C08_read(uchar waddr, uchar *rev, int num)
{
	int i=0;
	
	rIICDS = 0xa0;
	rIICSTAT =0xf0;//发送从机地址
	while(flag == 1)
		delay(100);//等待中断
	flag = 1;
	
	rIICDS = waddr;//初始地址
	rIICCON = 0xaf;//清除中断标志位
	while(flag == 1)
		delay(100);//等待中断
	flag = 1;
	
	rIICDS = 0xa0;
	rIICSTAT =0xb0;//接收模式
	rIICCON = 0xaf;//清除中断标志位
	while(flag == 1)
		delay(100);//等待中断
	flag = 1;
	
	for(i=0;i<num;++i){
		if(i == (num-1)) rIICCON = 0x2f;
		else rIICCON = 0xaf;
		while(flag == 1)
			delay(100);//等待中断
		flag = 1;
		*(rev + i) = rIICDS;
		delay(100);
	}
	
	rIICSTAT =0x90;//发送停止位
	rIICCON = 0xaf;//清除中断标志位
	delay(100);
}

//中断服务函数
void __irq IIC_INT(void)
{
	flag = 0;
	rSRCPND |= 1<<27;//先清除SRCPND
	rINTPND |= 1<<27;
}


/*
#include "2440addr.h"

typedef unsigned char uchar;
typedef unsigned int uint;

void delay(int time)
{
	int i,j,k;
	for(i=0;i<100;++i)
		for(k=0;k<100;++k)
			for(j=0;j<time;j++);
}

//IIC初始化
void IIC_init(void)
{
	//设置GPE15、GPE14为SDA、SCL
	rGPECON |=  (0xa0<<28);
	//允许ACK、允许中断、发送频率Khz
	rIICCON |= (1<<7 | 0<<6 | 1<<5 | 0xf);
	rIICSTAT |= (1<<4);//IICDS可写
}

//AT24C08页写
void AT24C08_wirte(uchar waddr, uchar *dat, int num)
{
	int i=0;

	rIICDS = 0xa0;//AT24C08地址
	rIICSTAT =0xf0;//发送模式、发送开始信号
	while(!(rIICCON & 0x10))
		delay(100);//等待中断

	rIICDS = waddr;//起始地址
	rIICCON &= ~(1<<4);//清除中断标志位
	while(!(rIICCON & 0x10))
		delay(100);//等待中断
	
	for(i=0;i<num;++i){
		rIICDS = *(dat + i);
		rIICCON &= ~(1<<4);//清除中断标志位
		while(!(rIICCON & 0x10))
			delay(100);//等待中断
	}
	
	rIICSTAT =0xd0;//发送停止位
	rIICCON &= ~(1<<4);//清除中断标志位
	delay(100);
}

//AT24C08连续读		
void AT24C08_read(uchar waddr, uchar *rev, int num)
{
	int i=0;

	rIICDS = 0xa0;
	rIICSTAT =0xf0;//发送EEPROM地址
	while(!(rIICCON & 0x10))
		delay(100);//等待中断
	
	rIICDS = waddr;//初始地址
	rIICCON &= ~(1<<4);//清除中断标志位
	while(!(rIICCON & 0x10))
		delay(100);//等待中断
		
	rIICDS = 0xa0;
	rIICSTAT =0xb0;//接收模式
	rIICCON &= ~(1<<4);//清除中断标志位
	while(!(rIICCON & 0x10))
			delay(100);//等待中断
	
	for(i=0;i<num;++i){
		if(i == (num-1)) rIICCON &= ~(1<<7 | 1<<4);//不发送ACK
		else rIICCON &= ~(1<<4);//清除中断标志位
		while(!(rIICCON & 0x10))
			delay(100);//等待中断
		*(rev+i) = rIICDS;
		delay(100);
	}

	rIICSTAT =0x90;//发送停止位
	rIICCON &= ~(1<<4);//清除中断标志位
	delay(100);
}
*/


这里包含了中断和轮询方式,其中轮询方式被我注释了。

 

我们通过代码在结合芯片手册可以直观的了解IIC接口的读写。代码中有很多注释方便大家阅读。

程序运行结果:


将0123456789abcdef写入EEPROM,然后读出0123456789。


源代码地址:http://download.csdn.net/detail/a16839678/6010041

 

你可能感兴趣的:(ini)