找回密码
 立即注册
查看: 62|回复: 3

请教大神:为什么AI8051U休眠唤醒后ADC采样总是死循环?

[复制链接]
  • 打卡等级:常住居民I
  • 打卡总天数:75
  • 最近打卡:2025-08-13 11:21:33
已绑定手机

11

主题

34

回帖

427

积分

中级会员

积分
427
发表于 3 天前 | 显示全部楼层 |阅读模式
用AI8051U设计了一个温度、压力信号采样电路。之前都是在主循环里完成调用ADC采样、显示、存储等函数,运行一切正常。后来为了降低功耗,设计了休眠机制,CPU平时休眠,每分钟醒来后依次执行ADC采样、显示、存储功能,再休眠。用的是RTC的分钟中断唤醒CPU。ADC采样函数还是之前运行正常的代码。但是在休眠醒来后,调用ADC采样函数就出问题,陷入死循环,造成后续无法再次进入休眠状态。后来怀疑是系统时钟、或ADC的电源、ADC设置问题,就把程序上电后的初始化过程,比如时钟设置、IO口设置、ADC初始化等代码,在醒来后依次执行一遍,再运行ADC采样函数,也就是醒来后执行的的代码,与上电后的代码几乎一致了。可即使这样,ADC采样还是陷入死循环(后来改为返回错误值)。调试了几天了,我也快没招了,特向大神请教,如何解决?有关代码见附后://**************************   Main 函数 *****************************************************
void main(void)       
{       
               
                //初始化设置内容......
               
        RTC_config();               //配置RTC,1分钟中断唤醒CPU
        EA = 1;                       //打开总中断
        
             while (1)
        {   
                Flag_1min=0;

                 
                PD = 1  ;  //进入休眠,RTC 1分钟唤醒
                _nop_();

                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
                _nop_();
         

            
             SYS_Init();    //系统初始化,包含时钟、IO口、等设置

           //-------------------ADC设置--------------------------------------
            ADCTIM = 0x3f;  //设置通道选择时间、保持时间、采样时间
            ADCCFG = RES_FMT + ADC_SPEED;
            //ADC模块电源打开后,需等待1ms,MCU内部ADC电源稳定后再进行AD转换
            ADC_CONTR = 0x80 + 0;        //ADC on + channel
            delay_ms(100);
        
            
        //  提前启动温度转换,节省时间
            DS18B20_Reset();                //设备复位
            DS18B20_WriteByte(0xCC);        //跳过ROM命令
            DS18B20_WriteByte(0x44);        //开始转换命令

            
         
           //Flag_1min = 1;    //1分钟到标志
           Time_1min_Counter++;  
        
      
            TARGET_LED = 0; //指示灯亮
            delay_ms(300);
            TARGET_LED = 1; //指示灯灭


//----------------挂起这一段,唤醒、采样、显示、存储就正常工作  ,打开就不正常         

             //--------------采集ADC8通道------压力传感器-------------------
             ADC_convert(8);                //采集P00口8通道AD值
             ii=ADC_resuit[8];        //滤波计算
             ii=Adc2Voltage(ii);   //采样值转换成电压值,mV
             Pre8 = ii;             //压力值的mV数
//---------------------------------------------------------------------------------                         
                        //后续为显示、存储函数......
                       
}


/***********************************
查询方式做一次ADC, chn为通道号, chn=0~7对应P1.0~P1.7, chn=8~14对应P0.0~P0.6, chn=15对应BandGap电压.
***********************************/
void ADC_convert(u8 chn)
{
        u16        j;
        u8        k;                //平均值滤波时使用
   
   

        Get_ADC12bitResult(chn);                //参数i=0~15,查询方式做一次ADC, 切换通道后第一次转换结果丢弃. 避免采样电容的残存电压影响.
        Get_ADC12bitResult(chn);                //参数i=0~15,查询方式做一次ADC, 切换通道后第二次转换结果丢弃. 避免采样电容的残存电压影响.

    //使用冒泡排序,去掉最高值、最低值,求中间平均值

        for(k=0; k<16; k++)        ADC_Buffer[k] = Get_ADC12bitResult(chn);
        BubbleSort(ADC_Buffer,16);  //冒泡排序
        for(k=4, j=0; k<12; k++) j += ADC_Buffer[k];  //舍弃前后各4个数,取中间8个数据累加
        j = j / 8;                // 求平均
   
    ADC_resuit[chn]=j;   //结果保存在ADC_resuit[chn]
   
    //P25=0;   //灯亮
    delay_ms(10);
    //P25=1;   //灯灭
   

}

//========================================================================
// 函数: u16 Get_ADC12bitResult(u8 channel))        //channel = 0~15
// 描述: 查询法读一次ADC结果.
// 参数: channel: 选择要转换的ADC, 0~15.
// 返回: 12位ADC结果.
// 版本: V1.0, 2016-4-28
//========================================================================
u16        Get_ADC12bitResult(u8 channel)        //channel = 0~15
{
        u16 timeout = 60000;
   
    ADC_RES = 0;
    ADC_RESL = 0;

    ADC_CONTR = (ADC_CONTR & 0xF0) | channel  ; //设置ADC转换通道
    ADC_CONTR |= ADC_START;//启动ADC转换
    _nop_();
    _nop_();
    _nop_();
   
//    while((ADC_CONTR & ADC_FLAG)==0);   //waitforADCfinish
//    ADC_CONTR &= ~ADC_FLAG;     //清除ADC结束标志
   
    // 3. 等待转换完成,添加超时机制
   
    while ((ADC_CONTR & ADC_FLAG) == 0)         //之前每次在这里陷入死循环,后增加了限时离开的功能
    {
        if (timeout-- == 0)
        {
            // 超时处理:清除标志位,返回错误值(避免死循环)
            ADC_CONTR &= ~ADC_FLAG;  // 尝试清除标志
            return 0xFFFF;    // 返回错误值(0xFFFF表示ADC失败)
        }
    }        
   
    return (((u16)ADC_RES << 8) | ADC_RESL);
}//****************************** system初始化 **************************************************
void SYS_Init(void)
{
    EnableAccessXFR();                  //使能访问扩展XFR
    AccessCodeFastest();                //设置最快速度访问程序代码
    AccessIXramFastest();               //设置最快速度访问内部XDATA
    IAP_SetTimeBase();                  //设置IAP等待参数,产生1us时基

    P0M1 = 0xFF;  P0M0 = 0x00;     //设置为高阻输入,  P00--ADC通道8
    P1M1 = 0x00;  P1M0 = 0xFF;     //P1口开漏输出(驱动led、MOS管)
    P2M1 = 0x00;  P2M0 = 0x00;     //设置为准双向口
    P3M1 = 0x04;  P3M0 = 0x00;     //P32高阻输入(按键);其余设置为准双向口
    //P3M1 = 0x00;  P3M0 = 0x00;     //设置为准双向口
    P4M1 = 0x00;  P4M0 = 0x00;     //设置为准双向口
    P5M1 = 0x00;  P5M0 = 0x08;     //P53推挽输出,控制电源;其余准双向口
    P6M1 = 0x00;  P6M0 = 0x00;     //设置为准双向口
    P7M1 = 0x00;  P7M0 = 0x00;     //设置为准双向口   
       
                                     
    // P1.0和P1.1设置为推挽输出 M1M0=01
    P1M1 &= ~0x03;       // P1.0/P1.1推挽输出模式       
    P1M0 |= 0x03;                 //
    TARGET_LED = 1;             // 初始关闭信号灯
    TASK_LED = 0;             // 初始关闭MOS管

    Vcc_EN = 1;              //打开负载电源

    CLK_Init();                        //时钟模块初始化(24M进行4分频=6M)
    I2C_Init();                         //I2C初始化
    OLED_Init();                      //OLED初始化
   
    IRCDB = 0x10;   //内部高速IRC时钟,停振后被唤醒后起振,等待多少个时钟数后给MCU供应时钟
    IAP_TPS =24;    //根据系统工作频率设置此寄存器
                    //(如果系统工作频率为40MHz,则1AP_TPS设置为40;
                    //11如果系统工作频率为24MHz,则IAP_TPS设置为24)   
   
}

回复

使用道具 举报 送花

  • 打卡等级:常住居民I
  • 打卡总天数:75
  • 最近打卡:2025-08-13 11:21:33
已绑定手机

11

主题

34

回帖

427

积分

中级会员

积分
427
发表于 3 天前 | 显示全部楼层
出问题的就是ADC函数, 每次卡在这一句:while ((ADC_CONTR & ADC_FLAG) == 0)   。但是把这里面的ADC函数,换回没有休眠功能的程序里,循环执行,都没有问题。到底正常上电与休眠醒来,还有什么异同?请赐教!
回复 支持 反对

使用道具 举报 送花

  • 打卡等级:以坛为家II
  • 打卡总天数:490
  • 最近打卡:2025-08-14 08:53:38
已绑定手机

44

主题

2163

回帖

7525

积分

论坛元老

积分
7525
发表于 前天 09:14 | 显示全部楼层
附件例子ADC轮询采样中间休眠,唤醒定时器2秒钟唤醒一次。
32G系、Ai8051U-32bit模式都可直接使用,供参考。
截图202508120913156022.jpg

16路ADC轮询-串口1返回结果-中间休眠.zip

10.35 KB, 下载次数: 3

回复 支持 反对

使用道具 举报 送花

  • 打卡等级:以坛为家I
  • 打卡总天数:267
  • 最近打卡:2025-08-14 08:41:33
已绑定手机

79

主题

2890

回帖

6127

积分

荣誉版主

无情的代码机器

积分
6127
发表于 前天 09:36 | 显示全部楼层

测了下未复现,建议上传完整工程:
08.1-16路ADC轮询-每路16次ADC转换取中间8次求平均值-串口1返回结果-休眠测试.zip (70.63 KB, 下载次数: 3)
三天不学习,赶不上刘少奇~
回复 支持 反对

使用道具 举报 送花

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

本版积分规则

QQ|手机版|深圳国芯人工智能有限公司 ( 粤ICP备2022108929号-2 )

GMT+8, 2025-8-14 18:47 , Processed in 0.123024 second(s), 70 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

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