STM32独立看门狗和低功耗模式_RTC定时唤醒来喂狗

发布者:SparklingMelody最新更新时间:2023-09-21 来源: elecfans关键字:STM32  独立看门狗  低功耗模式  RTC  定时唤醒  喂狗 手机看文章 扫描二维码
随时随地手机看文章

在STM32开发中经常会用到独立看门狗(IWDG)和低功耗模式,看门狗是为了检测和解决由软件错误引起的故障,低功耗模式是为了在CPU不需要继续运行时进入到休眠模式用以节省电能。其中独立看门狗的时钟由独立的RC振荡器(STM32F10x一般为40kHz)提供,即使在主时钟出现故障时,也仍然有效,因此可以在停止和待机模式下工作。而且独立看门狗一旦启动,除了系统复位,它不能再被停止。但这样引发的一个问题是当MCU进入到低功耗模式后由于CPU停止运行无法喂狗,会导致系统频繁复位。那如何解决这个问题呢,难道独立看门狗和低功耗模式没法同时使用?


一个很好的方式是在休眠模式下通过RTC定时唤醒来喂狗,喂完够在进入继续进入到休眠模式。比如看门狗复位的时间间隔为10s。那么在进入休眠模式前设置RTC闹钟中断时间为5s。这样每隔5s唤醒一次喂一次狗。便可以很好的解决这个问题。


while(1)  

{  

// 执行任务  

Task1();  

Task2();  

// ..  

// 喂狗  

dev_iwdg_feed();  

// 进入待机模式开关  

if(m_bEnterStandByMode)  

{     

// 使能外部中断,GPIOB3,用以MCU从待机模式唤醒  

dev_exti_enable(TRUE);  

ENTERSTOPMODE:    

// 设置RTC闹钟,5秒钟产生一次RTC闹钟中断*/  

dev_rtc_setAlarm(5);  

// 进入停止模式(低功耗),直至外部中断触发时被唤醒  

PWR_EnterSTOPMode(PWR_Regulator_LowPower, PWR_STOPEntry_WFI);  

// 是否是RTC闹钟中断唤醒  

if(dev_rtc_isAlarm())  

{  

// 喂狗  

dev_iwdg_feed();  

// 喂完狗继续进入停止模式  

goto ENTERSTOPMODE;   

}  

// 禁止外部中断   

dev_exti_enable(FALSE);  

// 从停止模式唤醒后恢复系统时钟  

dev_clk_restore();  

}               

}  

以下是完整的参考代码:

//**********************************************************************************************       

//  STM32F10x StopMode RTC Feed Dog   

//  compiler: Keil UV3       

//  2013-01-04 , By friehood       

//**********************************************************************************************    

#include "stm32f10x_lib.h"  

#include "platform_config.h"  

static Boolean g_bRTCAlarm = FALSE;  

/*******************************************************************************  

* Function Name  : RCC_Configuration  

* Description    : Configures the different system clocks.  

* Input          : None  

* Output         : None  

* Return         : None  

*******************************************************************************/  

void RCC_Configuration(void)  

{  

/* RCC system reset(for debug purpose) */  

RCC_DeInit();  

/* Enable HSE */  

RCC_HSEConfig(RCC_HSE_ON);  

/* Wait till HSE is ready */  

if(RCC_WaitForHSEStartUp() == SUCCESS)  

{  

/* Enable Prefetch Buffer */  

FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);  

//FLASH时序控制   

//推荐值:SYSCLK = 0~24MHz   Latency=0   

//        SYSCLK = 24~48MHz  Latency=1   

//        SYSCLK = 48~72MHz  Latency=2  

//FLASH_SetLatency(FLASH_Latency_1);        //警告:修改为1会对DMA值有影响(如ADC采集值会错位)  

FLASH_SetLatency(FLASH_Latency_2);  

/* HCLK = SYSCLK */  

RCC_HCLKConfig(RCC_SYSCLK_Div1);   

/* PCLK2 = HCLK */  

RCC_PCLK2Config(RCC_HCLK_Div1);   

/* PCLK1 = HCLK/2 */  

RCC_PCLK1Config(RCC_HCLK_Div2);  

/* PLLCLK = 12MHz * 3 = 36 MHz */  

RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_3);  

/* Enable PLL */   

