STOP mode唤醒之后就是MSI作为系统时钟,如果你之前不是用MSI,你不需要重新配置?
官方的例程里早就写明了需要重新配置时钟的,而且,必须要跟你启动时的时钟一样才有意义。
现在问题好像不是时钟问题,是ADC读取的问题了。唤醒后,读取ADC是失败的。
https://www.stmcu.org.cn/module/forum/data/attachment/forum/201808/28/223840c7ed2z4ezeem823y.png
butterflyspring 发表于 2018-8-29 11:47
STOP mode唤醒之后就是MSI作为系统时钟,如果你之前不是用MSI,你不需要重新配置?
我现在可以确定,确实是因为添加了stop模式之后出的问题,如果屏蔽掉stop模式 就没问题,每次需要复位adc才恢复正常,然后又运行一段时间又出问题
void work_200ms(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //打开复用功能和adc时钟
if(Press_set==0)//如果处于暂停状态进入停机
{
TIM_SetAutoreload(TIM1,100);//改频率
TIM_SetCompare4(TIM1,0);//改占空比
Ledless_L;
warn_voice(off,100,500);//关闭漏气报警
warn_time=0;
stop_mode();
}
if(Press_set==125&&time2_flag==1&&press_end==1)//如果处于工作状态并且 达到200ms定时,并且 压力达到设定值进入停机
{ TIM_Cmd(TIM2, DISABLE); //关闭计时
time2_flag=0;
work_time=0;
debug_string("Press_adc:");
debug_16bit_assic(Press_adc);
debug_string("Vbat_adc:");
debug_16bit_assic(Vbat_adc);
//debug_string("Vref_adc:");
// debug_16bit_assic(Vref_adc);
//200ms工作完成,清除ad数据,下次唤醒重新采集
ready_num=0;
//adc_num=0;
//Adc_ready=0;
//清场完毕,打开rtc时钟,进入睡眠
if(Stop_flag==0)
{
RTC_init(); //配置rtc
RTC_EXTI_INITIAL(ENABLE);//打开闹钟中断
Stop_flag=1;
stop_mode();
}
}
Ledrun_L;
} 我有在唤醒后重新配置了时钟,唤醒后第一时间配置的时钟,和开机复位刚运行是同一个时钟函数
void clk_init(void)
{ ErrorStatus HSEStartUpStatus;
RCC_DeInit(); //RCC寄存器设置为默认值
RCC_HSEConfig(RCC_HSE_ON); //打开高速外部时钟
HSEStartUpStatus=RCC_WaitForHSEStartUp (); //等待时钟起振
if(HSEStartUpStatus == SUCCESS)
{
RCC_HCLKConfig(RCC_SYSCLK_Div1); //AHB 时钟为系统时钟
RCC_PCLK2Config(RCC_HCLK_Div1); //高速AHB总线时钟为系统时钟
RCC_PCLK1Config(RCC_HCLK_Div2); //低速AHB总线为1/2系统时钟
FLASH_SetLatency(FLASH_Latency_2);
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);//预取指缓存使能
RCC_PLLConfig(RCC_PLLSource_HSI_Div2, RCC_PLLMul_9); //设置PLL输入时钟分频和PLL 倍频系数
RCC_PLLCmd(ENABLE);
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY)==RESET);
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //设置pll输出时钟为系统时钟
while(RCC_GetSYSCLKSource()!=0x08);
}
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //打开复用功能和adc时钟
}
void out_stop(void)
{
clk_init();
USART_Cmd(USART1, ENABLE);
work_time=0;
time2_flag=0;
debug_string("wake\n");
}
void stop_mode(void)
{
debug_string("in stop\n");
USART_Cmd(USART1, DISABLE);
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR,ENABLE);
//PWR_EnterSTOPMode(PWR_Regulator_LowPower,PWR_STOPEntry_WFI);
}
这是进入和退出停机模式的函数,请高手指教,停机模式里我屏蔽掉pwr那条指令,程序运行就不会出问题,加上这个过段时间就adc数据就保持不变了 程序重新规划,只写了一个最简单定时一秒进入闹钟,唤醒 后立即停机
RTC_init();//闹钟初始化
while(1)
{
stop_mode();
if(Stop_flag==0)
{
debug_16bit_assic(sleep_num);
debug_16bit_assic(ADC_ConvertedValue);
}
}
adc的死机问题也是存在
有兴趣的朋友希望自己试验一下,谢谢大家 请问这个问题怎么解决的,我也是低功耗唤醒后ADC出错 自己回复下解决方案吧,就是每次唤醒之后要对adc重新初始化,否则adc死机问题还是存在,但是又会带来一个新的问题,刚初始化后数值稳定性会差些 在进入低功耗前,也需要对ADC模块进行相应的关闭,好像没找到相关部分。建议增加相关部分。
页:
1
[2]