www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當前位置:首頁 > 單片機 > 單片機
[導讀] *-------------------------------------------------------------------------------------------------PIC24F之EEPROM讀寫中斷事件處理函數(shù)要點及說明注意:這是一個通用的I2C/SMBUS通訊中斷處理程序?qū)?

*-------------------------------------------------------------------------------------------------
PIC24FEEPROM讀寫中斷事件處理函數(shù)要點及說明



注意:這是一個通用的I2C/SMBUS通訊中斷處理程序
對于EEPROM來講,從機后面需要跟EEPROM需要讀寫的地址(I2CRegs.RWAddr)
對于SMBUS來說,從機后面需要跟SMBUS需要的命令(I2CRegs.RWAddr改為I2CRegs.CMD即可)
由于PIC24F的I2C不太標準,I2C1STAT被搞得很倒塌?。。∫稽c都沒I2C的"大家閨秀"的樣子~~~
不過它的STOP還能激活中斷確實比LPCARM/AVR好一點點~~~

為什么I2C收發(fā)都用中斷呢???
這主要是為了高低速靈活變化的總線通訊所做,主要是SMBUS總線的通信.


如果為I2cExit()也配上鉤子函數(shù),那么任何錯誤都在手掌中~~~

這個PIC程序雖沒SMBUS的PEC校驗部分,但"異步"還是完美的.
當然也要注意對寫保護硬件管腳的控制時機的把握,原則是關保護的時間最短就更好~~~


主要看到人們編寫MCU程序太死板~~~特別是I2C程序.網(wǎng)上收發(fā)全中斷的很少,可以說幾乎沒有.

隨貼附老外倒塌的非中斷I2C狀態(tài)機讀寫程序i2cEmem.c~~~可以比較經(jīng)典和非典的差異在何處~~~

菜農(nóng)近期將整理出LPCARM和AVR的I2C/SMBUS/TWI/USI收發(fā)全中斷實戰(zhàn)例程供大家“游玩”~~~

如果精通DELPHI程序的人一定會為“事件驅(qū)動”機制而癡迷~~~為什么不在MCU上"聲東擊西"呢???

"有事件才處理"---這才是編程的硬道理~~~輪循的“癡迷等待”最終還是“單相思”~~~
-------------------------------------------------------------------------------------------------*/
#include"i2c.h"

_PERSISTENTvolatileI2CREGSI2CRegs;
_PERSISTENTvolatileI2CBITSI2CBits;

voidI2cInit(void)
{
unsignedinti;
TRIS_WP=PORTOUTMODE;//定義WP為輸出IO
TRIS_SCL1=PORTOUTMODE;//定義SCL為輸出IO
TRIS_SDA1=PORTINPUTMODE;//定義SDA為輸出入IO
ODC_SCL1=1;//OC輸出
ODC_SDA1=1;//OC輸出
WP=1;//寫保護
I2CRegs.MaxCount=0x200;//8KByte
I2CRegs.I2CAddr=0xa0;//器件地址
I2CRegs.RWAddr=0;//EEPROM讀寫地址
I2CRegs.TxCount=0;//發(fā)送數(shù)據(jù)字節(jié)個數(shù)
I2CRegs.RxCount=0;//接收數(shù)據(jù)字節(jié)個數(shù)
for(i=0;i<16;i++)
{
I2CRegs.TxBuffer[i]=0;//發(fā)送緩沖區(qū)清零
}
for(i=0;i<256;i++)
{
I2CRegs.RxBuffer[i]=0;//接收緩沖區(qū)清零
}

I2C1CON=0;
//I2C1CONbits.A10M=0;//7位地址模式
I2C1CONbits.SCLREL=1;
I2C1MSK=0;
I2C1STAT=0;
_MI2C1IF=0;
_SI2C1IF=0;
I2C1BRG=(FCY/(2*I2CBAUD))-1;//波特率計算
/*------------------------------------------------------------------------
定義I2C串口2中斷優(yōu)先級位1111)
-------------------------------------------------------------------------*/
IPC4bits.MI2C1P0=1;
IPC4bits.MI2C1P1=1;
IPC4bits.MI2C1P2=1;

I2C1CONbits.I2CEN=1;//允許I2C功能
_MI2C1IE=1;//允許主設備中斷

//I2cStop();
}

