STM32中的独立看门狗和窗口看门狗

在早期的MCU中是没有看门狗这种东西的,所以产品就很容易出现死机,跑飞的情况。为了避免这种情况的出现,后期的MCU都集成了看门狗的功能。但是,目前看门狗发展到今天基本上分为两大类:独立看门狗和窗口看门狗

  • 独立看门狗使用的是外部时钟,即使主频不工作了,看门狗也能正常工作。只要在到达喂狗时间的上限前喂狗即表示程序是正常的,这点和窗口看门狗是有区别的。另外独立看门狗是独立于整个系统之外的,这也是独立看门狗名字的由来,他有自己独立的时钟,不受整个系统的影响,所以独立看门狗主要用来监控硬件上的错误。
  • 窗口看门狗使用芯片内部时钟。喂狗的时间既有上限又有下限,即喂狗太早或者太晚都不行,比如我要求你在0.8s到0.9s内完成喂狗动作,如果你在0.8s之前或者在0.9s之后喂狗都是不可以的,都会认为MCU出现了异常,从而复位MCU。窗口看门狗是系统内部故障探测器,如果系统时钟出现了错误,那么窗口看门狗也就失去了作用,主要用于监视软件的错误。

一、独立看门狗

看完上面的简单介绍,相信大家对于独立看门狗已经有了一定的了解。下面,我们就详细地给大家讲解一下独立看门狗,以及独立看门狗的实现原理。

在了解独立看门狗之前我想大家还是需要先了解一下看门狗到底是来干什么的,在由单片机构成的微机系统中,由于单片机工作常常会受到来自外界电磁场干扰导致程序跑飞,陷入死循环——即程序正常运行被打断,系统无法继续工作。

这种情况下会造成系统陷入停滞状态,发生不可预料的后果。因此出于对单片机运行状态进行实时监测的考虑,产生了一种专门用于监测单片机程序运行状态的模块或芯片,称为看门狗。

这里以大家熟悉的STM32为例给大家讲解一下独立看门狗的配置以及工作过程。STM32F10xxx内置两个看门狗:独立看门狗和窗口看门狗,提供了更高的安全性、时间的精确性和使用的灵活性。

注意:0对应4分频,1对应8分频,…所以时间计算公式要乘以4,为:

Tout=(4 * 2^PR * (RL+1))/40 ms

最小时间单位:1/40 ms

考虑分频系数 Prescaler(4-256):1*Prescaler/40 ms

考虑RL值(0-FFF):1Prescaler(RL+1)/40 ms

STM32中的独立看门狗时通过向键值寄存器(IWDG_KR)写入0xCCCC来进行配置的,当开启了独立看门狗之后其计数器就开始从0xFFF递减计数。当计数器计数到末尾0x000时,会产生一个复位信号(IWDG_RESET)。无论何时,只要键寄存器IWDG_KR中被写入0xAAAAIWDG_RLR中的值就会被重新加载到计数器中从而避免产生看门狗复位。

IWDG_PRIWDG_RLR寄存器具有写保护功能。要修改这两个寄存器的值,必须先向IWDG_KR寄存器中写入0x5555。将其他值写入这个寄存器将会打乱操作顺序,寄存器将重新被保护。重装载操作(即写入0xAAAA)也会启动写保护功能。

知道了上面配置的基本原则之后,我们就可以开始配置我们的看门狗了。具体配置过程及配置代码如下所示:

  1. 取消寄存器写保护
  2. 设置独立看门狗的与分频系数,确定时钟
  3. 设置看门狗重装载值
  4. 使能看门狗
  5. 应用程序喂狗
/*** 初始化独立看门狗* prer:分频数:0~7(只有低 3 位有效!)* 分频因子=4*2^prer.但最大值只能是 256!* rlr:重装载寄存器值:低 11 位有效.* 时间计算(大概):Tout=((4*2^prer)*rlr)/40 (ms).*/
void IWDG_Init(u8 prer,u16 rlr)
{IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable); /* 使能对寄存器IWDG_PR和IWDG_RLR的写操作*/IWDG_SetPrescaler(prer);    /*设置IWDG预分频值:设置IWDG预分频值*/IWDG_SetReload(rlr);     /*设置IWDG重装载值*/IWDG_ReloadCounter();    /*按照IWDG重装载寄存器的值重装载IWDG计数器*/IWDG_Enable();        /*使能IWDG*/
}/*** 喂独立看门狗*/
void IWDG_Feed(void)
{IWDG_ReloadCounter();    /*reload*/
}/***main函数*/
void main(void)
{NVIC_Configuration();//优先级配置IWDG_Init(4,625);//初始化独立看门狗,分频数为64,重装载值为625,溢出时间计算为:64*625/40=1000ms=1swhile(1){delay_ms(500);//0.5秒喂一次狗IWDG_Feed();//喂狗}
}

