找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 310|回复: 1
收起左侧

关于单片机大电容缓下电掉EEPROM数据的问题

[复制链接]
回帖奖励 200 黑币 回复本帖可获得 10 黑币奖励! 每人限 1 次
ID:1102088 发表于 2024-2-28 23:41 | 显示全部楼层 |阅读模式
求助,辉芒微FT62F211在开启触摸功能的时候,下电到2.5V在重新上电就会清除EEPROM数据,但是程序也是正常跑没什么问题啥。关闭触摸功能就不掉数据了

下面是代码
  1. /**********************************************************************************
  2.   * @芯片资源
  3.   * 1. Device: FT62F21X
  4.   * 2. Memory: Flash 1KX14b, SRAM:64X8b EEPROM 256X8b
  5.   *
  6.   * 3.  文件名:main.c   
  7.   *
  8. //*********************************************************************************/

  9. #include "Touch.h"
  10. unsigned int    led1_pwm;           //led1_pwm值

  11. unsigned char    key1_cnt;          //key1按键计数

  12. unsigned char   key1_short_flag;    //key1按键短按标志

  13. unsigned char   led1_short_state;   //led1短按状态


  14. unsigned char   key1_long_flag; //key1按键长按标志

  15. unsigned char   led1_long_state;    //led1长按状态

  16. unsigned char   led1_long_darkness_lock;    //led1熄灭长按状态


  17. unsigned char   led1_pwm_h;
  18. unsigned char   led1_pwm_l;

  19. unsigned char TK_REVal;    //触摸按键返回值

  20. unsigned char EEPROMread(unsigned char EEAddr);
  21. void EEPROMwrite(unsigned char EEAddr , unsigned char Data);

  22. void scann_key(unsigned char  ucKeySingle)
  23. {
  24.     switch(ucKeySingle)
  25.     {
  26.     /******************************** case 0 *************************************/
  27.         case 0:
  28.             /* 按键松开,结束计数*/
  29.             key1_cnt = 0;      

  30.             /*key1按键短按处理*/
  31.             if(key1_short_flag)     
  32.             {                  
  33.                 if(led1_short_state == 0)//led1熄灭时点亮
  34.                 {
  35.                     led1_pwm = led1_pwm + 100;
  36.                     if(led1_pwm > 10000)
  37.                     {
  38.                         key1_short_flag = 0;//清除key1短按标志
  39.                         led1_short_state = 1; //led1切换短按状态

  40.                         EEPROMwrite(0x00,P1ADTH );
  41.                         EEPROMwrite(0x01,P1ADTL );
  42.                         EEPROMwrite(0x02,key1_short_flag );
  43.                         EEPROMwrite(0x03,led1_short_state  );
  44.                         EEPROMwrite(0x04,key1_long_flag );
  45.                         EEPROMwrite(0x05,led1_long_state  );
  46.                     }
  47.                 }

  48.                 if(led1_short_state == 1)//led1点亮时熄灭
  49.                 {
  50.                     if(led1_pwm < 100)
  51.                     {
  52.                         led1_pwm = 0;
  53.                         key1_short_flag = 0;//清除key1短按标志
  54.                         led1_short_state = 0; //led1切换短按状态
  55.                         led1_long_state   = 0;//清除led1长按状态
  56.                         key1_long_flag      = 0;
  57.                         EEPROMwrite(0x00,(led1_pwm>>8) );
  58.                         EEPROMwrite(0x01,led1_pwm );
  59.                         EEPROMwrite(0x02,key1_short_flag );
  60.                         EEPROMwrite(0x03,led1_short_state  );
  61.                     }
  62.                     else
  63.                         led1_pwm = led1_pwm - 100;
  64.                 }

  65.             }

  66.             /*key1按键长按处理*/
  67.             if(key1_long_flag)
  68.             {
  69.                 key1_long_flag = 0;//清除key1长按标志
  70.                 led1_long_state = led1_long_state ^ 0x01; //切换led1长按状态

  71.                 EEPROMwrite(0x00,P1ADTH );
  72.                 EEPROMwrite(0x01,P1ADTL );
  73.                 EEPROMwrite(0x04,key1_long_flag );
  74.                 EEPROMwrite(0x05,led1_long_state  );
  75.             }

  76.             if(led1_long_darkness_lock)
  77.             {
  78.                 led1_long_darkness_lock = 0;
  79.                 led1_short_state = 1;
  80.                 if(led1_long_state)
  81.                     led1_long_state =0;
  82.                 EEPROMwrite(0x00,P1ADTH );
  83.                 EEPROMwrite(0x01,P1ADTL );               
  84.                 EEPROMwrite(0x03,led1_short_state );
  85.                 EEPROMwrite(0x06,led1_long_darkness_lock );
  86.             }

  87.             break;

  88.     /******************************** case 1 *************************************/
  89.         case 1:
  90.             /*按键按下,开始计数*/
  91.             key1_cnt++;

  92.             /*计数100毫秒消抖,大约150毫秒为按键短按*/
  93.             if(key1_cnt > 20)
  94.                 key1_short_flag = 1;

  95.             /*计数大于1秒为按键长按*/
  96.             if(key1_cnt > 150)
  97.             {
  98.                 key1_cnt = 150;
  99.                 key1_short_flag = 0;//清除按键短按标志


  100.                 if(led1_short_state)//检查led1状态
  101.                 {
  102.                     if(!led1_long_state)//长按减亮度
  103.                     {
  104.                         if(led1_pwm < 100)
  105.                             led1_pwm = 100;
  106.                         else                           
  107.                             led1_pwm = led1_pwm - 20;   

  108.                         key1_long_flag = 1;
  109.                     }



  110.                     else if(led1_long_state)//长按加亮度
  111.                     {
  112.                         if(led1_pwm > 10000)
  113.                             led1_pwm = 10000;
  114.                         else
  115.                             led1_pwm = led1_pwm + 20;   

  116.                         key1_long_flag = 1;
  117.                     }
  118.                 }

  119.                 else if(!led1_short_state)//用以led1熄灭时长按亮起
  120.                 {
  121.                     if(led1_pwm > 10000)
  122.                         led1_pwm = 10000;
  123.                     else
  124.                         led1_pwm = led1_pwm + 20;

  125.                     led1_long_darkness_lock = 1;    //通知短按函数 led1 已经亮起         
  126.                 }
  127.             }
  128.             break;

  129.     }
  130. }
  131. /*-------------------------------------------------
  132. * 函数名:DelayUs
  133. * 功能:  短延时函数 --16M-4T--大概快1%左右.
  134. * 输入:  Time延时时间长度 延时时长Time*2Us
  135. * 输出:  无
  136. -------------------------------------------------*/
  137. void DelayUs(unsigned char Time)
  138. {
  139.     unsigned char a;
  140.     for(a=0;a<Time;a++)
  141.     {
  142.         NOP();
  143.     }
  144. }
  145. /*-------------------------------------------------
  146. * 函数名:DelayMs
  147. * 功能:  短延时函数 快1%
  148. * 输入: Time延时时间长度 延时时长Time ms
  149. * 输出: 无
  150. -------------------------------------------------*/
  151. void DelayMs(unsigned char Time)
  152. {
  153.     unsigned char a,b;
  154.     for(a=0;a<Time;a++)
  155.     {
  156.         for(b=0;b<5;b++)
  157.         {
  158.             DelayUs(98);
  159.         }
  160.     }
  161. }
  162. /*-------------------------------------------------
  163. *  函数名称:interrupt ISR
  164. *  功能:   定时器中断处理函数
  165. *  输入参数:无
  166. *  返回参数:无
  167. -------------------------------------------------*/

  168. void interrupt ISR(void)//PIC_HI-TECH使用
  169. {
  170.     if(TMR2IE && TMR2IF)            //10ms中断一次
  171.     {
  172.         TMR2IF = 0;
  173.         TSC_Start();
  174.         TSC_DataProcessing();
  175.         TK_REVal = TSC_GetSingle();
  176.     }
  177. }

  178. /*-------------------------------------------------
  179. *  函数名:SYS_INITIAL
  180. *  功能:  上电系统初始化
  181. *  输入:  无
  182. *  输出:  无
  183. --------------------------------------------------*/   

  184. void SYS_INITIAL (void)
  185. {
  186.     //时钟先8M 写完PCON (bit7),OPTION(all bit),LVDCON(bit2),后换回16M
  187.     OSCCON = 0B01100000;   
  188.     //WDT 32KHZ IRCF=110=8MHZ/4=4MHZ,0.25US/T
  189.     //Bit0=1,系统时钟为内部振荡器
  190.     //Bit0=0,时钟源由FOSC<2:0>决定即编译选项时选择

  191.     OPTION = 0B00000100;   
  192.     //Bit5 T0CS Timer0时钟源选择
  193.     //1-外部引脚电平变化T0CKI 0-内部时钟(FOSC/2)
  194.     //Bit4 T0CKI引脚触发方式 1-下降沿 0-上升沿
  195.     //Bit3 PSA 预分频器分配位 0-Timer0 1-WDT
  196.     //Bit2:0 PS2 8个预分频比 100 - 1:32

  197.     LVDCON  = 0x03;         
  198. //    //bit2 0:不经过去抖电路  bit[1:0] 00:vdd 01:pa5 10:pa4 11:touch
  199. //   
  200.     PCON = 0x88;               
  201.     //开LVD  bit3 低电压侦测使能   bit[7:4]=1xxx 1.2v

  202.     OSCCON = SYS_OSCCON;   
  203.     //WDT 32KHZ IRCF=110=16MHZ/4=4MHZ,0.25US/T

  204.     T0CON0 = 0;                  
  205.     //复位上T0ON默认为1,所以需要清0

  206.     MSCON = 0;                  
  207.     //软件关闭LVREN不然低功耗电流会变大(Options下必须选择LVREN=SLVREN)

  208.     INTCON = 0;                 
  209.     //暂禁止所有中断

  210.     PORTA = 0;
  211.     ///////////特别注意与TKCAP合打线的PIN//////////////////////////
  212.     //对于6pin 封装, TKCAP 与PA2 共用引脚3。
  213.     //对于DFN8 封装, TKCAP 与PA4 共用引脚2。
  214.     //引脚用作TKCAP 功能时,则PA2/PA4 必须相应地设置为输入;
  215.     //当关闭Touch模块功能时,则PA2/PA4 为通用IO功能。

  216.     TRISA = TRISA_CONFIG;   
  217.     // 0-输出 1-输入  
  218.     ///////////特别注意与TKCAP合打线的PIN//////////////////////////

  219.     WPUA = 0B00000000;   
  220.     //PA端口上拉控制 1-开上拉 0-关上拉

  221.     PSRCA = 0B00000000;   
  222.     //PA4,PA5源电流能力设置为0                             
  223. }

  224. /*-------------------------------------------------
  225. * 函数名:TIMER2_INITIAL
  226. * 功能:  初始化设置定时器1
  227. * 设置Timer2定时时长 = (1/系统时钟频率)*4*预分频值*后分频值*PR2
  228. *                    = (1/16000000)*4*4*1*10000=10ms
  229. -------------------------------------------------*/
  230. void TIMER2_INITIAL (void)
  231. {
  232.     T2CON0 = 0B00000001;
  233.     //Bit[6:3]=0000,T2时钟后分频比1:1
  234.     //Bit[1:0]=01,T2时钟预分频比1:4

  235.     T2CON1 = 0B00000000;            //Bit[2:0] 000: 指令周期 100:HIRC Timer2时钟源选择   


  236.     TMR2H = 0;
  237.     TMR2L = 0;                      //TMR2赋初    0010 0001 0011 0100

  238.     PR2H = 0x27;
  239.     PR2L = 0x10;                    //设置PR2=10000

  240.     //pwm初始化
  241.     P1ADTH=0;                       //P1ADT赋值
  242.     P1ADTL=0;
  243.     P1OE=0B00000001;                //P1A0输出使能
  244.     P1POL=0B00000000;               //P1A0高电平有效
  245.     P1CON=0B00000000;

  246.     TMR2IF = 0;                     //清TIMER2中断标志
  247.     TMR2IE = 1;                     //使能TIMER2的中断
  248.     TMR2ON = 1;                     //使能TIMER2启动

  249.     PEIE=1;                         //使能外设中断
  250.     GIE = 1;                        //使能全局中断
  251. }

  252. /*-------------------------------------------------
  253. * 函数名:EEPROMread
  254. * 功能:  读EEPROM数据
  255. * 输入:  需要读取数据的地址 EEAddr
  256. * 输出:  对应地址读出的数据 ReEEPROMread
  257. --------------------------------------------------*/
  258. unsigned char EEPROMread(unsigned char EEAddr)
  259. {
  260.     unsigned char ReEEPROMread;
  261.     EEADR = EEAddr;
  262.     RD=1;
  263.     NOP();
  264.     NOP();
  265.     NOP();
  266.     NOP();
  267.     ReEEPROMread =EEDAT;
  268.     return ReEEPROMread;
  269. }
  270. /*-------------------------------------------------
  271. * 函数名:EEPROMwrite
  272. * 功能:  写数据到EEPROM
  273. * 输入:  需要读取数据的地址 EEAddr
  274.            需要写入的数据  Data
  275. * 输出:  无
  276. --------------------------------------------------*/
  277. void EEPROMwrite(unsigned char EEAddr , unsigned char Data)
  278. {
  279.     GIE = 0;                    //写数据必须关闭中断
  280.     while(GIE);                 //等待GIE为0
  281.     EEADR = EEAddr;             //EEPROM的地址
  282.     EEDAT = Data;               //EEPROM的写数据  EEDATA = Data;
  283.     EEIF = 0;
  284.     EECON1 |= 0x34;             //置位WREN1,WREN2,WREN3三个变量.
  285.     WR = 1;                     //置位WR启动
  286.     NOP();
  287.     NOP();
  288.     NOP();
  289.     NOP();
  290.     while(WR);                  //等待EE写入完成
  291.     GIE = 1;
  292. }

  293. /*------------------------------------------------------------------------------------------
  294. *  函数名:  main
  295. *  功能:  主函数
  296. *  输入:  无
  297. *  输出:  无
  298. ----------------------------------------------------------------------------------------------*/

  299. void main(void)
  300. {
  301.     /* 系统初始化 */
  302.     SYS_INITIAL();  
  303.     /* 按键初始化 */
  304.     TOUCH_INITIAL();
  305.     /* 定时器2初始化 */
  306.     TIMER2_INITIAL();

  307.     led1_pwm_h = EEPROMread(0x00);
  308.     led1_pwm_l  = EEPROMread(0x01);

  309.     led1_pwm =  led1_pwm_l | (unsigned int)(led1_pwm_h<<8) ;

  310.     key1_short_flag  = EEPROMread(0x02);
  311.     led1_short_state = EEPROMread(0x03);
  312.     key1_long_flag    = EEPROMread(0x04);
  313.     led1_long_state   = EEPROMread(0x05);
  314.     led1_long_darkness_lock = EEPROMread(0x06);




  315.     while(1)
  316.     {  
  317.         P1ADTH =  (led1_pwm>>8);
  318.         P1ADTL = (unsigned char)led1_pwm;              
  319.         scann_key(TK_REVal);
  320.         DelayMs(2);
  321.     }

  322. }
复制代码
电路图

图
回复

使用道具 举报

ID:883242 发表于 2024-3-1 16:56 | 显示全部楼层
单片机应该是支持brownout,比如AVR单片机可以通过熔丝位设定当电源电压下降到4.3V或者2.7V锁定单片机,不让单片机低压时候程序乱飞,避免这种错误的发生。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表