日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當(dāng)前位置:首頁 > 單片機 > 單片機
[導(dǎo)讀]/******************************************************說 明:S3C2440 I2C實現(xiàn)*****************************************************/1:I2C原理 總線的構(gòu)成及信號類型 I2C總線是由數(shù)據(jù)線SDA和時鐘SCL構(gòu)成的

/*****************************************************

*說 明:S3C2440 I2C實現(xiàn)

*****************************************************/


1:I2C原理

總線的構(gòu)成及信號類型 I2C總線是由數(shù)據(jù)線SDA和時鐘SCL構(gòu)成的串行總線,可發(fā)送和接收數(shù)據(jù)。在CPU與被控IC之間、IC與IC之間進行雙向傳送,最高傳送速率100kbps。各種被控制電路均并聯(lián)在這條總線上,但就像電話機一樣只有撥通各自的號碼才能工作,所以每個電路和模塊都有唯一的地址,在信息的傳輸過程中,I2C總線上并接的每一模塊電路既是主控器(或被控器),又是發(fā)送器(或接收器),這取決于它所要完成的功能。CPU發(fā)出的控制信號分為地址碼和控制量兩部分,地址碼用來選址,即接通需要控制的電路,確定控制的種類;控制量決定該調(diào)整的類別(如對比度、亮度等)及需要調(diào)整的量。這樣,各控制電路雖然掛在同一條總線上,卻彼此獨立,互不相關(guān)。 I2C總線在傳送數(shù)據(jù)過程中共有三種類型信號, 它們分別是:開始信號、結(jié)束信號和應(yīng)答信號。 開始信號:SCL為高電平時,SDA由高電平向低電平跳變,開始傳送數(shù)據(jù)。 結(jié)束信號:SCL為高電平時,SDA由低電平向高電平跳變,結(jié)束傳送數(shù)據(jù)。 應(yīng)答信號:接收數(shù)據(jù)的從控器在接收到8bit數(shù)據(jù)后,向發(fā)送數(shù)據(jù)的主控器發(fā)出特定的低電平脈沖,表示已收到數(shù)據(jù)。CPU向從控器發(fā)出一個信號后,等待從控器發(fā)出一個應(yīng)答信號,CPU接收到應(yīng)答信號后,根據(jù)實際情況作出是否繼續(xù)傳遞信號的判斷。若未收到應(yīng)答信號,判斷為受控單元出現(xiàn)故障。 這些信號中,起始信號是必需的,結(jié)束信號和應(yīng)答信號,都可以不要。


2:I2C實驗代碼

/*
---------------------------------------------------------------
文件名稱:I2C.c
說 明:I2C協(xié)議 讀寫AT24C08
作 者:溫子祺
創(chuàng)建時間:2010-08-17
測試結(jié)果:[OK]
注意事項:

(1)24C02數(shù)據(jù)速率I2C總線的數(shù)據(jù)傳送速率在標準工作方式下為100kbit/s,
在快速方式下,最高傳送速率可達400kbit/s。
(2)當(dāng)前S3C2440各頻率如下:FCLK 405MHz
HCLK 135MHz
PCLK 67.5MHz
(3)當(dāng)前I2C協(xié)議在三星提供的源代碼進行修改,并提升代碼的容錯能力
如I2C進行讀寫時,都有進行超時處理。
---------------------------------------------------------------
*/
#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ù)計數(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; //臨時保存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);


/******************************************************
*文件名稱:I2CWriteByte
*輸 入:unSlaveAddress 從機地址
unWriteAddress 寫地址
pucWriteByte 寫字節(jié)
*輸 出:TRUE/FALSE
*功能說明:I2C 寫單個字節(jié)
*注意事項:
主機發(fā)送起始信號后,發(fā)送一個尋址字節(jié),收到應(yīng)答后緊跟著的就是數(shù)據(jù)傳輸,
數(shù)據(jù)傳輸一般由主機產(chǎn)生的停止位終止。但是,如果主機仍希望在總線上通訊,
它可以產(chǎn)生重復(fù)起始信號和尋址另一個從機,而不是首先產(chǎn)生一個停止信號。
在這種傳輸中,可能有不同的讀/寫格式。
*******************************************************/
static BOOL I2CWriteByte(UINT32 unSlaveAddress,
UINT32 unWriteAddress,
UINT8 *pucWriteByte)
{
BOOL bRt=TRUE;
UINT32 unTimeouts;

g_unI2CCurMode = WRDATA; //當(dāng)前I2C模式:寫
g_unI2CCurDataOffset= 0; //I2C數(shù)據(jù)緩沖區(qū)偏移量為0
g_ucI2CDataBuf[0] = (UINT8)unWriteAddress; //寫地址
g_ucI2CDataBuf[1] = *pucWriteByte; //寫數(shù)據(jù)
g_unI2CCurDataCount = 2; //當(dāng)前數(shù)據(jù)計數(shù)值(即地址+數(shù)據(jù)=2字節(jié))

rIICDS = unSlaveAddress; //0xa0(高四位默認是1010,低四位為xxxx)
rIICSTAT = 0xf0; //主機發(fā)送啟動

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; //主機發(fā)送啟動

rIICCON=g_unIICCONSave; //恢復(fù)I2C運行

unTimeouts=1000;

while(g_unI2CCurStatus==0x100 && unTimeouts--)
{
DelayNus(1);
}

if(!unTimeouts)
{
bRt=FALSE;

goto end;
}


if(!(g_unI2CCurStatus&0x1))
{
break; //接收到應(yīng)答(ACK)信號
}

}

