PIC24F之EEPROM读写中断事件处理函数要点及说明

发布者:SerendipityJoy最新更新时间:2017-12-06 来源: eefocus关键字:PIC24F  EEPROM  读写中断事件  处理函数 手机看文章 扫描二维码
随时随地手机看文章

/*----------------------------------------------------------------------
PIC24F之EEPROM读写中断事件处理函数要点及说明 
注意: 这是一个通用的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" 

_PERSISTENT volatile I2CREGS I2CRegs; 
_PERSISTENT volatile I2CBITS I2CBits; 

void I2cInit(void) 

unsigned int i; 
    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中得到读出的数据) 
-------------------------------------------------------------------*/ 
void I2CReadBuffers(unsigned int E2RomAddr, unsigned int ReadSize) 

    if (ReadSize && (ReadSize <= 256)) 
    { 
        I2CRegs.TxCount = 0; 
        I2CRegs.RxCount = ReadSize; 
        I2CRegs.RWAddr = E2RomAddr; 
        I2CRegs.I2CAddr |= 1;//0xa1 
        I2cStart(); 
    } 


void I2CReadByte(unsigned int E2RomAddr) 

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


/*------------------------------------------------------------------ 
    EEPROM写块函数 
-------------------------------------------------------------------*/ 
void I2CWriteBuffers(unsigned int E2RomAddr, unsigned int WriteSize) 

    if (WriteSize && (WriteSize <= 16)) 
    { 
        I2CRegs.TxCount = WriteSize; 
        I2CRegs.RxCount = 0; 
        I2CRegs.RWAddr = E2RomAddr; 
        I2CRegs.I2CAddr &= 0xfe;//0xa0 
        I2cStart(); 
    } 


void I2CWriteByte(unsigned int E2RomAddr, unsigned char cData) 

    I2CRegs.TxBuffer[0] = cData; 
    I2CRegs.TxCount = 1; 
    I2CRegs.RxCount = 0; 
    I2CRegs.RWAddr = E2RomAddr; 
    I2CRegs.I2CAddr &= 0xfe;//0xa0 
    I2cStart(); 


/*------------------------------------------------------------------ 
    用户读回调函数 
-------------------------------------------------------------------*/ 
void I2CReadCallBack(void) 

    if ((I2CRegs.RWAddr + I2CRegs.RxCount) <= I2CRegs.MaxCount)  
    { 
//        I2CRegs.RWAddr += I2CRegs.RxCount; 
//        I2CReadBuffers(I2CRegs.RWAddr, I2CRegs.RxCount);//继续读 
    } 


/*------------------------------------------------------------------ 
    用户写回调函数 
-------------------------------------------------------------------*/ 
void I2CWriteCallBack(void) 

    if ((I2CRegs.RWAddr + I2CRegs.TxCount) <= I2CRegs.MaxCount)  
    { 
//        I2CRegs.RWAddr += I2CRegs.TxCount; 
//        I2CWriteBuffers(I2CRegs.RWAddr, I2CRegs.TxCount);//继续写 
    } 