RCC_PLLCmd(ENABLE);  

/* Wait till PLL is ready */  

while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)  

{  

}  

/* Select PLL as system clock source */  

RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);  

/* Wait till PLL is used as system clock source */  

while(RCC_GetSYSCLKSource() != 0x08)  

{  

}  

}  

/* Enable PWR and BKP clock */  

RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR | RCC_APB1Periph_BKP, ENABLE);  

/* Enable AFIO clock */  

RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO, ENABLE);  

}  

/******************************************************************************* 

* Function Name  : NVIC_Configuration 

* Description    : Configures the nested vectored interrupt controller. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void NVIC_Configuration(void)  

{  

NVIC_InitTypeDef NVIC_InitStructure;  

#ifdef  VECT_TAB_RAM  

/* Set the Vector Table base location at 0x20000000 */  

NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0);  

#else  /* VECT_TAB_FLASH  */  

/* Set the Vector Table base location at 0x08000000 */  

NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);  

#endif  

/* Configure one bit for preemption priority */  

NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);  

}  

/******************************************************************************* 

* Function Name  : SysTick_Configuration 

* Description    : Configures the SysTick to generate an interrupt each 1 millisecond. 

* Input          : None 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void SysTick_Configuration(void)  

{  

/* Select AHB clock(HCLK) as SysTick clock source */  

SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK);  

/* Set SysTick Priority to 3 */  

NVIC_SystemHandlerPriorityConfig(SystemHandler_SysTick, 3, 0);  

/* SysTick interrupt each 1ms with HCLK equal to 72MHz */  

SysTick_SetReload(72000);  

/* Enable the SysTick Interrupt */  

SysTick_ITConfig(ENABLE);  

}  

/******************************************************************************* 

* Function Name  : Delay 

* Description    : Inserts a delay time. 

* Input          : nTime: specifies the delay time length, in milliseconds. 

* Output         : None 

* Return         : None 

*******************************************************************************/  

void Delay(u32 nTime)  

{  

/* Enable the SysTick Counter */  

SysTick_CounterCmd(SysTick_Counter_Enable);  

TimingDelay = nTime;  

while(TimingDelay != 0);  

/* Disable the SysTick Counter */  

SysTick_CounterCmd(SysTick_Counter_Disable);  

/* Clear the SysTick Counter */  

SysTick_CounterCmd(SysTick_Counter_Clear);  

}  

/******************************************************************************* 

* Function Name  : RTC_Configuration 

* Description    : Configures RTC clock source and prescaler. 

* Input          : None 

* Output : None

void RTC_Configuration(void)

{

EXTI_InitTypeDef EXTI_InitStructure;

NVIC_InitTypeDef NVIC_InitStructure;

/* RTC clock source configuration ------------------------------------------*/

/* Allow access to BKP Domain */

PWR_BackupAccessCmd(ENABLE);

/* Reset Backup Domain */

BKP_DeInit();

/* Enable the LSI OSC */

RCC_LSICmd(ENABLE);

/* Wait till LSI is ready */

while (RCC_GetFlagStatus(RCC_FLAG_LSIRDY) == RESET){}

/* Select the RTC Clock Source */

RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);

/* Enable the RTC Clock */

RCC_RTCCLKCmd(ENABLE);

/* RTC configuration -------------------------------------------------------*/

/* Wait for RTC APB registers synchronisation */

RTC_WaitForSynchro();

/* Set RTC prescaler: set RTC period to 1sec */

RTC_SetPrescaler(40000);

/* Wait until last write operation on RTC registers has finished */

RTC_WaitForLastTask();

/* Enable the RTC Alarm interrupt */

RTC_ITConfig(RTC_IT_ALR, ENABLE);

/* Wait until last write operation on RTC registers has finished */

RTC_WaitForLastTask();

/* Configure EXTI Line17(RTC Alarm) to generate an interrupt on rising edge */

EXTI_ClearITPendingBit(EXTI_Line17);

EXTI_InitStructure.EXTI_Line = EXTI_Line17;

EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;

EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Rising;

EXTI_InitStructure.EXTI_LineCmd = ENABLE;

EXTI_Init(&EXTI_InitStructure);

/* Enable the RTC Interrupt */

NVIC_InitStructure.NVIC_IRQChannel = RTCAlarm_IRQChannel;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

