当前位置:首页 > 单片机 > 单片机
[导读] /*----------------------------------------------------------------------PIC24F之EEPROM读写中断事件处理函数要点及说明注意:这是一个通用的I2C/SMBUS通讯中断处理程序对于EEPROM来讲,从机后面需要

/*----------------------------------------------------------------------
PIC24FEEPROM读写中断事件处理函数要点及说明
注意:这是一个通用的I2C/SMBUS通讯中断处理程序
对于EEPROM来讲,从机后面需要跟EEPROM需要读写的地址(I2CRegs.RWAddr)
对于SMBUS来说,从机后面需要跟SMBUS需要的命令(I2CRegs.RWAddr改为I2CRegs.CMD即可)


由于PIC24F的I2C不太标准,I2C1STAT被搞得很倒塌!!!一点都没I2C的"大家闺秀"的样子~~~
不过它的STOP还能激活中断确实比LPCARM/AVR好一点点~~~

为什么I2C收发都用中断呢???
这主要是为了高低速灵活变化的总线通讯所做,主要是SMBUS总线的通信.


如果为I2cExit()也配上钩子函数,那么任何错误都在手掌中~~~

这个PIC程序虽没SMBUS的PEC校验部分,但"异步"还是完美的.
当然也要注意对写保护硬件管脚的控制时机的把握,原则是关保护的时间最短就更好~~~


主要看到人们编写MCU程序太死板~~~特别是I2C程序.网上收发全中断的很少,可以说几乎没有.

随贴附老外倒塌的非中断I2C状态机读写程序i2cEmem.c~~~可以比较经典和非典的差异在何处~~~

菜农近期将整理出LPCARM和AVR的I2C/SMBUS/TWI/USI收发全中断实战例程供大家“游玩”~~~

如果精通DELPHI程序的人一定会为“事件驱动”机制而痴迷~~~为什么不在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;//发送数据字节个数
I2CRegs.RxCount=0;//接收数据字节个数
for(i=0;i<16;i++)
{
I2CRegs.TxBuffer[i]=0;//发送缓冲区清零
}
for(i=0;i<256;i++)
{
I2CRegs.RxBuffer[i]=0;//接收缓冲区清零
}

I2C1CON=0;
//I2C1CONbits.A10M=0;//7位地址模式
I2C1CONbits.SCLREL=1;
I2C1MSK=0;
I2C1STAT=0;
_MI2C1IF=0;
_SI2C1IF=0;
I2C1BRG=(FCY/(2*I2CBAUD))-1;//波特率计算
/*------------------------------------------------------------------------
定义I2C串口2中断优先级位1111)
-------------------------------------------------------------------------*/
IPC4bits.MI2C1P0=1;
IPC4bits.MI2C1P1=1;
IPC4bits.MI2C1P2=1;

I2C1CONbits.I2CEN=1;//允许I2C功能
_MI2C1IE=1;//允许主设备中断

//I2cStop();
}

/*------------------------------------------------------------------
EEPROM读块函数(只能在回调函数I2CReadCallBack中得到读出的数据)
-------------------------------------------------------------------*/
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写块函数
-------------------------------------------------------------------*/
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();
}

/*------------------------------------------------------------------
用户读回调函数
-------------------------------------------------------------------*/
voidI2CReadCallBack(void)
{
if((I2CRegs.RWAddr+I2CRegs.RxCount)<=I2CRegs.MaxCount)
{
//I2CRegs.RWAddr+=I2CRegs.RxCount;
//I2CReadBuffers(I2CRegs.RWAddr,I2CRegs.RxCount);//继续读
}
}

/*------------------------------------------------------------------
用户写回调函数
-------------------------------------------------------------------*/
voidI2CWriteCallBack(void)
{
if((I2CRegs.RWAddr+I2CRegs.TxCount)<=I2CRegs.MaxCount)
{
//I2CRegs.RWAddr+=I2CRegs.TxCount;
//I2CWriteBuffers(I2CRegs.RWAddr,I2CRegs.TxCount);//继续写
}
}


/*------------------------------------------------------------------
EEPROM读写启动函数
-------------------------------------------------------------------*/
voidI2cStart(void)
{
/*------------------------------------------------------------------------
//本程序在状态I2C_MT_ADDRL_ACK下进行瞬间打开,也可在此打开,不过安全不好
if(I2CRegs.TxCount)//需要写入字节
{
WP=0;//不写保护
}
else
{
WP=1;//写保护
}
--------------------------------------------------------------------------*/
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=1;
I2CRegs.State=I2C_START;//主机准备发送启始位
I2CRegs.Count=0;//发送数据个数
I2CBits.I2CFlag=0;
I2C1CONbits.SEN=1;//发送Start信号
}

/*------------------------------------------------------------------
EEPROM读再启动函数
-------------------------------------------------------------------*/
voidI2cReStart(void)
{
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=1;
I2CRegs.State=I2C_REP_START;//主机准备发送重新启始位
I2CRegs.Count=0;//发送数据个数
I2C1CONbits.RSEN=1;//发送ReStart信号
I2C1CONbits.ACKEN=0;
}

