本帖最后由 495392996 于 2016-12-25 00:29 编辑
这两天搞了下待机模式玩玩。但是突然发现在上电时进入待机模式可以进行唤醒。但是唤醒后再次进入待机模式后就不能唤醒了。很奇怪。本以为代码问题就把代码移植到STM32上试。经测试在STM上是正常的。为什么MM32就不行呢?是否还存在什么设置吗?还有就是ADC功能的电位器是否连接错了?怎么可能连接到PA0上呢?貌似是连接到PA1上,刚发现由于没有万用表不敢下此定论是否连接到PA1上。但是这个板子确实问题越来越多了。还需改进的地方还是不少。
现在的程序是上电初始化后进入待机模式。按下3秒后唤醒。再次按下3秒就会再次进入待机模式并复位所有IO口时钟。IO口时钟复位后不是高阻态吗?为何我的程序复位了所有的IO口时钟后LED还是会亮呢?主要的是为何不能再次唤醒了呢?这次真想不明白了。实在找不出原因了。是否哪里出了错?望大神指导!
接下来上程序。少的程序直接截图。
#include "wkup.h"
#include "led.h"
#include "delay.h"
#include "usart.h"
/////////////////////////////////////////////////////////////////////////////////
void Sys_Standby(void)
{
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE); //使能PWR外设时钟
PWR_WakeUpPinCmd(ENABLE); //使能唤醒管脚功能
printf ("进入待机模式 \r\n");
PWR_EnterSTANDBYMode(); //进入待命(STANDBY)模式
}
//系统进入待机模式
void Sys_Enter_Standby(void)
{
RCC_APB2PeriphResetCmd(0X01FC,DISABLE); //复位所有IO口
Sys_Standby();
}
//检测WKUP脚的信号
//返回值1:连续按下3s以上
// 0:错误的触发
u8 Check_WKUP(void)
{
u8 t=0; //记录按下的时间
PBout(4)=0; //亮灯DS0
delay_ms (10);
printf ("按键按下\r\n");
while(1)
{
if(WKUP_KD)
{
t++; //已经按下了
delay_ms(30);
if(t>=100) //按下超过3秒钟
{
PBout(4)=0; //点亮DS0
printf ("按键按下3S\r\n");
return 1; //按下3s以上了
}
}else
{
PBout(4)=1;
return 0; //按下不足3秒
}
}
}
//中断,检测到PA0脚的一个上升沿.
//中断线0线上的中断检测
void EXTI0_IRQHandler(void)
{
EXTI_ClearITPendingBit(EXTI_Line0); // 清除LINE10上的中断标志位
if(Check_WKUP())//关机?
{
printf ("关机\r\n");
Sys_Enter_Standby();
}
}
//PA0 WKUP唤醒初始化
void WKUP_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
EXTI_InitTypeDef EXTI_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);//使能GPIOA和复用功能时钟
GPIO_InitStructure.GPIO_Pin =GPIO_Pin_0; //PA.0
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IPD;//下拉输入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化IO
//使用外部中断方式
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0); //中断线0连接GPIOA.0
EXTI_InitStructure.EXTI_Line = EXTI_Line0; //设置按键所有的外部线路
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //设外外部中断模式:EXTI线路为中断请求
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; //上升沿触发
EXTI_InitStructure.EXTI_LineCmd = ENABLE;
EXTI_Init(&EXTI_InitStructure); // 初始化外部中断
NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn; //使能按键所在的外部中断通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2; //先占优先级2级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; //从优先级2级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //使能外部中断通道
NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器
printf ("\r\n初始化完成\r\n");
if(Check_WKUP()==0) Sys_Standby(); //不是开机,进入待机模式
}
本帖最后由 495392996 于 2016-12-25 00:29 编辑
这两天搞了下待机模式玩玩。但是突然发现在上电时进入待机模式可以进行唤醒。但是唤醒后再次进入待机模式后就不能唤醒了。很奇怪。本以为代码问题就把代码移植到STM32上试。经测试在STM上是正常的。为什么MM32就不行呢?是否还存在什么设置吗?还有就是ADC功能的电位器是否连接错了?怎么可能连接到PA0上呢?貌似是连接到PA1上,刚发现由于没有万用表不敢下此定论是否连接到PA1上。但是这个板子确实问题越来越多了。还需改进的地方还是不少。
现在的程序是上电初始化后进入待机模式。按下3秒后唤醒。再次按下3秒就会再次进入待机模式并复位所有IO口时钟。IO口时钟复位后不是高阻态吗?为何我的程序复位了所有的IO口时钟后LED还是会亮呢?主要的是为何不能再次唤醒了呢?这次真想不明白了。实在找不出原因了。是否哪里出了错?望大神指导!
接下来上程序。少的程序直接截图。
#include "wkup.h"
#include "led.h"
#include "delay.h"
#include "usart.h"
/////////////////////////////////////////////////////////////////////////////////
void Sys_Standby(void)
{
RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR, ENABLE); //使能PWR外设时钟
PWR_WakeUpPinCmd(ENABLE); //使能唤醒管脚功能
printf ("进入待机模式 \r\n");
PWR_EnterSTANDBYMode(); //进入待命(STANDBY)模式
}
//系统进入待机模式
void Sys_Enter_Standby(void)
{
RCC_APB2PeriphResetCmd(0X01FC,DISABLE); //复位所有IO口
Sys_Standby();
}
//检测WKUP脚的信号
//返回值1:连续按下3s以上
// 0:错误的触发
u8 Check_WKUP(void)
{
u8 t=0; //记录按下的时间
PBout(4)=0; //亮灯DS0
delay_ms (10);
printf ("按键按下\r\n");
while(1)
{
if(WKUP_KD)
{
t++; //已经按下了
delay_ms(30);
if(t>=100) //按下超过3秒钟
{
PBout(4)=0; //点亮DS0
printf ("按键按下3S\r\n");
return 1; //按下3s以上了
}
}else
{
PBout(4)=1;
return 0; //按下不足3秒
}
}
}
//中断,检测到PA0脚的一个上升沿.
//中断线0线上的中断检测
void EXTI0_IRQHandler(void)
{
EXTI_ClearITPendingBit(EXTI_Line0); // 清除LINE10上的中断标志位
if(Check_WKUP())//关机?
{
printf ("关机\r\n");
Sys_Enter_Standby();
}
}
//PA0 WKUP唤醒初始化
void WKUP_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
EXTI_InitTypeDef EXTI_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);//使能GPIOA和复用功能时钟
GPIO_InitStructure.GPIO_Pin =GPIO_Pin_0; //PA.0
GPIO_InitStructure.GPIO_Mode =GPIO_Mode_IPD;//下拉输入
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化IO
//使用外部中断方式
GPIO_EXTILineConfig(GPIO_PortSourceGPIOA, GPIO_PinSource0); //中断线0连接GPIOA.0
EXTI_InitStructure.EXTI_Line = EXTI_Line0; //设置按键所有的外部线路
EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //设外外部中断模式:EXTI线路为中断请求
EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising; //上升沿触发
EXTI_InitStructure.EXTI_LineCmd = ENABLE;
EXTI_Init(&EXTI_InitStructure); // 初始化外部中断
NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn; //使能按键所在的外部中断通道
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2; //先占优先级2级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2; //从优先级2级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //使能外部中断通道
NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器
printf ("\r\n初始化完成\r\n");
if(Check_WKUP()==0) Sys_Standby(); //不是开机,进入待机模式
}