end:
rIICSTAT = 0xd0; //停止主機發(fā)送狀態(tài)Stop MasTx condition
rIICCON = g_unIICCONSave; //恢復(fù)I2C運行
DelayNus(10); //等待直到停止條件是有效的

return bRt;
}

/******************************************************
*文件名稱:I2CReadByte
*輸 入:unSlaveAddress 從機地址
unReadAddress 讀地址
pucReadByte 讀字節(jié)
*輸 出:TRUE/FALSE
*功能說明:I2C 讀單個字節(jié)
*注意事項:
主機發(fā)送完尋址字節(jié)后,主機立即讀取從機中的數(shù)據(jù)。
當(dāng)尋址字節(jié)的"R/W"位為1時,在從機產(chǎn)生應(yīng)答信號后,
主機發(fā)送器變成主機接收器,從機接收器變成從機發(fā)送器。
之后,數(shù)據(jù)由從機發(fā)送,主機接收,每個應(yīng)答由主機產(chǎn)生,
時鐘信號CLK仍由主機產(chǎn)生。若主機要終止本次傳輸,則發(fā)送
一個非應(yīng)答信號,接著主機產(chǎn)生停止信號
*******************************************************/
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; //主機發(fā)送啟動

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; //主機接收啟動
rIICCON = g_unIICCONSave; //恢復(fù)I2C運行

unTimeouts=1000;

while(g_unI2CCurDataCount!=-1 && unTimeouts--)
{
DelayNus(1);
}

if(!unTimeouts)
{
bRt=FALSE;

goto end;
}

*pucReadByte= g_ucI2CDataBuf[1];

end:

return bRt;
}

/******************************************************
*文件名稱:I2CWriteNBytes
*輸 入:unSlaveAddress 從機地址
unWriteAddress 寫地址
pucWriteByte 寫字節(jié)
unNumOfBytes 寫字節(jié)數(shù)
*輸 出:TRUE/FALSE
*功能說明:I2C 寫多個字節(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;
}
/******************************************************
*文件名稱:I2CReadNBytes
*輸 入:unSlaveAddress 從機地址
unReadAddress 讀地址
unNumOfBytes
*輸 出:TRUE/FALSE
*功能說明:I2C 讀多個字節(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ù)

====================================================
*/
/******************************************************
*文件名稱:I2CISR
*輸 入:無
*輸 出:無
*功能說明: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運行
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運行

if((g_unI2CCurDataCount--)==0)
{
rIICSTAT = 0xd0; //Stop MasTx condition
rIICCON = g_unIICCONSave; //恢復(fù)I2C運行
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運行
break;

default:
break;
}

rSRCPND = BIT_IIC; //Clear pending bit
rINTPND = BIT_IIC;
}
/*
====================================================

測試代碼

====================================================
*/
/******************************************************
*文件名稱:I2CTest
*輸 入:無
*輸 出:無
*功能說明:I2C 測試代碼
*******************************************************/
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è)置好發(fā)送時鐘和接收數(shù)據(jù)時鐘
當(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 從機地址設(shè)置
rIICSTAT = 0x10; //I2C總線數(shù)據(jù)輸出使能(Rx/Tx)
rIICLC =(1<<2)|(1); //濾波器使能,SDA數(shù)據(jù)延時輸出

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;

}

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動電源

在工業(yè)自動化蓬勃發(fā)展的當(dāng)下,工業(yè)電機作為核心動力設(shè)備,其驅(qū)動電源的性能直接關(guān)系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅(qū)動電源設(shè)計中至關(guān)重要的兩個環(huán)節(jié),集成化方案的設(shè)計成為提升電機驅(qū)動性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機 驅(qū)動電源

LED 驅(qū)動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設(shè)備的使用壽命。然而,在實際應(yīng)用中,LED 驅(qū)動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設(shè)計、生...

關(guān)鍵字: 驅(qū)動電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計 驅(qū)動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術(shù)之一是電機驅(qū)動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅(qū)動系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動汽車的動力性能和...

關(guān)鍵字: 電動汽車 新能源 驅(qū)動電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動電源 LED

LED通用照明設(shè)計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現(xiàn)在的LED驅(qū)動電源

關(guān)鍵字: LED 驅(qū)動電源 開關(guān)電源

LED驅(qū)動電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動電源
關(guān)閉