/*------------------------------------------------------------------
EEPROM读写正确停止函数
-------------------------------------------------------------------*/
voidI2cStop(void)
{
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=0;
I2CRegs.State=I2C_SUCCEEDED;//通讯成功
I2C1CONbits.PEN=1;//发送Stop信号
WP=1;//写保护
}


/*------------------------------------------------------------------
EEPROM读写错误退出函数
-------------------------------------------------------------------*/
voidI2cExit(void)
{
I2C1STATbits.IWCOL=0;
I2CBits.BusyFlag=0;
I2CRegs.State=I2C_FAILED;
I2C1CONbits.PEN=1;//发送Stop信号
WP=1;//写保护
}

/*------------------------------------------------------------------
EEPROM读写中断事件处理函数(说明见文件头部)
-------------------------------------------------------------------*/
voidI2CExec(void)
{
if(I2C1STATbits.S)//收到Start过信号
{
switch(I2CRegs.State)
{
caseI2C_START://收到Start信号
I2C1TRN=I2CRegs.I2CAddr&0xfe;//发送器件写地址(通知从机只能听)
I2CRegs.State=I2C_MT_SLA_ACK;//下次应该接收器件写地址应答信号
break;
caseI2C_MT_SLA_ACK://收到器件写地址应答信号
if(!I2C1STATbits.ACKSTAT)//收到Ack信号
{
if(I2CRegs.MaxCount>0x100)//EEPROM容量超过256个字节,EEPROM地址需要两次发送
{
I2C1TRN=I2CRegs.RWAddr>>8;//发送EEPROM写高8位地址
I2CRegs.State=I2C_MT_ADDRH_ACK;//下次应该接收EEPROM写高8位地址应答信号
}
else//小容量只需一次发送!!!
{
I2C1TRN=I2CRegs.RWAddr;//发送EEPROM写低8位地址
I2CRegs.State=I2C_MT_ADDRL_ACK;//下次应该接收EEPROM写低8位地址应答信号
I2CRegs.Count=0;//清空发送缓冲计数器
}
}
else//收到NAck信号
{
I2cExit();//错误的ACK信号
}
break;
caseI2C_MT_ADDRH_ACK://收到EEPROM写高8位地址应答信号
if(!I2C1STATbits.ACKSTAT)//收到Ack信号
{
I2C1TRN=I2CRegs.RWAddr&0xff;//发送EEPROM写低8位地址
I2CRegs.State=I2C_MT_ADDRL_ACK;//下次应该接收EEPROM写低8位地址应答信号
I2CRegs.Count=0;//清空发送缓冲计数器
}
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++];//继续发送数据
}
elseif(I2CRegs.Count==I2CRegs.TxCount)//缓冲区已空
{
if(I2CRegs.I2CAddr&1)//应该开始接收数据
{
I2cReStart();//发送重复位命令
}
else//只写退出
{
I2cStop();//正常发送结束
}
}
else//干扰出错
{
I2cExit();//错误
}
}
else//收到NAck信号(可能被写保护)
{
I2cExit();//错误的ACK信号
}
break;
caseI2C_REP_START://收到ReStart信号
I2C1TRN=I2CRegs.I2CAddr|I2C_READ;//发送器件读地址(通知从机可以说话)
I2CRegs.State=I2C_MR_SLA_ACK;//下次应该接收器件写读地址应答信号
break;
caseI2C_MR_SLA_ACK://收到器件读地址应答信号
if(!I2C1STATbits.ACKSTAT)//收到Ack信号
{
I2C1CONbits.RCEN=1;//开始接收数据
I2CRegs.State=I2C_MR_DATA;//下次应该收接收数据
}
else//收到NAck信号
{
I2cExit();//错误的ACK信号
}
break;
caseI2C_MR_DATA://收到接收数据
if(I2CRegs.Count{
//I2C1STATbits.I2COV=0;
I2CRegs.RxBuffer[I2CRegs.Count++]=I2C1RCV;
if(I2CRegs.Count{
I2C1CONbits.ACKDT=0;//应答子机
I2CRegs.State=I2C_MR_DATA_EN;//下次应该收到器件允许继续读信号
}
else
{
I2C1CONbits.ACKDT=1;//非应答子机
I2CRegs.State=I2C_MR_DATA_STOP;//下次应该收到退出信号
}
I2C1CONbits.ACKEN=1;//向从机发送(非)应答信号
}
else//正确的状态已分支到I2C_MR_DATA_STOP
{
I2cExit();//错误
}
break;
caseI2C_MR_DATA_EN://收到器件允许继续读信号
I2C1CONbits.RCEN=1;//开始接收数据
I2CRegs.State=I2C_MR_DATA;//下次应该继续接收数据
break;
caseI2C_MR_DATA_STOP://收到器件退出信号
I2cStop();//正常接收结束
break;
default://其他不可预料的错误
I2cExit();//错误
}
}
elseif(I2C1STATbits.P)//收到Stop信号
{
if(I2CRegs.State==I2C_SUCCEEDED)//成功,回调
{
if(I2CRegs.I2CAddr&1)//读
{
I2CBits.ReadFlag=1;//激活用户读回调函数I2CReadCallBack()
}
else//写
{
I2CBits.WriteFlag=1;//激活用户写回调函数I2CWriteCallBack()
}
}
}
else//无法确认的复杂错误
{
I2cExit();//错误出错退出
}
}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