只要对以上三个寄存器进行相应的设置,我们就可以启动 STM32 的独立看门狗,启动过程可以按如下步骤实现(独立看门狗相关的库函数和定义分布在文件 stm32f10x_iwdg.h 和stm32f10x_iwdg.c 中) :
1)取消寄存器写保护(向 IWDG_KR 写入 0X5555)
通过这步,我们取消 IWDG_PR 和 IWDG_RLR 的写保护,使后面可以操作这两个寄存器,设置 IWDG_PR 和 IWDG_RLR 的值。这在库函数中的实现函数是:

IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);

2)设置独立看门狗的预分频系数和重装载值
设置看门狗的分频系数的函数是:
void IWDG_SetPrescaler(uint8_t IWDG_Prescaler); //设置 IWDG 预分频值
设置看门狗的重装载值的函数是:
void IWDG_SetReload(uint16_t Reload); //设置 IWDG 重装载值
设置好看门狗的分频系数 prer 和重装载值就可以知道看门狗的喂狗时间 (也就是看门狗溢出时间) ,该时间的计算方式为:
Tout=((4×2^prer) ×rlr) /40

其中 Tout 为看门狗溢出时间(单位为 ms) ;prer 为看门狗时钟预分频值(IWDG_PR 值),范围为 0~7;rlr 为看门狗的重装载值(IWDG_RLR 的值) ;

比如我们设定 prer 值为 4, rlr 值为 625,那么就可以得到 Tout=64×625/40=1000ms,这样,看门狗的溢出时间就是 1s,只要你在一秒钟之内,有一次写入 0XAAAA 到 IWDG_KR,就不会导致看门狗复位(当然写入多次也是可以的)。这里需要提醒大家的是,看门狗的时钟不是准确的 40Khz,所以在喂狗的时候,最好不要太晚了,否则,有可能发生看门狗复位。

3)重载计数值喂狗(向 IWDG_KR 写入 0XAAAA)
库函数里面重载计数值的函数是:
IWDG_ReloadCounter(); //按照 IWDG 重装载寄存器的值重装载 IWDG 计数器
通过这句,将使 STM32 重新加载 IWDG_RLR 的值到看门狗计数器里面。 即实现独立看门狗的喂狗操作。

4)启动看门狗(向 IWDG_KR 写入 0XCCCC)
库函数里面启动独立看门狗的函数是:
IWDG_Enable(); //使能 IWDG
通过这句,来启动 STM32 的看门狗。注意 IWDG 在一旦启用,就不能再被关闭!想要关闭,只能重启,并且重启之后不能打开 IWDG,否则问题依旧,所以在这里提醒大家,如果不用 IWDG 的话,就不要去打开它,免得麻烦。

关于看门狗的时间计算

对于溢出时间的计算大家可以按照下面的公式计算:Tout=((4×2^prer) ×rlr) /40 (M3)

独立看门狗所用到的库函数:

void WWDG_DeInit(void);
void WWDG_SetPrescaler(uint32_t WWDG_Prescaler);
void WWDG_SetWindowValue(uint8_t WindowValue);
void WWDG_EnableIT(void);
void WWDG_SetCounter(uint8_t Counter);
void WWDG_Enable(uint8_t Counter);
FlagStatus WWDG_GetFlagStatus(void);
void WWDG_ClearFlag(void);

二、窗口看门狗

窗口看门狗(WWDG)通常被用来监测由外部干扰或不可预见的逻辑条件造成的应用程序背离正常的运行序列而产生的软件故障。除非递减计数器的值在 T6 位 (WWDG->CR 的第六位)变成 0 前被刷新,看门狗电路在达到预置的时间周期时,会产生一个 MCU 复位。

在递减计数器达到窗口配置寄存器(WWDG->CFR)数值之前,如果 7 位的递减计数器数值(在控制寄存器中)被刷新,那么也将产生一个 MCU 复位。这表明递减计数器需要在一个有限的时间窗口中被刷新。

但在使用窗口看门狗的时候,需要注意写入WWDG_CR 寄存器时,始终将 1 写入 T6 位,以避免生成立即复位。

下面来看一下窗口看门狗的配置步骤以及配置代码:

  1. 使能 WWDG 时钟
  2. 设置窗口值和分频数
  3. 开启 WWDG 中断并分组
  4. 设置计数器初始值并使能看门狗

窗体看门狗需要用到的库函数:

void RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); // WWDG 时钟使能
void WWDG_SetWindowValue(uint8_t WindowValue);//设置窗口值的函数
void WWDG_SetPrescaler(uint32_t WWDG_Prescaler);//设置分频数的函数
void WWDG_EnableIT(); //开启窗口看门狗中断
void WWDG_Enable(uint8_t Counter);//设置计数器初始值并使能看门狗