NVIC_Init(&NVIC_InitStructure);

}

/*******************************************************************************

* Function Name : RTCAlarm_IRQHandler

* Description : This function handles RTC Alarm interrupt request.

* Input : None

* Output : None

* Return : None

*******************************************************************************/

void RTCAlarm_IRQHandler(void)

{

if(RTC_GetITStatus(RTC_IT_ALR) != RESET)

{

/* Set the RTC alarm flag */

g_bRTCAlarm = TRUE;

/* Clear EXTI line17 pending bit */

EXTI_ClearITPendingBit(EXTI_Line17);

/* Check if the Wake-Up flag is set */

if(PWR_GetFlagStatus(PWR_FLAG_WU) != RESET)

{

/* Clear Wake Up flag */

PWR_ClearFlag(PWR_FLAG_WU);

}

/* Wait until last write operation on RTC registers has finished */

RTC_WaitForLastTask();

/* Clear RTC Alarm interrupt pending bit */

RTC_ClearITPendingBit(RTC_IT_ALR);

/* Wait until last write operation on RTC registers has finished */

RTC_WaitForLastTask();

}

}

/*******************************************************************************

* Function Name : dev_rtc_setAlarm

* Description : 设置RTC闹钟。

* Input : 闹钟时间

* Output : None

* Return : None

*******************************************************************************/

void dev_rtc_setAlarm(u32 AlarmValue)

{

/* Clear the RTC SEC flag */

RTC_ClearFlag(RTC_FLAG_SEC);

/* Wait clear RTC flag sccess */

while(RTC_GetFlagStatus(RTC_FLAG_SEC) == RESET);

/* Wait until last write operation on RTC registers has finished */

RTC_WaitForLastTask();

/* Sets the RTC alarm value */

RTC_SetAlarm(RTC_GetCounter() + AlarmValue);

/* Wait until last write operation on RTC registers has finished */

RTC_WaitForLastTask();

}