/*------------------------------------------------------------------
EEPROM讀塊函數(shù)(只能在回調(diào)函數(shù)I2CReadCallBack中得到讀出的數(shù)據(jù))
-------------------------------------------------------------------*/
voidI2CReadBuffers(unsignedintE2RomAddr,unsignedintReadSize)
{
if(ReadSize&&(ReadSize<=256))
{
I2CRegs.TxCount=0;
I2CRegs.RxCount=ReadSize;
I2CRegs.RWAddr=E2RomAddr;
I2CRegs.I2CAddr|=1;//0xa1
I2cStart();
}
}

voidI2CReadByte(unsignedintE2RomAddr)
{
I2CRegs.TxCount=0;
I2CRegs.RxCount=1;
I2CRegs.RWAddr=E2RomAddr;
I2CRegs.I2CAddr|=1;//0xa1
I2cStart();
}

/*------------------------------------------------------------------
EEPROM寫塊函數(shù)
-------------------------------------------------------------------*/
voidI2CWriteBuffers(unsignedintE2RomAddr,unsignedintWriteSize)
{
if(WriteSize&&(WriteSize<=16))
{
I2CRegs.TxCount=WriteSize;
I2CRegs.RxCount=0;
I2CRegs.RWAddr=E2RomAddr;
I2CRegs.I2CAddr&=0xfe;//0xa0
I2cStart();
}
}

voidI2CWriteByte(unsignedintE2RomAddr,unsignedcharcData)
{
I2CRegs.TxBuffer[0]=cData;
I2CRegs.TxCount=1;
I2CRegs.RxCount=0;
I2CRegs.RWAddr=E2RomAddr;
I2CRegs.I2CAddr&=0xfe;//0xa0
I2cStart();
}

/*------------------------------------------------------------------
用戶讀回調(diào)函數(shù)
-------------------------------------------------------------------*/
voidI2CReadCallBack(void)
{
if((I2CRegs.RWAddr+I2CRegs.RxCount)<=I2CRegs.MaxCount)
{
//I2CRegs.RWAddr+=I2CRegs.RxCount;
//I2CReadBuffers(I2CRegs.RWAddr,I2CRegs.RxCount);//繼續(xù)讀
}
}

/*------------------------------------------------------------------
用戶寫回調(diào)函數(shù)
-------------------------------------------------------------------*/
voidI2CWriteCallBack(void)
{
if((I2CRegs.RWAddr+I2CRegs.TxCount)<=I2CRegs.MaxCount)
{
//I2CRegs.RWAddr+=I2CRegs.TxCount;
//I2CWriteBuffers(I2CRegs.RWAddr,I2CRegs.TxCount);//繼續(xù)寫
}
}


/*------------------------------------------------------------------
EEPROM讀寫啟動函數(shù)
-------------------------------------------------------------------*/
voidI2cStart(void)
{
/*------------------------------------------------------------------------
//本程序在狀態(tài)I2C_MT_ADDRL_ACK下進行瞬間打開,也可在此打開,不過安全不好
if(I2CRegs.TxCount)//需要寫入字節(jié)
{
WP=0;//不寫保護
}
else
{
WP=1;//寫保護
}
--------------------------------------------------------------------------*/
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=1;
I2CRegs.State=I2C_START;//主機準備發(fā)送啟始位
I2CRegs.Count=0;//發(fā)送數(shù)據(jù)個數(shù)
I2CBits.I2CFlag=0;
I2C1CONbits.SEN=1;//發(fā)送Start信號
}

/*------------------------------------------------------------------
EEPROM讀再啟動函數(shù)
-------------------------------------------------------------------*/
voidI2cReStart(void)
{
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=1;
I2CRegs.State=I2C_REP_START;//主機準備發(fā)送重新啟始位
I2CRegs.Count=0;//發(fā)送數(shù)據(jù)個數(shù)
I2C1CONbits.RSEN=1;//發(fā)送ReStart信號
I2C1CONbits.ACKEN=0;
}

/*------------------------------------------------------------------
EEPROM讀寫正確停止函數(shù)
-------------------------------------------------------------------*/
voidI2cStop(void)
{
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=0;
I2CRegs.State=I2C_SUCCEEDED;//通訊成功
I2C1CONbits.PEN=1;//發(fā)送Stop信號
WP=1;//寫保護
}


/*------------------------------------------------------------------
EEPROM讀寫錯誤退出函數(shù)
-------------------------------------------------------------------*/
voidI2cExit(void)
{
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=0;
I2CRegs.State=I2C_FAILED;
I2C1CONbits.PEN=1;//發(fā)送Stop信號
WP=1;//寫保護
}