/*------------------------------------------------------------------ 
    EEPROM读写启动函数 
-------------------------------------------------------------------*/ 
void I2cStart(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读再启动函数 
-------------------------------------------------------------------*/ 
void I2cReStart(void) 

    I2C1STATbits.IWCOL = 0; 
    I2CBits.BusyFlag = 1; 
    I2CRegs.State = I2C_REP_START;//主机准备发送重新启始位 
    I2CRegs.Count = 0;//发送数据个数 
    I2C1CONbits.RSEN = 1;//发送ReStart信号 
    I2C1CONbits.ACKEN = 0; 


/*------------------------------------------------------------------ 
    EEPROM读写正确停止函数 
-------------------------------------------------------------------*/ 
void I2cStop(void) 

    I2C1STATbits.IWCOL = 0; 
    I2CBits.BusyFlag = 0; 
    I2CRegs.State = I2C_SUCCEEDED;//通讯成功 
    I2C1CONbits.PEN = 1;//发送Stop信号 
    WP = 1;//写保护 



/*------------------------------------------------------------------ 
    EEPROM读写错误退出函数 
-------------------------------------------------------------------*/ 
void I2cExit(void) 

    I2C1STATbits.IWCOL = 0; 
    I2CBits.BusyFlag = 0; 
    I2CRegs.State = I2C_FAILED
    I2C1CONbits.PEN = 1;//发送Stop信号 
    WP = 1;//写保护 


/*------------------------------------------------------------------ 
     EEPROM读写中断事件处理函数(说明见文件头部) 
-------------------------------------------------------------------*/ 
void I2CExec(void) 

    if (I2C1STATbits.S)//收到Start过信号 
    { 
        switch (I2CRegs.State) 
        { 
            case I2C_START://收到Start信号 
                I2C1TRN = I2CRegs.I2CAddr & 0xfe;//发送器件写地址(通知从机只能听) 
                I2CRegs.State = I2C_MT_SLA_ACK;//下次应该接收器件写地址应答信号 
                break; 
            case I2C_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; 
            case I2C_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; 
            case I2C_MT_ADDRL_ACK://收到EEPROM写高低8位地址应答信号 
                if (I2CRegs.TxCount)//写保护只在写入期间不保护,增加了对误写入的安全防护能力!!! 
                { 
                     WP = 0;//不写保护 
                } 
            case I2C_MT_DATA_ACK://收到应答信号 
                if (!I2C1STATbits.ACKSTAT)//收到Ack信号 
                { 
                    if (I2CRegs.Count < I2CRegs.TxCount)//缓冲区未空 
                    { 
                        I2C1TRN = I2CRegs.TxBuffer[I2CRegs.Count ++];//继续发送数据     
                    } 
                    else if (I2CRegs.Count == I2CRegs.TxCount)//缓冲区已空 
                    { 
                        if (I2CRegs.I2CAddr & 1)//应该开始接收数据 
                        { 
                            I2cReStart();//发送重复位命令 
                        } 
                        else//只写退出 
                        { 
                            I2cStop();//正常发送结束 
                        } 
                    } 
                    else//干扰出错 
                    { 
                        I2cExit();//错误 
                    } 
                }     
                else//收到NAck信号(可能被写保护) 
                { 
                    I2cExit();//错误的ACK信号     
                }     
                break; 
            case I2C_REP_START://收到ReStart信号 
                I2C1TRN = I2CRegs.I2CAddr | I2C_READ;//发送器件读地址(通知从机可以说话) 
                I2CRegs.State = I2C_MR_SLA_ACK;//下次应该接收器件写读地址应答信号 
                break; 
            case I2C_MR_SLA_ACK://收到器件读地址应答信号 
                if (!I2C1STATbits.ACKSTAT)//收到Ack信号 
                { 
                    I2C1CONbits.RCEN = 1;//开始接收数据 
                    I2CRegs.State = I2C_MR_DATA;//下次应该收接收数据 
                }     
                else//收到NAck信号 
                { 
                    I2cExit();//错误的ACK信号     
                }     
                break; 
            case I2C_MR_DATA://收到接收数据 
                if (I2CRegs.Count < I2CRegs.RxCount) 
                { 
//                    I2C1STATbits.I2COV = 0; 
                    I2CRegs.RxBuffer[I2CRegs.Count ++] = I2C1RCV; 
                    if (I2CRegs.Count < I2CRegs.RxCount) 
                    { 
                         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; 
            case I2C_MR_DATA_EN://收到器件允许继续读信号 
                I2C1CONbits.RCEN = 1;//开始接收数据 
                I2CRegs.State = I2C_MR_DATA;//下次应该继续接收数据 
                break; 
            case I2C_MR_DATA_STOP://收到器件退出信号 
                I2cStop();//正常接收结束 
                break; 
            default://其他不可预料的错误 
                I2cExit();//错误 
        } 
    } 
    else if (I2C1STATbits.P)//收到Stop信号 
    { 
        if (I2CRegs.State == I2C_SUCCEEDED)//成功,回调 
        { 
            if (I2CRegs.I2CAddr & 1)//读 
            { 
                I2CBits.ReadFlag = 1;//激活用户读回调函数I2CReadCallBack() 
            } 
            else//写 
            { 
                I2CBits.WriteFlag = 1;//激活用户写回调函数I2CWriteCallBack() 
            } 
        } 
    } 
    else//无法确认的复杂错误  
    { 
        I2cExit();//错误出错退出 
    } 


关键字:PIC24F  EEPROM  读写中断事件  处理函数 引用地址:PIC24F之EEPROM读写中断事件处理函数要点及说明

上一篇:PIC单片机PWM模块应用实验程序
下一篇:PIC单只数码管循环显示0-9

推荐阅读最新更新时间:2024-03-16 15:48

AT93C46/56/55串行EEPROM及单片机程序
16位单片机以其适于高速控制场合及功能多等优点已在工业控制领域中占领了一定的市场。由于EEPROM能在不脱离系统的情况下修改其存储单元中的内容,故在16位单片机中的应用愈来愈广泛。本文结合16位机的特点,详细介绍AT93系列EEPROM及其使用方法。 AT93C46/56/66是ATMEL公司推出的低功耗、低电压电可擦除的可编程只读存储器。它采用CMOS技术和Fairchild Semiconductor 公司的Mi-croWire工业标准3线串行接口,具有1kB/2kB/4kB的容量,并可通过ORG管脚配置成128×8/256×8/512×8或64×16/128×16/256×16等结构。该系列存储器可靠性高,能够重复写10
[单片机]
AT93C46/56/55串行<font color='red'>EEPROM</font>及单片机程序
Microchip推出业界最快的1 Mb串行EEPROM
新的串行EEPROM提供字节级读/写功能,可在高达125°C的全电压范围下工作 全球领先的单片机和模拟半导体供应商——Microchip Technology Inc.(美国微芯科技公司)宣布推出业界速度最快(20 MHz)的1 Mb串行EEPROM器件25AA1024及25LC1024(25XX1024)。同时,公司还推出了25AA128、25LC128、25AA512及25LC512(25XX128/512)等多款128 Kb和512 Kb串行EEPROM器件,涵盖整个串行外设接口(SPI)存储密度范围(1 Kb至1 Mb)。 所有新器件均可在高达125°C的温度下工作,并可提供Microchip所有串行EEPROM产品一
[新品]
关于STM32F407开发板的几种内存总结,SRAM,FLASH,EEPROM
常见存储器概念:RAM、SRAM、SDRAM、ROM、EPROM、EEPROM、Flash存储器可以分为很多种类,其中根据掉电数据是否丢失可以分为RAM(随机存取存储器)和ROM(只读存储器),其中RAM的访问速度比较快,但掉电后数据会丢失,而ROM掉电后数据不会丢失。 可以知道SRAM属于RAM,掉电后数据丢失;FLASH和EEPROM属于ROM,掉电后数据不丢失。 FLASH和EEPROM的区别在于:Flash存储器,适用于速度要求高,容量要求大,掉电时要求数据不丢失的场合;EEPROM适用于速度不高,容量不大,掉电时要求数据不丢失的场合。 了解这些之后,翻看开发板的开发手册统计一下用到了哪些内存。 1.芯片自
[单片机]
AVR 内部EEPROM读写范例
*********************************************** **** AVR 内部EEPROM读写范例 *** **** 编译器:WINAVR20050214 *** **** *** **** www.OurAVR.com 2005.9.24 *** ***********************************************/ /* 本程序简单的示范了如何使用ATMEGA16的EERPOM EEPROM的简介 EEPROM的写操作 EEPROM的读操作 出于简化程序考虑,各种数据没有对外输出,学习时建议使用JTAG ICE硬件仿真器 在打开调试文
[单片机]
STC单片机 EEPROM/IAP 功能测试程序演示
/* --- STC International Limited ---------------- 一个完整的EEPROM 测试程序,用宏晶的下载板可以直接测试 STC12C5AxxAD 系列单片机 EEPROM/IAP 功能测试程序演示 STC12C52xxAD 系列单片机 EEPROM/IAP 功能测试程序演示 STC11xx 系列单片机 EEPROM/IAP 功能测试程序演示 STC10xx 系列单片机 EEPROM/IAP 功能测试程序演示 --- STC International Limited ------------------ --- 宏晶科技 设计 2009/1/12 V1.0 --------------
[单片机]
[STM32/8经验] 经验分享——EEPROM读写及数据管理
经常有人在QQ群讨论有关E2PROM操作、保护、失效等一些问题,今天闲来没事,简单概括总结一下咯。(才疏学浅,路过的英雄记得补刀) 1)有关 芯片 内部自带的E2PROM 有的资深的工程师可能不是很愿意用MCU自带的E2PROM,大概是这些家伙上过什么当、受过骗还是什么的。简单说一下优缺点吧: (1)片内集成的感觉挺便宜的,和MCU一起卖的,对成本要求严格的,写次数很少的场合一般会考虑,至少能省个2、3毛钱吧; (2)缺点的话,可能擦写次数、稳定性没有外挂的专用芯片厉害; 2)有关常用的外挂E2PROM 如果是学生或者刚从学生时代走来的人们,大概也就知道ATMEL 24C0X系列吧,就好像我毕业的时候以为世界
[单片机]
I2C总线结构的EEPROM
常用芯片AT24C02 AT24C02是一个2K位串行CMOS E2PROM, 内部含有256个8位字节。AT24C02有一个8字节页写缓冲器。该器件通过IIC总线接口进行操作,有一个专门的写保护功能 引脚 SCL 串行时钟 AT24C02串行时钟输入管脚用于产生器件所有数据发送或接收的时钟,这是一个输入管脚。 SDA 串行数据/地址 AT24C02 双向串行数据/地址管脚用于器件所有数据的发送或接收,SDA 是一个开漏输出管脚,可与其它开漏输出或集电极开路输出进行线或(wire-OR)。 A0、A1、A2 器件地址输入端 这些输入脚用于多个器件级联时设置器件地址,当这些脚悬空时默认值为0。当使用AT24C02 时最大可级联
[单片机]
AT240C02中EEPROM保存数据
/*********************** 程序功能: 利用定时器产生一个0~99秒变化的秒表,并且显示在数码管上, 每过一秒将这个变化的数写入板上AT24C02内部。当关闭电源, 并再次打开电源时,单片机先从AT24C02中将原来写入的数读取出来, 接着此数继续变化并显示在数码管上。 ************************/ /************************ CODE: ************************/ #include reg52.h #define uint unsigned int #define uchar unsigned char #define AT
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved