S3C2440 IIC讀寫(xiě)AT24C02A
S3C2440讀寫(xiě)AT24C02A只需要采用主機(jī)發(fā)送模式和主機(jī)接收模式即可,手冊(cè)上提供有這兩個(gè)模式的流程圖,可以參考。
AT24C02A有幾點(diǎn)需要注意的:
1.AT24C02A連續(xù)讀多個(gè)字節(jié)時(shí)最后一個(gè)字節(jié)不用產(chǎn)生應(yīng)答信號(hào)。
2.讀的時(shí)候要先用寫(xiě)的方式寫(xiě)入硬件地址寫(xiě)方式和數(shù)據(jù)地址,此時(shí)不用發(fā)送STOP信號(hào),接著繼續(xù)寫(xiě)入硬件地址讀方式,然后開(kāi)始讀數(shù)據(jù)。
3.其寫(xiě)的方式只有單字節(jié)寫(xiě)或者頁(yè)寫(xiě)兩種,頁(yè)寫(xiě)時(shí)每次只能寫(xiě)一頁(yè)不能超過(guò),AT24C02A的一頁(yè)大小是8字節(jié)。
以下是程序:
#include "uart.h"
#include "2440addr.h"
#include "iic_lhg.h"
#define U32 unsigned int
#define devaddr 0xa0
//設(shè)置IIC時(shí)鐘,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
}
void __irq iicINT(void) //中斷函數(shù)
{
Uart_Printf("waring:now in iicINT function!!!");
rSRCPND |= 0x1<<27;//清除IIC中斷標(biāo)志位
rINTPND |= 0x1<<27;
}
void wait_ack(void)//等待傳輸完成
{
while (!(rSRCPND&0x1<<27));
rSRCPND |= 0x1<<27;//清除IIC中斷標(biāo)志位
rINTPND |= 0x1<<27;
}
void write_at24c02a_byte(char addr,char da)//隨機(jī)寫(xiě)
{
rIICDS = devaddr;//寫(xiě)硬件地址到IICDS
rIICCON&=~0x10;//清標(biāo)志位
delay_iic(2,5);//延時(shí)一小會(huì)
rIICSTAT = 0xf0;
wait_ack();//等待傳輸完成
rIICDS = addr;//寫(xiě)傳輸?shù)刂?br/>
rIICCON&=~0x10;//清標(biāo)志位
wait_ack();//等待傳輸完成
rIICDS = da;
rIICCON&=~0x10;//清標(biāo)志位
wait_ack();//等待傳輸完成
rIICSTAT = 0xd0;
delay_iic(20,10);//延時(shí)一會(huì)等待停止
rIICCON = rIICCON_init;
}
void write_at24c02a(char addr,char da[],int n)//寫(xiě)函數(shù)地址、要寫(xiě)的數(shù)據(jù)、數(shù)據(jù)的個(gè)數(shù)
{
int i;
i=0;
if (n<1)
return;
for (i=0;i
write_at24c02a_byte(addr+i,da[i]);
}
}
void read_at24c02a(char addr,char da[],int n)//讀數(shù)據(jù)addr地址,*da為指向讀出的數(shù)組,n為個(gè)數(shù)
{
int i;
i=0;
if (n<1)
return;
////////////////////
rIICDS = devaddr;//寫(xiě)硬件地址到IICDS
rIICCON&=~0x10;//清標(biāo)志位
rIICSTAT = 0xf0;
wait_ack();//等待傳輸完成
rIICDS = addr;//寫(xiě)傳輸?shù)刂?br/> rIICCON&=~0x10;//清標(biāo)志位
wait_ack();//等待傳輸完成
/////////////////////
rIICDS = devaddr;//寫(xiě)硬件地址到IICDS
rIICCON&=~0x10;//清標(biāo)志位
rIICSTAT = 0xb0;
wait_ack();//等待傳輸完成
da[i] = rIICDS;
rIICCON&=~0x10;//清標(biāo)志位
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);//延時(shí)一會(huì)等待停止
rIICCON = rIICCON_init;
return;//寫(xiě)入結(jié)束
}
if (i==n-1)
rIICCON &= ~0x80; //讀最后一個(gè)字節(jié)時(shí)不再響應(yīng)
da[i] = rIICDS;
rIICCON&=~0x10;//清標(biāo)志位
wait_ack();//等待傳輸完成
i++;
}
}
void iic_lhg_init(void)//初始化
{
pISR_IIC = (U32)iicINT;
rGPECON &= ~((U32)0xf<<28);//設(shè)置GPIO為IIC功能
rGPECON |= (U32)0xa<<28;
rIICCON = rIICCON_init;//設(shè)置IIC時(shí)鐘,PCLK=50M,IICCLK=50M/16,Tx CLOCK=IICCLK/11=284.09KHZ
//關(guān)于中斷
rINTMOD &= ~((U32)0x1<<27);
rSRCPND |= (U32)0x1<<27;//清除IIC中斷標(biāo)志位
rINTPND |= (U32)0x1<<27;
rINTMSK |= (0x1<<27); //禁止中斷,現(xiàn)在不使用中斷
}