注意:在编写中断服务函数时喂狗一定要快,因为窗口看门狗的时效性比较强

窗口看门狗的代码如下:

.c

#ifndef __WDG_H
#define __WDG_H
#include "sys.h"
//独立看门狗
void IWDG_Init(u8 prer,u16 rlr);
void IWDG_Feed(void);
//窗口看门狗
void WWDG_Init(u8 tr,u8 wr,u32 fprer);//初始化WWDG
void WWDG_Set_Counter(u8 cnt);       //设置WWDG的计数器
void WWDG_NVIC_Init(void);
#endif

.h

#include "wdg.h"
#include "led.h"//窗口看门狗
//保存WWDG计数器的设置值,默认为最大.
u8 WWDG_CNT=0x7f;
//初始化窗口看门狗
//tr   :T[6:0],计数器值
//wr   :W[6:0],窗口值
//fprer:分频系数(WDGTB),仅最低2位有效
//Fwwdg=PCLK1/(4096*2^fprer).
void WWDG_Init(u8 tr,u8 wr,u32 fprer)
{ RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE);  //   WWDG时钟使能WWDG_CNT=tr&WWDG_CNT;   //初始化WWDG_CNT.   WWDG_SetPrescaler(fprer);设置IWDG预分频值WWDG_SetWindowValue(wr);//设置窗口值WWDG_Enable(WWDG_CNT);  //使能看门狗 , 设置 counter .                  WWDG_ClearFlag();//清除提前唤醒中断标志位 WWDG_NVIC_Init();//初始化窗口看门狗 NVICWWDG_EnableIT(); //开启窗口看门狗中断
}
//重设置WWDG计数器的值
void WWDG_Set_Counter(u8 cnt)
{WWDG_Enable(cnt);//使能看门狗 , 设置 counter .
}
//窗口看门狗中断服务程序
void WWDG_NVIC_Init()
{NVIC_InitTypeDef NVIC_InitStructure;NVIC_InitStructure.NVIC_IRQChannel = WWDG_IRQn;    //WWDG中断NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;   //抢占2,子优先级3,组2 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;  //抢占2,子优先级3,组2 NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE; NVIC_Init(&NVIC_InitStructure);//NVIC初始化
}void WWDG_IRQHandler(void)
{WWDG_SetCounter(WWDG_CNT);   //当禁掉此句后,窗口看门狗将产生复位WWDG_ClearFlag();   //清除提前唤醒中断标志位LED1=!LED1;   //LED状态翻转
}

main.c

#include "led.h"
#include "delay.h"
#include "key.h"
#include "sys.h"
#include "usart.h"
#include "wdg.h"int main(void)
{  delay_init();       //延时函数初始化   NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//设置中断优先级分组为组2:2位抢占优先级,2位响应优先级uart_init(115200);  //串口初始化为115200LED_Init();KEY_Init();          //按键初始化  LED0=0;delay_ms(500);   WWDG_Init(0X7F,0X5F,WWDG_Prescaler_8);//计数器值为7f,窗口寄存器为5f,分频数为8    while(1){LED0=1;         }
}

在main函数里通过 LED0 来指示 STM32 是否被复位了,如果被复位了就会点亮 500ms。LED0 用来指示中断喂狗,每次中断喂狗翻转一次。

在main函数里通过 LED0 来指示 STM32 是否被复位了,如果被复位了就会点亮 500ms。LED0 用来指示中断喂狗,每次中断喂狗翻转一次。

小总结:

1、有个7位递减计数器(WWDG->CR),就这个计数器和窗口计数器(WWDG->CFR)决定什么时候喂狗。狗喂早了,复位——“早”体现在 计数器值(tr)>窗口值(wr),也就是计数器值还没有减到窗口值以下;

2、当 0x40 < 计数器值(tr) < 窗口值(wr) 时,这时候最适合喂狗了,也只有在这时候喂狗才合适;

3、当 计数器的值 从0x40变到0x3F的时候,将产生看门狗复位;当然在要产生复位的前一段时间,如果开启了提前唤醒中断,那么就会进入中断,在中断函数里,我们需要及时喂狗,否则会产生复位;

4、据网上资料介绍,在这个中断里面一般不进行喂狗,一般是系统去世前的“遗嘱”,比如存储重要的数据等。这个就需要根据个人需要设计。

STM32中的独立看门狗和窗口看门狗相关推荐

  1. stm32 定时器_如何计算STM32定时器、独立看门狗和窗口看门狗

    点击上方蓝色字体,关注我们 本博文简要阐述STM32L011微控制器定时器的参数配置(其他型号大同小异,本文侧重讲解配置,至于各类定时器的特点后续再述),STM32定时器种类繁多有通用定时器.基本定时 ...

  2. STM32之独立看门狗与窗口看门狗总结

    一.独立看门狗 STM32 的独立看门狗由内部专门的 40Khz 低速时钟驱动,即使主时钟发生故障,它也仍然有效. 看门狗的原理:单片机系统在外界的干扰下会出现程序跑飞的现象导致出现死循环,看门狗电路 ...

  3. stm32看门狗_STM32单片机:独立看门狗、窗口看门狗的配置

    SATM32单片机的看门狗有独立看门狗和窗口看门狗之分,这两者的工作原理却完全不同,今天来看一下他们的具体区别和配置方法.▍STM32独立看门狗由专门的低速时钟(LSI)驱动,即便是主时钟发生故障它仍 ...

  4. 如何计算STM32定时器、独立看门狗和窗口看门狗

    目录 1.基本.通用类型定时器 2.独立看门狗定时器 3.窗口看门狗定时器 之前分享了STM32 GPIO的原理.特性.选型和配置.如何计算RTC时钟异步预分频和同步预分频,这次简要阐述STM32L0 ...

  5. stm32 独立看门狗和窗口看门狗区别

    原文地址https://www.cnblogs.com/pertor/p/9483445.html 一.使用调条件 二.特点对比 三.手册介绍 [注]图中的WDGTB为3表示2^3=8,表示8分频,最 ...

  6. STM32—看门狗详解入门(独立看门狗、窗口看门狗)

    目录 一.什么是看门狗? 二.独立看门狗IWDG 1.特点.本质及原理 2.配置 3.main.c代码 三.窗口看门狗WWDG 1.特点.本质及原理 2.配置 3.main.c代码 四.独立看门狗与窗 ...

  7. 看门狗的喂法(独立看门狗和窗口看门狗的区别)!

    在嵌入式系统中,由于MCU(微控制单元:Microcontroller Unit)的工作常常会受到来自外界电磁场的干扰,造成程序的跑飞,而陷入死循环,程序的正常运行被打断,由单片机控制的系统无法继续工 ...

  8. MCU独立看门狗 vs 窗口看门狗

    关注+星标公众号,不错过精彩内容 素材来源 | STM32 微信公众号 | 嵌入式专栏 早期的MCU没有看门狗,就容易引起有些产品死机了不能重启工作.为了避免这个问题,后期的MCU在内部集成了看门狗的 ...

  9. 独立看门狗和窗口看门狗

    为什么要开门狗? 在由单片机构成的微型计算机系统中,由于单片机的工作常常会受到来自外界电磁场的干扰,造成程序的跑飞,而陷入死循环,程序的正常运行被打断,由单片机控制的系统无法继续工作,会造成整个系统的 ...

最新文章

  1. 创建一个好的Sprint Backlog 的8个小贴士
  2. (Spring)Spring的使用
  3. 从0搭建一个Springboot+vue前后端分离项目(四)利用Element框架搭建页面主体部分表格与侧边栏
  4. JQ 全选后获取选中的值_JQ完全学习版本
  5. java空心菱形_java 空心菱形
  6. 武术与软件设计 - 简单即是最好
  7. linux 下恢复文件权限设置,备份和恢复Linux文件权限的方法
  8. 【java笔记】线程(5):线程安全问题
  9. 查找字符串中元素出现的次数
  10. 为什么mysql填不了数据库_求助,为何我的数据不能写入数据库
  11. 进博会高端自行车领导品牌辐轮王盛赞第三届进博会精彩而富有成效
  12. 某些app无法抓包问题
  13. 录屏储存失败因为5823_屏幕录制储存失败因为-5823怎么找回
  14. MATLAB信号处理——信号与系统的分析基础(1)
  15. R语言 去掉NA求均值
  16. 输入一个8bit数,输出其中1的个数。如果只能使用1bit全加器,最少需要几个,请使用verilog进行描述?(附verilog代码)
  17. 【iOS开发】---- Reachability 网络监测
  18. R 关于NA的处理办法
  19. CTFSHOW-信息搜集
  20. 急,求一个类似的ppt模板,感谢好人

热门文章

  1. 视频剪辑,在多个视频同时调整亮度,添加滚动字幕
  2. 【高编 6】 北邮 高级网络程序设计 6. JSPJavaBean
  3. Java JDK path环境变量配置
  4. 苹果CMS10播放地址及图片批量替换
  5. 小电容的实用检测方法总结
  6. 证书转换-SSL证书生成:cer,jks文件 韩俊强的博客
  7. Jetpack Compose 从入门到入门(四)
  8. bzoj 3670: [Noi2014]动物园 kmp与扩展kmp
  9. 基于Beacon技术的智慧动物园解决方案
  10. 求一份DNF过三方。方法等谢谢(悬赏)