/*------------------------------------------------------------------
EEPROM讀寫中斷事件處理函數(shù)(說明見文件頭部)
-------------------------------------------------------------------*/
voidI2CExec(void)
{
if(I2C1STATbits.S)//收到Start過信號
{
switch(I2CRegs.State)
{
caseI2C_START://收到Start信號
I2C1TRN=I2CRegs.I2CAddr&0xfe;//發(fā)送器件寫地址(通知從機只能聽)
I2CRegs.State=I2C_MT_SLA_ACK;//下次應該接收器件寫地址應答信號
break;
caseI2C_MT_SLA_ACK://收到器件寫地址應答信號
if(!I2C1STATbits.ACKSTAT)//收到Ack信號
{
if(I2CRegs.MaxCount>0x100)//EEPROM容量超過256個字節(jié),EEPROM地址需要兩次發(fā)送
{
I2C1TRN=I2CRegs.RWAddr>>8;//發(fā)送EEPROM寫高8位地址
I2CRegs.State=I2C_MT_ADDRH_ACK;//下次應該接收EEPROM寫高8位地址應答信號
}
else//小容量只需一次發(fā)送!!!
{
I2C1TRN=I2CRegs.RWAddr;//發(fā)送EEPROM寫低8位地址
I2CRegs.State=I2C_MT_ADDRL_ACK;//下次應該接收EEPROM寫低8位地址應答信號
I2CRegs.Count=0;//清空發(fā)送緩沖計數(shù)器
}
}
else//收到NAck信號
{
I2cExit();//錯誤的ACK信號
}
break;
caseI2C_MT_ADDRH_ACK://收到EEPROM寫高8位地址應答信號
if(!I2C1STATbits.ACKSTAT)//收到Ack信號
{
I2C1TRN=I2CRegs.RWAddr&0xff;//發(fā)送EEPROM寫低8位地址
I2CRegs.State=I2C_MT_ADDRL_ACK;//下次應該接收EEPROM寫低8位地址應答信號
I2CRegs.Count=0;//清空發(fā)送緩沖計數(shù)器
}
else//收到NAck信號
{
I2cExit();//錯誤的ACK信號
}
break;
caseI2C_MT_ADDRL_ACK://收到EEPROM寫高低8位地址應答信號
if(I2CRegs.TxCount)//寫保護只在寫入期間不保護,增加了對誤寫入的安全防護能力!!!
{
WP=0;//不寫保護
}
caseI2C_MT_DATA_ACK://收到應答信號
if(!I2C1STATbits.ACKSTAT)//收到Ack信號
{
if(I2CRegs.Count{
I2C1TRN=I2CRegs.TxBuffer[I2CRegs.Count++];//繼續(xù)發(fā)送數(shù)據(jù)
}
elseif(I2CRegs.Count==I2CRegs.TxCount)//緩沖區(qū)已空
{
if(I2CRegs.I2CAddr&1)//應該開始接收數(shù)據(jù)
{
I2cReStart();//發(fā)送重復位命令
}
else//只寫退出
{
I2cStop();//正常發(fā)送結束
}
}
else//干擾出錯
{
I2cExit();//錯誤
}
}
else//收到NAck信號(可能被寫保護)
{
I2cExit();//錯誤的ACK信號
}
break;
caseI2C_REP_START://收到ReStart信號
I2C1TRN=I2CRegs.I2CAddr|I2C_READ;//發(fā)送器件讀地址(通知從機可以說話)
I2CRegs.State=I2C_MR_SLA_ACK;//下次應該接收器件寫讀地址應答信號
break;
caseI2C_MR_SLA_ACK://收到器件讀地址應答信號
if(!I2C1STATbits.ACKSTAT)//收到Ack信號
{
I2C1CONbits.RCEN=1;//開始接收數(shù)據(jù)
I2CRegs.State=I2C_MR_DATA;//下次應該收接收數(shù)據(jù)
}
else//收到NAck信號
{
I2cExit();//錯誤的ACK信號
}
break;
caseI2C_MR_DATA://收到接收數(shù)據(jù)
if(I2CRegs.Count{
//I2C1STATbits.I2COV=0;
I2CRegs.RxBuffer[I2CRegs.Count++]=I2C1RCV;
if(I2CRegs.Count{
I2C1CONbits.ACKDT=0;//應答子機
I2CRegs.State=I2C_MR_DATA_EN;//下次應該收到器件允許繼續(xù)讀信號
}
else
{
I2C1CONbits.ACKDT=1;//非應答子機
I2CRegs.State=I2C_MR_DATA_STOP;//下次應該收到退出信號
}
I2C1CONbits.ACKEN=1;//向從機發(fā)送(非)應答信號
}
else//正確的狀態(tài)已分支到I2C_MR_DATA_STOP
{
I2cExit();//錯誤
}
break;
caseI2C_MR_DATA_EN://收到器件允許繼續(xù)讀信號
I2C1CONbits.RCEN=1;//開始接收數(shù)據(jù)
I2CRegs.State=I2C_MR_DATA;//下次應該繼續(xù)接收數(shù)據(jù)
break;
caseI2C_MR_DATA_STOP://收到器件退出信號
I2cStop();//正常接收結束
break;
default://其他不可預料的錯誤
I2cExit();//錯誤
}
}
elseif(I2C1STATbits.P)//收到Stop信號
{
if(I2CRegs.State==I2C_SUCCEEDED)//成功,回調(diào)
{
if(I2CRegs.I2CAddr&1)//讀
{
I2CBits.ReadFlag=1;//激活用戶讀回調(diào)函數(shù)I2CReadCallBack()
}
else//寫
{
I2CBits.WriteFlag=1;//激活用戶寫回調(diào)函數(shù)I2CWriteCallBack()
}
}
}
else//無法確認的復雜錯誤
{
I2cExit();//錯誤出錯退出
}
}

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

北京——2024年10月29日 亞馬遜云科技日前宣布,通過與光環(huán)新網(wǎng)與西云數(shù)據(jù)的緊密合作,在亞馬遜云科技(北京)區(qū)域和(寧夏)區(qū)域推出Amazon Lambda SnapStart功能,將基于Java Lambda函數(shù)的...

關鍵字: 函數(shù) 數(shù)據(jù)處理

北京2024年10月29日 /美通社/ -- 亞馬遜云科技日前宣布,通過與光環(huán)新網(wǎng)與西云數(shù)據(jù)的緊密合作,在亞馬遜云科技(北京)區(qū)域和(寧夏)區(qū)域推出Amazon Lambda SnapStart功能,將基于Java La...

關鍵字: START 亞馬遜 PS 函數(shù)

關于數(shù)據(jù)分析與可視化,以下介紹Pandas讀取數(shù)據(jù)以及保存數(shù)據(jù)的方法,畢竟我們很多時候需要讀取各種形式的數(shù)據(jù),以及將我們需要將所做的統(tǒng)計分析保存成特定的格式。

關鍵字: 函數(shù) Panda

深圳2024年8月15日 /美通社/ -- 近日,知名量化交易平臺水母量化正式推出了突破性的輪動交易策略回測功能,目前已進入公測階段。該功能實現(xiàn)了從策略設計、歷史回測到實盤自動交易一體化流程,幫用戶解決了策略驗證難、交易...

關鍵字: 編程 仿真 函數(shù) 開源

Python由荷蘭數(shù)學和計算機科學研究學會的吉多·范羅蘇姆于1990年代初設計,作為一門叫做ABC語言的替代品。 Python提供了高效的高級數(shù)據(jù)結構,還能簡單有效地面向?qū)ο缶幊獭?/p> 關鍵字: python 函數(shù) 對象編程

測試數(shù)據(jù)綜合分析的絕佳工具,深受工程師和研究員歡迎

關鍵字: 后處理分析軟件 向?qū)?/a> 函數(shù)

由上圖中可以知道進程地址空間中最頂部的段是棧,代碼中調(diào)用函數(shù)、定義局部變量(但不包含static修飾的變量)或聲明的類的實例等等都要使用??臻g,當函數(shù)執(zhí)行完(也就是程序執(zhí)行超過了這個函數(shù)的作用范圍的時候),操作系統(tǒng)會把該...

關鍵字: 進程地址 局部變量 函數(shù)

星標/置頂?公眾號,硬核文章第一時間送達!鏈接|?https://zhuanlan.zhihu.com/p/274473971題很多,先上題后上答案,便于大家思考問題點:1、C和C的特點與區(qū)別?2、C的多態(tài)3、虛函數(shù)實現(xiàn)...

關鍵字: 騰訊 函數(shù) 進程 AI

程序接口是操作系統(tǒng)為用戶提供的兩類接口之一,編程人員在程序中通過程序接口來請求操作系統(tǒng)提供服務。面向過程語言最基本的單元是過程和函數(shù)。

關鍵字: 程序接口 過程 函數(shù)

星標「嵌入式大雜燴」,一起進步!鏈接:https://www.cnblogs.com/jozochen/p/8541714.html一、問題復現(xiàn)穩(wěn)定復現(xiàn)問題才能正確的對問題進行定位、解決以及驗證。一般來說,越容易復現(xiàn)的問...

關鍵字: 嵌入式開發(fā) 函數(shù) 代碼 寄存器
關閉