- 打卡等级:常住居民I
- 打卡总天数:75
- 最近打卡:2025-08-13 11:21:33
已绑定手机
中级会员
- 积分
- 427
|
用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)
}
|
|