S3C2440 IIC读写AT24C02A

来源:互联网 发布:怎样让人帮打理淘宝店 编辑:程序博客网 时间:2024/05/17 04:32

S3C2440读写AT24C02A只需要采用主机发送模式和主机接收模式即可,手册上提供有这两个模式的流程图,可以参考。

 

AT24C02A有几点需要注意的:

1.AT24C02A连续读多个字节时最后一个字节不用产生应答信号。

2.读的时候要先用写的方式写入硬件地址写方式和数据地址,此时不用发送STOP信号,接着继续写入硬件地址读方式,然后开始读数据。

3.其写的方式只有单字节写或者页写两种,页写时每次只能写一页不能超过,AT24C02A的一页大小是8字节。

 

 

以下是程序:

#include "uart.h"
#include "2440addr.h"
#include "iic_lhg.h"

#define U32 unsigned int
#define devaddr 0xa0

//设置IIC时钟,PCLK=50M,IICCLK=50M/16,Tx CLOCK=IICCLK/11=284.09KHZ
#define rIICCON_init (1<<7)|(1<<5)|(0xa)

void delay_iic(int a,int b)
{
 int c;
 if (a==0 || b==0)
 return;
 while (a--)
 for (c=0;c<b;c++);
 
}

void __irq iicINT(void) //中断函数
{
 Uart_Printf("waring:now in iicINT function!!!");
 rSRCPND |= 0x1<<27;//清除IIC中断标志位
 rINTPND |= 0x1<<27;
}

void wait_ack(void)//等待传输完成
{
 while (!(rSRCPND&0x1<<27));
 rSRCPND |= 0x1<<27;//清除IIC中断标志位
 rINTPND |= 0x1<<27;
}

void write_at24c02a_byte(char addr,char da)//随机写
{
 
  rIICDS = devaddr;//写硬件地址到IICDS
  rIICCON&=~0x10;//清标志位
  delay_iic(2,5);//延时一小会
  rIICSTAT = 0xf0;
  wait_ack();//等待传输完成
  
  rIICDS = addr;//写传输地址

  rIICCON&=~0x10;//清标志位
  wait_ack();//等待传输完成
 
  rIICDS = da;
  rIICCON&=~0x10;//清标志位
  wait_ack();//等待传输完成
 
  rIICSTAT = 0xd0; 
  delay_iic(20,10);//延时一会等待停止
  rIICCON = rIICCON_init;
}

void write_at24c02a(char addr,char da[],int n)//写函数地址、要写的数据、数据的个数
{
  int i;
  i=0;
 
  if (n<1)
  return;
 
  for (i=0;i<n;i++)
  {
   write_at24c02a_byte(addr+i,da[i]);
  }
 
}

void read_at24c02a(char addr,char da[],int n)//读数据addr地址,*da为指向读出的数组,n为个数
{
  int i;
  i=0;
  if (n<1)
  return;
 
  ////////////////////
  rIICDS = devaddr;//写硬件地址到IICDS
  rIICCON&=~0x10;//清标志位
  rIICSTAT = 0xf0;
  wait_ack();//等待传输完成
  
  rIICDS = addr;//写传输地址
  rIICCON&=~0x10;//清标志位
  wait_ack();//等待传输完成
  /////////////////////
  rIICDS = devaddr;//写硬件地址到IICDS
  rIICCON&=~0x10;//清标志位 
  rIICSTAT = 0xb0;
  wait_ack();//等待传输完成
 
  da[i] = rIICDS;
  rIICCON&=~0x10;//清标志位
  wait_ack();//等待传输完成
 // Uart_Printf("/r/nLHG:addr=%d c=%c /r/n",da[0],da[0]);
  while(1)
  {
 
 
   if (i>=n)//STOP?
   {
    //stop
    rIICSTAT = 0x90;
   
    delay_iic(20,50);//延时一会等待停止
   
    rIICCON = rIICCON_init;
    return;//写入结束
   }
   if (i==n-1)
    rIICCON &= ~0x80;      //读最后一个字节时不再响应
   da[i] = rIICDS;
   rIICCON&=~0x10;//清标志位
   wait_ack();//等待传输完成
   i++;
  } 
}

void iic_lhg_init(void)//初始化
{
 pISR_IIC = (U32)iicINT;
 
 rGPECON &= ~((U32)0xf<<28);//设置GPIO为IIC功能
 rGPECON |= (U32)0xa<<28;
 rIICCON = rIICCON_init;//设置IIC时钟,PCLK=50M,IICCLK=50M/16,Tx CLOCK=IICCLK/11=284.09KHZ
  //关于中断
 rINTMOD &= ~((U32)0x1<<27);
 rSRCPND |= (U32)0x1<<27;//清除IIC中断标志位
 rINTPND |= (U32)0x1<<27;
 rINTMSK |= (0x1<<27);   //禁止中断,现在不使用中断
}

原创粉丝点击