S3C2440 I2C實(shí)現(xiàn)
/*****************************************************
*說(shuō) 明:S3C2440 I2C實(shí)現(xiàn)
*****************************************************/
1:I2C原理
總線的構(gòu)成及信號(hào)類(lèi)型 I2C總線是由數(shù)據(jù)線SDA和時(shí)鐘SCL構(gòu)成的串行總線,可發(fā)送和接收數(shù)據(jù)。在CPU與被控IC之間、IC與IC之間進(jìn)行雙向傳送,最高傳送速率100kbps。各種被控制電路均并聯(lián)在這條總線上,但就像電話機(jī)一樣只有撥通各自的號(hào)碼才能工作,所以每個(gè)電路和模塊都有唯一的地址,在信息的傳輸過(guò)程中,I2C總線上并接的每一模塊電路既是主控器(或被控器),又是發(fā)送器(或接收器),這取決于它所要完成的功能。CPU發(fā)出的控制信號(hào)分為地址碼和控制量?jī)刹糠郑刂反a用來(lái)選址,即接通需要控制的電路,確定控制的種類(lèi);控制量決定該調(diào)整的類(lèi)別(如對(duì)比度、亮度等)及需要調(diào)整的量。這樣,各控制電路雖然掛在同一條總線上,卻彼此獨(dú)立,互不相關(guān)。 I2C總線在傳送數(shù)據(jù)過(guò)程中共有三種類(lèi)型信號(hào), 它們分別是:開(kāi)始信號(hào)、結(jié)束信號(hào)和應(yīng)答信號(hào)。 開(kāi)始信號(hào):SCL為高電平時(shí),SDA由高電平向低電平跳變,開(kāi)始傳送數(shù)據(jù)。 結(jié)束信號(hào):SCL為高電平時(shí),SDA由低電平向高電平跳變,結(jié)束傳送數(shù)據(jù)。 應(yīng)答信號(hào):接收數(shù)據(jù)的從控器在接收到8bit數(shù)據(jù)后,向發(fā)送數(shù)據(jù)的主控器發(fā)出特定的低電平脈沖,表示已收到數(shù)據(jù)。CPU向從控器發(fā)出一個(gè)信號(hào)后,等待從控器發(fā)出一個(gè)應(yīng)答信號(hào),CPU接收到應(yīng)答信號(hào)后,根據(jù)實(shí)際情況作出是否繼續(xù)傳遞信號(hào)的判斷。若未收到應(yīng)答信號(hào),判斷為受控單元出現(xiàn)故障。 這些信號(hào)中,起始信號(hào)是必需的,結(jié)束信號(hào)和應(yīng)答信號(hào),都可以不要。
2:I2C實(shí)驗(yàn)代碼
/*
---------------------------------------------------------------
文件名稱(chēng):I2C.c
說(shuō) 明:I2C協(xié)議 讀寫(xiě)AT24C08
作 者:溫子祺
創(chuàng)建時(shí)間:2010-08-17
測(cè)試結(jié)果:[OK]
注意事項(xiàng):
(1)24C02數(shù)據(jù)速率I2C總線的數(shù)據(jù)傳送速率在標(biāo)準(zhǔn)工作方式下為100kbit/s,
在快速方式下,最高傳送速率可達(dá)400kbit/s。
(2)當(dāng)前S3C2440各頻率如下:FCLK 405MHz
HCLK 135MHz
PCLK 67.5MHz
(3)當(dāng)前I2C協(xié)議在三星提供的源代碼進(jìn)行修改,并提升代碼的容錯(cuò)能力
如I2C進(jìn)行讀寫(xiě)時(shí),都有進(jìn)行超時(shí)處理。
---------------------------------------------------------------
*/
#include "S3C244x.h"
#include "Global.h"
#include "IIC.h"
/*
1:rIICON IIC總線控制寄存器
2:rIICSTAT IIC總線控制狀態(tài)寄存器
3:rIICADD IIC總線地址寄存器
4:rIICDS IIC總線發(fā)送接收數(shù)據(jù)移位寄存器
5:rIICLC IIC總線多主設(shè)備線路控制寄存器
*/
/*
====================================================
I2C基本函數(shù)接口
====================================================
*/
static volatile UINT8 g_ucI2CDataBuf[256]; //I2C發(fā)送數(shù)據(jù)緩沖區(qū)
static volatile UINT32 g_unI2CCurDataCount; //I2C當(dāng)前數(shù)據(jù)計(jì)數(shù)
static volatile UINT32 g_unI2CCurStatus; //I2C當(dāng)前狀態(tài)
static volatile UINT32 g_unI2CCurDataOffset; //I2C當(dāng)前發(fā)送數(shù)據(jù)偏移量
static UINT32 g_unI2CCurMode; //I2C當(dāng)前模式
static UINT32 g_unIICCONSave; //臨時(shí)保存rIICCON寄存器值
static void __irq I2CISR(void) ; //I2C中斷服務(wù)函數(shù)
static BOOL I2CWriteByte(UINT32 unSlaveAddress,UINT32 ucWriteAddress,UINT8 *pucWriteByte);
static BOOL I2CReadByte (UINT32 unSlaveAddress,UINT32 ucReadAddress ,UINT8 *pucReadByte);
/******************************************************
*文件名稱(chēng):I2CWriteByte
*輸 入:unSlaveAddress 從機(jī)地址
unWriteAddress 寫(xiě)地址
pucWriteByte 寫(xiě)字節(jié)
*輸 出:TRUE/FALSE
*功能說(shuō)明:I2C 寫(xiě)單個(gè)字節(jié)
*注意事項(xiàng):
主機(jī)發(fā)送起始信號(hào)后,發(fā)送一個(gè)尋址字節(jié),收到應(yīng)答后緊跟著的就是數(shù)據(jù)傳輸,
數(shù)據(jù)傳輸一般由主機(jī)產(chǎn)生的停止位終止。但是,如果主機(jī)仍希望在總線上通訊,
它可以產(chǎn)生重復(fù)起始信號(hào)和尋址另一個(gè)從機(jī),而不是首先產(chǎn)生一個(gè)停止信號(hào)。
在這種傳輸中,可能有不同的讀/寫(xiě)格式。
*******************************************************/
static BOOL I2CWriteByte(UINT32 unSlaveAddress,
UINT32 unWriteAddress,
UINT8 *pucWriteByte)
{
BOOL bRt=TRUE;
UINT32 unTimeouts;
g_unI2CCurMode = WRDATA; //當(dāng)前I2C模式:寫(xiě)
g_unI2CCurDataOffset= 0; //I2C數(shù)據(jù)緩沖區(qū)偏移量為0
g_ucI2CDataBuf[0] = (UINT8)unWriteAddress; //寫(xiě)地址
g_ucI2CDataBuf[1] = *pucWriteByte; //寫(xiě)數(shù)據(jù)
g_unI2CCurDataCount = 2; //當(dāng)前數(shù)據(jù)計(jì)數(shù)值(即地址+數(shù)據(jù)=2字節(jié))
rIICDS = unSlaveAddress; //0xa0(高四位默認(rèn)是1010,低四位為xxxx)
rIICSTAT = 0xf0; //主機(jī)發(fā)送啟動(dòng)
unTimeouts=1000;
while(g_unI2CCurDataCount!=-1 && unTimeouts--)
{
DelayNus(1);
}
if(!unTimeouts)
{
bRt=FALSE;
goto end;
}
g_unI2CCurMode = POLLACK;
while(1)
{
rIICDS = unSlaveAddress;
g_unI2CCurStatus = 0x100;
rIICSTAT = 0xf0; //主機(jī)發(fā)送啟動(dòng)
rIICCON=g_unIICCONSave; //恢復(fù)I2C運(yùn)行
unTimeouts=1000;
while(g_unI2CCurStatus==0x100 && unTimeouts--)
{
DelayNus(1);
}
if(!unTimeouts)
{
bRt=FALSE;
goto end;
}
if(!(g_unI2CCurStatus&0x1))
{
break; //接收到應(yīng)答(ACK)信號(hào)
}
}
end:
rIICSTAT = 0xd0; //停止主機(jī)發(fā)送狀態(tài)Stop MasTx condition
rIICCON = g_unIICCONSave; //恢復(fù)I2C運(yùn)行
DelayNus(10); //等待直到停止條件是有效的
return bRt;
}
/******************************************************
*文件名稱(chēng):I2CReadByte
*輸 入:unSlaveAddress 從機(jī)地址
unReadAddress 讀地址
pucReadByte 讀字節(jié)
*輸 出:TRUE/FALSE
*功能說(shuō)明:I2C 讀單個(gè)字節(jié)
*注意事項(xiàng):
主機(jī)發(fā)送完尋址字節(jié)后,主機(jī)立即讀取從機(jī)中的數(shù)據(jù)。
當(dāng)尋址字節(jié)的"R/W"位為1時(shí),在從機(jī)產(chǎn)生應(yīng)答信號(hào)后,
主機(jī)發(fā)送器變成主機(jī)接收器,從機(jī)接收器變成從機(jī)發(fā)送器。
之后,數(shù)據(jù)由從機(jī)發(fā)送,主機(jī)接收,每個(gè)應(yīng)答由主機(jī)產(chǎn)生,
時(shí)鐘信號(hào)CLK仍由主機(jī)產(chǎn)生。若主機(jī)要終止本次傳輸,則發(fā)送
一個(gè)非應(yīng)答信號(hào),接著主機(jī)產(chǎn)生停止信號(hào)
*******************************************************/
static BOOL I2CReadByte(UINT32 unSlaveAddress,
UINT32 unReadAddress,
UINT8 *pucReadByte)
{
BOOL bRt=TRUE;
UINT32 unTimeouts;
g_unI2CCurMode = SETRDADDR;
g_unI2CCurDataOffset= 0;
g_ucI2CDataBuf[0] = (UINT8)unReadAddress;
g_unI2CCurDataCount = 1;
rIICDS = unSlaveAddress;
rIICSTAT = 0xf0; //主機(jī)發(fā)送啟動(dòng)
unTimeouts=1000;
while(g_unI2CCurDataCount!=-1 && unTimeouts--)
{
DelayNus(1);
}
if(!unTimeouts)
{
bRt=FALSE;
goto end;
}
g_unI2CCurMode = RDDATA;
g_unI2CCurDataOffset = 0;
g_unI2CCurDataCount = 1;
rIICDS = unSlaveAddress;
rIICSTAT = 0xb0; //主機(jī)接收啟動(dòng)
rIICCON = g_unIICCONSave; //恢復(fù)I2C運(yùn)行
unTimeouts=1000;
while(g_unI2CCurDataCount!=-1 && unTimeouts--)
{
DelayNus(1);
}
if(!unTimeouts)
{
bRt=FALSE;
goto end;
}
*pucReadByte= g_ucI2CDataBuf[1];
end:
return bRt;
}
/******************************************************
*文件名稱(chēng):I2CWriteNBytes
*輸 入:unSlaveAddress 從機(jī)地址
unWriteAddress 寫(xiě)地址
pucWriteByte 寫(xiě)字節(jié)
unNumOfBytes 寫(xiě)字節(jié)數(shù)
*輸 出:TRUE/FALSE
*功能說(shuō)明:I2C 寫(xiě)多個(gè)字節(jié)
*******************************************************/
BOOL I2CWriteNBytes(UINT32 unSlaveAddress,
UINT32 unWriteAddress,
UINT8 *pucWriteBytes,
UINT32 unNumOfBytes)
{
UINT32 unSpareOfBytes=unNumOfBytes;
while(unSpareOfBytes--)
{
if(!I2CWriteByte( unSlaveAddress,
unWriteAddress,
pucWriteBytes))
{
I2CMSG("I2C[ERROR]:fail to write data fail at address %d
success to write %d bytes rn",
unWriteAddress,(unNumOfBytes-unSpareOfBytes));
return FALSE;
}
unWriteAddress++;
pucWriteBytes++;
}
return TRUE;
}
/******************************************************
*文件名稱(chēng):I2CReadNBytes
*輸 入:unSlaveAddress 從機(jī)地址
unReadAddress 讀地址
unNumOfBytes
*輸 出:TRUE/FALSE
*功能說(shuō)明:I2C 讀多個(gè)字節(jié)
*******************************************************/
BOOL I2CReadNBytes(UINT32 unSlaveAddress,
UINT32 unReadAddress,
UINT8 *pucReadByte,
UINT32 unNumOfBytes)
{
UINT32 unSpareOfBytes=unNumOfBytes;
while(unSpareOfBytes--)
{
if(!I2CReadByte( unSlaveAddress,
unReadAddress,
pucReadByte))
{
I2CMSG("I2C[ERROR]:fail to read data fail at address %d
success to read %d bytes rn",
unReadAddress,(unNumOfBytes-unSpareOfBytes));
return FALSE;
}
unReadAddress++;
pucReadByte++;
}
return TRUE;
}
/*
====================================================
中斷服務(wù)函數(shù)
====================================================
*/
/******************************************************
*文件名稱(chēng):I2CISR
*輸 入:無(wú)
*輸 出:無(wú)
*功能說(shuō)明:I2C 中斷服務(wù)函數(shù)
*******************************************************/
void __irq I2CISR(void)
{
UINT32 unI2CStatus;
unI2CStatus = rIICSTAT;
if(unI2CStatus & 0x8){} //When bus arbitration is failed.
if(unI2CStatus & 0x4){} //When a slave address is matched with IICADD
if(unI2CStatus & 0x2){} //When a slave address is 0000000b
if(unI2CStatus & 0x1){} //When ACK isn't received
switch(g_unI2CCurMode)
{
case POLLACK:
g_unI2CCurStatus = unI2CStatus;
break;
case RDDATA:
if((g_unI2CCurDataCount--)==0)
{
g_ucI2CDataBuf[g_unI2CCurDataOffset++] = rIICDS;
rIICSTAT = 0x90; //停止I2C接收狀態(tài)
rIICCON = g_unIICCONSave; //恢復(fù)I2C運(yùn)行
DelayNus(1); //等待直到停止條件是有效的
//The pending bit will not be set after issuing stop condition.
break;
}
g_ucI2CDataBuf[g_unI2CCurDataOffset++] = rIICDS; //The last data has to be read with no ack.
if((g_unI2CCurDataCount)==0)
rIICCON = 0x2f; //Resumes IIC operation with NOACK.
else
rIICCON = g_unIICCONSave; //Resumes IIC operation with ACK
break;
case WRDATA:
rIICDS = g_ucI2CDataBuf[g_unI2CCurDataOffset++]; //g_ucI2CDataBuf[0] has dummy.
DelayNus(10); //for setup time until rising edge of IICSCL
rIICCON = g_unIICCONSave; //恢復(fù)I2C運(yùn)行
if((g_unI2CCurDataCount--)==0)
{
rIICSTAT = 0xd0; //Stop MasTx condition
rIICCON = g_unIICCONSave; //恢復(fù)I2C運(yùn)行
DelayNus(10); //Wait until stop condtion is in effect.
//The pending bit will not be set after issuing stop condition.
}
break;
case SETRDADDR:
if((g_unI2CCurDataCount--)==0)
{
break;
}
//IIC operation is stopped because of IICCON[4]
rIICDS = g_ucI2CDataBuf[g_unI2CCurDataOffset++];
DelayNus(10); //For setup time until rising edge of IICSCL
rIICCON = g_unIICCONSave; //恢復(fù)I2C運(yùn)行
break;
default:
break;
}
rSRCPND = BIT_IIC; //Clear pending bit
rINTPND = BIT_IIC;
}
/*
====================================================
測(cè)試代碼
====================================================
*/
/******************************************************
*文件名稱(chēng):I2CTest
*輸 入:無(wú)
*輸 出:無(wú)
*功能說(shuō)明:I2C 測(cè)試代碼
*******************************************************/
void I2CTest(void)
{
UINT32 i;
UINT8 buf[256];
I2CMSG("nIIC Test(Interrupt) using AT24C02n");
rGPEUP |= 0xc000; //Pull-up disable
rGPECON |= 0xa00000; //GPE15:IICSDA , GPE14:IICSCL
rCLKCON |= 1<<16;
pISR_IIC = (UINT32)I2CISR;
rINTMSK &= ~(BIT_IIC);
/*
IIC時(shí)序太重要了,要認(rèn)真設(shè)置好發(fā)送時(shí)鐘和接收數(shù)據(jù)時(shí)鐘
當(dāng)前PCLK = 405/6 = 67.5MHz
IICCLK=67.5/16= 4.22MHz
Tx Clock = 4.22/11=0.384MHz
*/
g_unIICCONSave=rIICCON = (1<<7) | (0<<6) | (1<<5) | (0xa);
rIICADD = 0x10; //S3C2440 從機(jī)地址設(shè)置
rIICSTAT = 0x10; //I2C總線數(shù)據(jù)輸出使能(Rx/Tx)
rIICLC =(1<<2)|(1); //濾波器使能,SDA數(shù)據(jù)延時(shí)輸出
I2CMSG("Write test data into AT24C02n");
for(i=0;i<256;i++)
{
buf[i]=i;
}
I2CWriteNBytes(0xA0,0,buf,256);
for(i=0;i<256;i++)
buf[i] = 0;
I2CMSG("Read test data from AT24C02n");
I2CReadNBytes(0xA0,0,buf,256);
I2CMSG("Read Data Finishrn");
for(i=0;i<256;i++)
{
I2CMSG("%d ",buf[i]);
}
rINTMSK |= BIT_IIC;
}