/*******************************************************************************

[1] [2]
关键字:STM32  独立看门狗  低功耗模式  RTC  定时唤醒  喂狗 引用地址:STM32独立看门狗和低功耗模式_RTC定时唤醒来喂狗

上一篇:以STM32单片机为控制系统核心的数据记录装置设计
下一篇:STM32F3系列MCU外围元器件配置参考(附BOM表)

推荐阅读最新更新时间:2024-11-20 17:29

STM32 IO口模拟串口通讯
前阵子,调项目时需要用到低波特率串口通讯(300的波特率),才发下发现在正常情况下(PCLK1时钟频率为72M,PCLK2时钟频率为36M):STM32的USART0的最低波特率只能设置到1200,;而USART1最低波特率只能设置到600。怎么设置STM32的600或以下的波特率呢?有两种方法:一种是改变外设时钟频率,而另一种方法就是使用IO口模拟串口通讯。今天就来讲讲,用IO口模拟串口通信! 1、串口传输协议 首先,必须要知道串口通讯时数据是怎样传输的?这里以异步传输字符为例子,如下图所示: 一般字符传输都采用:1位起始位,8位数据位,1位停止位,没有校验位 的形式传输,其他形式的这里不讲。串口异步传输在空闲状态时都必须是
[单片机]
<font color='red'>STM32</font> IO口模拟串口通讯
stm32 软件复位
根据《CM3权威指南》,软件复位有两种方法: 1、通过置位NVIC中应用程序中断与复位控制寄存器(AIRCR)的VECTRESET位: LDR R0, =0xE000ED0C ; NVIC AIRCR address LDR R1, =0x05FA0001 ; 置位 VECTRESET位,前面的0x05FA是访问钥匙 STR R1, ; 触发复位序列 deadloop B deadloop ; 该死循环保证后面的指令不可能被执行到 这种复位的作用范围覆盖了整个CM3 处理器中,除了调试逻辑之外的所有角落,但是它不会影响到CM3 处理器外部的任何电路,所以单片机上的各片上外设和其它电路都不受 影响。 2、置位同一个寄存器的SYSR
[单片机]
stm32实现LED灯亮灭
一、概述 本章中,不特殊说明都以stm32F103VET6为例,软件平台为iar6.4 通过查看原理图,我们的LED指示灯配置在PB8、PB9管脚上,通过配置GPIOB时钟及引脚来控制指示灯的亮灭。 二、本章学习目标 学会查看原理图 理解stm32端口配置 学会操作使用iar编程工具 三、GPIO端口配置 1.配置开启GPIO时钟 STM32上电时外设时钟默认不开启,用要时需要先开启时钟,本例开启GPIOA和GPIOB的时钟。 参考代码: RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2
[单片机]
<font color='red'>stm32</font>实现LED灯亮灭
STM32 Systick定时器在实现1us延时的问题与解决
问题: 使用systick_config()函数来实现计数,这个函数在下面代码中的 SysTick_CTRL_TICKINT_Msk 开启了中断。不论系统时钟为72Mhz或36Mhz若设置STM32每10us进入一次中断,计时是可以的;而每1us进入中断,由于中断指令较多,那么程序就会困在中断里出不来。 static __INLINE uint32_t SysTick_Config(uint32_t ticks) { if (ticks SysTick_LOAD_RELOAD_Msk) return (1); /* Reload value impossible */
[单片机]
stm32之DMA中断
AD转换之DMA 1、DMA的配置 //DMA的配置 void DMA_Configuration(void) { /* 允许 DMA1 */ RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); /* DMA通道1*/ DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr =(u32)( &(ADC1- DR)); //ADC1数据寄存器 DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADCCov; //获取ADC的数组 DMA_InitStructur
[单片机]
STM32实战案例分享:剖析STM32应用在电源项目上常见的那些难题
我们在从事STM32单片机的应用开发及调试过程中,往往会碰到各类异常。其中有不少比例的问题跟电源有关。对于一个电子产品而言,电源部分很关键、很重要,但在实际开发调试中,我们偶尔会有意无意的忽视它。这里分享几个实际案例,以加强刺激,加深印象。 毕竟因为电源问题可能导致的异常很多很多,这里分享几个案例算是抛砖引玉,希望大家在调试中对电源方面加以重视。个人认为,往往电源出问题时导致的异常时并不太好分析,多数时候异常表现得更为诡异或没章法。【注:下面提到的案例中异常原因都与电源有关,但并不是说出现类似异常时一定是电源的原因。】 下面主要分享几个基于STM32应用的案例。 案例1:ADC功能异常 某人使用STM32芯片的ADC功能
[单片机]
<font color='red'>STM32</font>实战案例分享:剖析<font color='red'>STM32</font>应用在电源项目上常见的那些难题
关于STM32 ST-LINK Utility查看内核运行状态,助你判断程序是否跑飞
本文围绕STM32 ST-LINK Utility讲几点主要功能及相关拓展知识: 1.STM32编程下载; 2.利用该编程工具查看内核运行状态; 3.Option Bytes选项字配置; 1STM32编程下载 STM32 ST-LINK Utility一个最重要的功能就是对STM32进行编程。支持常见的hex、bin文件,还有早期摩托罗拉定义的srec和s19格式的文件(说实话,我都不了解这两种格式的文件)。 这里主要想提示一下初学者:hex带有地址,而bin文件不带地址,下载时需要填写起始地址。 下面看两张在STM32 ST-LINK Utility中下载选择文件的图大家就明白了。 1.选择hex下载,地址不可修改(灰色)
[单片机]
关于<font color='red'>STM32</font> ST-LINK Utility查看内核运行状态,助你判断程序是否跑飞
STM32的瞬态运动参数存储测试系统设计
引言 存储测试技术是在特殊环境下记录运动物体参数行之有效的方法,先将测试数据存入存储器,待装置回收后通过特定接口与上位机进行通信,还原数据信息。在诸多领域的测试中,对数据采集存储系统的实时性和功耗提出了更高的要求,随着半导体技术的发展,各种技术的进步使得高速度、低功耗的存储测试系统能够实现。 本系统选择ST公司超低功耗的基于ARM Cortex M3四核的处理器STM32F103C8T6作为核心控制元件,采取内部A/D转换器与铁电存储器结合的方法,实现压阻式加速度传感器测试数据的采集、存储,并利用LabView开发平台设计上位机应用软件实现测试数据的USB回读及处理。 1 系统原理 存储测试系统由电源管理模块、调理模块、外部晶
[单片机]
<font color='red'>STM32</font>的瞬态运动参数存储测试系统设计
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved