STM32实战总结:HAL之看门狗

在由单片机构成的微型计算机系统中,单片机的工作常常会受到来自外界电磁场的干扰,造成程序的跑飞,而陷入死循环;或者因为用户配置代码出现BUG,导致芯片无法正常工作。出于对单片机运行状态进行实时监测的考虑,便产生了一种专门用于监测单片机程序运行状态的模块或者芯片,俗称“看门狗”(watchdog) 

简单说:看门狗的本质就是定时计数器,计数器使能之后一直在递增/递减,而喂狗就是重新写入计数器的值,使计数器重新递增/递减。

如果在一定时间内没有接收到喂狗信号(表示MCU已经挂了),便实现处理器的自动复位重启(发送复位信号)

所以,通常需要在看门狗定时器使能复位之前喂狗。

 STM32中的看门狗

在STM32中的看门狗具体是什么样的呢?

STM32F10xxx内置两个看门狗,独立看门狗和窗口看门狗。

STM32内置两个看门狗,提供了更高的安全性、时间的精确性和使用的灵活性。两个看门狗设备(独立看门狗、窗口看门狗)可以用来检测和解决由软件错误引起的故障。当计数器达到给定的超时值时,触发一个中断(仅适用窗口看门狗)或者产生系统复位。

独立看门狗(IWDG)由专用的低速时钟(LSI)驱动,即使主时钟发生故障它也仍然有效。因为是独立于主时钟的,所以这就是为什么被称为独立看门狗。

窗口看门狗由从APB1时钟分频后得到的时钟驱动,通过可配置的时间窗口来检测应用程序非正常的过迟或过早的操作。

注意二者的异同:

✔都是为了保证CPU程序在跑飞时可以进行处理;

IWDG最适合应用于那些需要看门狗作为一个在主程序之外,能够完全独立工作,并且对时间精度要求较低的场合(因为LSI其实并不精确)。WWDG最适合那些要求看门狗在精确计时窗口起作用的应用程序;

✔窗口看门狗严格限定喂狗时间段,独立看门狗则是只要没有到时间,都能喂狗。

独立看门狗

根据内部看门狗时钟频率,装载寄存器定一个时间值,比如是1000,那么独立看门狗就会按照时钟频率,从1000开始向下每隔一个时钟周期减1,如果在减到0之前,你用程序代码重新向向下计数器里面写1000(喂狗),那么定时器会重新从1000开始向下递减。如果在减到0的时候,你还没有喂狗(用新的数值覆盖计数器),就会产生复位信号。

在51单片机中,就只有独立看门狗。

IWDG主要性能

● 自由运行的递减计数器

● 时钟由独立的RC振荡器提供(可在停止和待机模式下工作)

● 看门狗被激活后,则在计数器计数至0x000时产生复位

独立看门狗框图如下:

注意:

如果我们要做低功耗模式,就要关闭独立看门狗,因为停机或者待机模式下不会喂狗,但是独立看门狗仍然正常工作,会导致系统复位而退出低功耗模式。

MX配置

看门狗是芯片内部设计,无外部引脚。

IWDG相关函数

IWDG除了初始化函数外,只有一个函数。

/** @defgroup IWDG_Exported_Functions_Group2 IO operation functions
  * @{
  */
/* I/O operation functions ****************************************************/
HAL_StatusTypeDef     HAL_IWDG_Refresh(IWDG_HandleTypeDef *hiwdg);

就是一个喂狗函数。

关键代码

/* Includes ------------------------------------------------------------------*/
#include "MyApplication.h"

/* Private define-------------------------------------------------------------*/

/* Private variables----------------------------------------------------------*/

/* Private function prototypes------------------------------------------------*/  
static void FeedDog(void);

/* Public variables-----------------------------------------------------------*/
MyIWDG_t MyIWDG = 
{
	TRUE,
	
	FeedDog
};    

/*
	* @name   FeedDog
	* @brief  喂狗
	* @param  None
	* @retval None      
*/
static void FeedDog(void)
{
	HAL_IWDG_Refresh(&hiwdg);
}
/********************************************************
  End Of File
********************************************************/

一开始,我尝试409.6ms喂一次狗,可系统还是复位了。

于是我在想,会不会从开始启动看门狗,到喂狗程序之前,时间就已经超过了409.6ms呢?

所以,我把时间做些调整。 

改成了32分频,计数2500,即2秒。

然后就可以了。

注意,喂狗不能只喂一次,需要不断地喂,所以不能就放在初始化代码中喂一次。

其实最好用定时器中断,每隔固定的时间喂狗。

窗口看门狗

配置寄存器(WWDG_CFR) 中包含窗口比较值:要避免产生复位,递减计数器必须在其值

小于窗口寄存器(也就是配置寄存器)的数值并且大于0x3F时被重新装载。

也就是说,只有递减计数器的值在[W[6:0], 0x40]区间内被重新装载时,才不会产生复位。

那什么时候会复位呢?

如果看门狗被启动 (WWDG_CR 寄存器中的 WDGA 位被置 ’1’),则:

1、当7(T[6:0])递减计数器从0x40翻转到0x3F(T6位清零)时,将产生一个复位;

2、如果软件在计数器值大于窗口寄存器中的数值时重新装载计数器,将产生一个复位;

T6位变成0,也就意味着数值降到了64以下。

1000000就是64,如果最高位变成0,则就是从40H翻转到了3FH,会产生一个MCU复位。

这只是一种算法技巧。

0x40对应的十进制数值是64,窗口的下限,是固定的值,不能改变。

窗口看门狗计数器的值必须在上窗口和下窗口之间才可以刷新(喂狗),这也是窗口看门狗中“窗口”两个字的含义。

产生复位时,T6位会被置0

看门狗的超时时间:指的是计数器减至0x3F的时间。

计数器位数:

窗口值位数:

可见,计数值和窗口值范围为0—127,又因为窗口下限值为64,所以实际可设置的范围为64到127。

窗口看门狗中断:

窗口看门狗还可以使能提前唤醒中断,那么在计数器由减到0x40  (0x3f+1)  的时候,便会进入中断,我们可以在中断里面喂狗 。

HAL库独立窗口狗函数库讲解:

看门狗初始化:

HAL_WWDG_Init(WWDG_HandleTypeDef *hwwdg);

喂狗:

HAL_WWDG_Refresh(WWDG_HandleTypeDef *hwwdg);

看门狗中断处理函数:

HAL_WWDG_IRQHandler(WWDG_HandleTypeDef *hwwdg);


看门狗中断回调函数:

 __weak HAL_WWDG_EarlyWakeupCallback(hwwdg);

MX配置

参数从上到下为:

  • 分频系数;
  • 窗口值;
  • 初始值;
  • 使能提前唤醒中断;

关键代码

/* Includes ------------------------------------------------------------------*/
#include "MyApplication.h"

/* Private define-------------------------------------------------------------*/

/* Private variables----------------------------------------------------------*/

/* Private function prototypes------------------------------------------------*/  
static void FeedDog(void);

/* Public variables-----------------------------------------------------------*/
MyWWDG_t MyWWDG = 
{
	TRUE,
	
	FeedDog
};    

/*
	* @name   FeedDog
	* @brief  喂狗
	* @param  None
	* @retval None      
*/
static void FeedDog(void)
{
	HAL_WWDG_Refresh(&hwwdg);
}
/********************************************************
  End Of File
********************************************************/
/**
  * @name   HAL_WWDG_EarlyWakeupCallback
  * @brief  窗口看门狗提前唤醒中断回调函数
  * @param  hwwdg : pointer to a WWDG_HandleTypeDef structure that contains
  *                the configuration information for the specified WWDG module.
  * @retval None
  */
void HAL_WWDG_EarlyWakeupCallback(WWDG_HandleTypeDef *hwwdg)
{
  //喂狗
	if(MyWWDG.FeedDog_Flag == TRUE)
	{
		printf("喂狗\r\n");
		MyWWDG.FeedDog();
	}
}

  • 3
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值