S3C2440硬件IIC詳解
S3C2440A RISC微處理器可以支持一個(gè)多主控IIC總線串行接口。一條專用串行數(shù)據(jù)線(SDA)和一條專用串行時(shí)鐘線(SCL)傳遞連接到IIC總線的總線主控和外設(shè)之間的信息。SDA和SCL線都為雙向的
圖上可見(jiàn),IIC時(shí)鐘從PCLK產(chǎn)生,并同時(shí)受到IICSTAT控制,IIC數(shù)據(jù)實(shí)際上是由一個(gè)移位寄存器送出,當(dāng)IIC處于從機(jī)狀態(tài)的時(shí)候,有一個(gè)地址比較器來(lái)檢測(cè)IIC總線,使用IIC總線主要有以下寄存器需要設(shè)置
1.設(shè)置相應(yīng)的端口復(fù)用為IIC端口,啟用IIC時(shí)鐘,連接IIC的中斷,自然還需要設(shè)置IIC相應(yīng)的中斷,這些在上一節(jié)已經(jīng)描述過(guò),現(xiàn)在不贅述
2.設(shè)置控制寄存器
設(shè)置應(yīng)答使能IIC時(shí)鐘,IIC發(fā)送時(shí)鐘,IIC中斷等
3.設(shè)置想要發(fā)送的從機(jī)地址
4.修改IIC總線的狀態(tài),并啟用發(fā)送
5.在發(fā)送過(guò)程中,檢測(cè)IIC不同的狀態(tài)作相應(yīng)的操作
具體使用過(guò)程將下列代碼
Iic.c
#include"iic.h"u8flag=0;//中斷標(biāo)志(在中斷子程序里清零,即未中斷flag=1,中斷后flag=0)void__irqI2CInt(void)//中斷子程序{rSRCPND"=BIT_IIC;//清除源掛起rINTPND|=BIT_IIC;//清除中斷掛起flag=0;}voidI2CInit(void){rCLKCON|=(1<<16);//打開(kāi)IIC時(shí)鐘rGPEUP|=0xc000;//關(guān)上拉rGPECON&=~0xf0000000;rGPECON|=0xa0000000;//GPE15:IICSDA,GPE14:IICSCLpISR_IIC=(unsigned)I2CInt;//設(shè)置中斷程序地址rSRCPND|=BIT_IIC;//清除源掛起rINTPND|=BIT_IIC;//清除中斷掛起rINTMOD&=~BIT_IIC;//設(shè)置中斷模式為IRQ模式//使能應(yīng)答,IIC總線時(shí)鐘IICCLK=PCLK/16,使能中斷,發(fā)送時(shí)鐘IICCLK/16rIICCON=(1<<7)|(0<<6)|(1<<5)|(0xf);rIICADD=0x10;//2440從機(jī)地址=[7:1]0位自動(dòng)代表輸入輸出rIICSTAT=0x10;//IIC總線數(shù)據(jù)輸出使能(Rx/Tx)rINTMSK&=~(BIT_IIC);//開(kāi)中斷源}//IIC主機(jī)發(fā)送數(shù)據(jù),voidIICMasterWriteData(u8data){flag=1;rIICDS=data;//從器件地址寫(xiě)入數(shù)據(jù)移位寄存器rIICSTAT=0xf0;//主發(fā)模式,產(chǎn)生起始信號(hào),使能Rx/TxrIICCON&=~0x10;//清除Tx/Rx中斷掛起標(biāo)志while(flag==1)//等待發(fā)送完成DelayMs(1);}//IIC從機(jī)發(fā)送數(shù)據(jù)voidIICSlverWriteData(u8data){flag=1;rIICDS=data;rIICSTAT=0xb0;//主接收模式,使能Rx/TxrIICCON&=~0x10;//清除Tx/Rx中斷掛起標(biāo)志while(flag==1)//等待發(fā)送完成DelayMs(1);}//iic普通數(shù)據(jù)發(fā)送,應(yīng)該在配置好主機(jī)發(fā)送或者從機(jī)發(fā)送之后voidIICWriteData(u8data){flag=1;rIICDS=data;//寫(xiě)入存儲(chǔ)字節(jié)的地址到數(shù)據(jù)移位寄存器rIICCON&=~0x10;//清除Tx/Rx中斷掛起標(biāo)志while(flag==1)//等待發(fā)送完成DelayMs(1);}//iic禁止發(fā)送接收中斷voidIICStopRxTx(void){rIICSTAT=0xd0;//禁止Rx/TxrIICCON=0xaf;//ResumesIICoperation.iic復(fù)位操作DelayMs(1);}//iic禁止接收中斷voidIICStopRx(void){rIICSTAT=0x90;//StopMasRxconditionrIICCON=0xaf;//ResumesIICoperation.DelayMs(1);//Waituntilstopcondtionisineffect.}
Iic.h
#ifndef__IIC_H_#define__IIC_H_#include"2440addr.h"#include"def.h"#include"uart0.h"#include"delay.h"externu8flag;voidI2CInit(void);voidIICMasterWriteData(u8data);voidIICSlverWriteData(u8data);voidIICWriteData(u8data);voidIICStopRxTx(void);voidIICStopRx(void);#endif
At24c02.c
#include"at24c02.h"voidAtWriteByte(u8regAddr,u8data){IICMasterWriteData(AT_DEVICE_ADDR);IICWriteData(regAddr);IICWriteData(data);IICStopRxTx();}u8AtReadByte(u8regAddr){u8temp=0;IICMasterWriteData(AT_DEVICE_ADDR);IICWriteData(regAddr);IICSlverWriteData(AT_DEVICE_ADDR);//注意:讀取下面這個(gè)字節(jié)必須進(jìn)行,因?yàn)樵诎l(fā)送帶有讀命令的從設(shè)備地址后,//AT24C02A會(huì)再返回一個(gè)從設(shè)備地址信息或從設(shè)備內(nèi)存地址信息作為應(yīng)答,所以//一定要把該字節(jié)讀取后拋棄,因?yàn)樗皇俏覀兯x取的信息;也就是一次偽讀取過(guò)程flag=1;temp=rIICDS;rIICCON&=~0x10;//清除Tx/Rx中斷掛起標(biāo)志while(flag==1)DelayMs(1);rIICCON=0x2f;//ResumesIICoperation.禁止應(yīng)答temp=rIICDS;DelayMs(1);IICStopRx();//Waituntilstopcondtionisineffect.returntemp;}//返回1檢測(cè)失敗返回0檢測(cè)成功u8AtCheck(void){u8test=0x88;AtWriteByte(0x01,test);DelayMs(10);test=0x99;test=AtReadByte(0x01);if(test==0x99)return1;elsereturn0;}
At24c02.h
#ifndef__AT24C02_H_#define__AT24C02_H_#include"iic.h"#defineAT_DEVICE_ADDR0xa0voidAtWriteByte(u8regAddr,u8data);u8AtReadByte(u8regAddr);u8AtCheck(void);#endif
注意發(fā)送過(guò)程中附帶起始信號(hào)以及模式的轉(zhuǎn)變,以及中斷接收到信號(hào)之后數(shù)據(jù)的變化