历史上的今天
返回首页

历史上的今天

今天是:2026年03月24日(星期二)

正在发生

2023年03月24日 | STM32中的独立看门狗和窗口看门狗

2023-03-24 来源:zhihu

一、前言

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


独立看门狗:使用的是外部时钟,即使主频不工作了,看门狗也能正常工作。只要在到达喂狗时间的上限前喂狗即表示程序是正常的,这点和窗口看门狗是有区别的。另外独立看门狗是独立于整个系统之外的,这也是独立看门狗名字的由来,他有自己独立的时钟,不受整个系统的影响,所以独立看门狗主要用来监控硬件上的错误。


窗口看门狗:使用芯片内部时钟。喂狗的时间既有上限又有下限,即喂狗太早或者太晚都不行,比如我要求你在0.8s到0.9s内完成喂狗动作,如果你在0.8s之前或者在0.9s之后喂狗都是不可以的,都会认为MCU出现了异常,从而复位MCU。窗口看门狗是系统内部故障探测器,如果系统时钟出现了错误,那么窗口看门狗也就失去了作用,主要用于监视软件的错误。


二、独立看门狗

从上面的简单对于相信大家对于独立看门狗已经有些了解了,这部分就详细的给大家讲解一下独立看门狗,以及独立看门狗的实现原理。


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


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


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


在这里插入图片描述

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

IWDG_PR和IWDG_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=1s

 while(1)

  {

    delay_ms(500);//0.5秒喂一次狗

      IWDG_Feed();//喂狗

  }        

}


对于溢出时间的计算大家可以按照下面的公式计算: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 位,以避免生成立即复位。


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

使能 WWDG 时钟

设置窗口值和分频数

开启 WWDG 中断并分组

设置计数器初始值并使能看门狗

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


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();//初始化窗口看门狗 NVIC


 WWDG_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);  //串口初始化为115200

  LED_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 用来指示中断喂狗,每次中断喂狗翻转一次。


推荐阅读

史海拾趣

Fairview Microwave Inc公司的发展小趣事

在国内市场取得一定成绩后,Fairview Microwave开始将目光投向国际市场。他们积极参加国际电子展会和技术交流活动,与全球各地的客户建立联系。同时,公司还积极开拓海外市场,通过设立分支机构和代理商等方式,将产品销售到全球各地。这些举措不仅为公司带来了更多的商机,也进一步提升了Fairview Microwave的品牌知名度和影响力。

3E SECURITY公司的发展小趣事

随着公司业务的不断扩张,3E SECURITY积极寻求与国际同行的合作机会。通过与国外知名安全公司开展技术合作、市场共享等方式,公司成功打开了国际市场的大门。同时,公司还积极参与国际安全标准的制定和推广,提升了公司在国际舞台上的影响力和竞争力。

ETRI公司的发展小趣事

在光通信技术领域,ETRI也取得了显著成果。他们与Eldis公司合作,成功将一种电吸收调制光源(EML)商业化。这种光源能够在30公里内每秒传输250亿比特,且在高温和室温下都能稳定工作。这一技术的成功商业化,对于提高数据传输速度和效率具有重要意义。

First Sensor公司的发展小趣事

在5G技术的推动下,ETRI又取得了另一项重大突破。他们成功开发了世界上首个5G室内分布式天线系统,该系统将5G服务无缝扩展至室内,为用户提供了前所未有的高速体验。这一技术的出现,不仅提高了传输容量,缩小了设备尺寸,还降低了部署成本。

Cables To Go公司的发展小趣事

在电子行业的激烈竞争中,Cables To Go公司以其卓越的产品品质和创新精神脱颖而出。公司从成立之初就注重产品研发,不断推出符合市场需求的高品质线缆产品。通过严格把控原材料采购和制造工艺,Cables To Go确保了每一根线缆都具备出色的性能和稳定性。同时,公司还积极投入研发,推出了一系列具有创新性的线缆解决方案,满足了客户多样化的需求。

Apex Tool Group公司的发展小趣事

随着电子设备的普及和互联网的发展,Cables To Go公司看到了巨大的市场潜力。公司积极拓展销售渠道,通过线上电商平台和线下实体店相结合的方式,将产品销往全国各地。此外,公司还积极开拓国际市场,与多个国家和地区的代理商建立了合作关系,进一步扩大了市场份额。这种跨越式的市场拓展策略为公司的快速发展奠定了坚实的基础。

问答坊 | AI 解惑

linux读书笔记

linux目录架构 /   根目录 /bin    常用的命令 binary file 的目錄 /boot   存放系统启动时必须读取的档案,包括核心 (kernel) 在内      /boot/grub/menu.lst   GRUB设置   ...…

查看全部问答>

轨迹机器人制作完整例子

本帖最后由 paulhyde 于 2014-9-15 09:09 编辑 采用ATMEL单片机AT89C2051和电机驱动芯片L293D直接完成机器人的主体。两个电机分别作为左右轮子驱动。图中sensor1~sensor3分别结红外发射和红外接收管。  …

查看全部问答>

pci驱动求助

各位:请教2个问题 1.pci卡上的bar2有256KB空间,用block dma可不可以一次传输256KB 2.如何将驱动程序中申请的内存地址映射到用户空间,我申请了32MB,这32MB空间在物理上连续吗 谢谢…

查看全部问答>

:linux mplayer编译问题

PC机上做了交叉编译,步骤如下: #./configure --host-cc=gcc --cc=arm-linux-gcc --target=arm-armv4l-linux --enable-static --prefix=/tmp/mplayer --disable-win32 --disable-dvdread --enable-fbdev --disable-mencoder --disable-live 2> ...…

查看全部问答>

eboot launch函数的一个疑问

;------------------------------------------------------------------------------ ; ; Function: void Launch (UINT32 *LaunchAddress) ; ; This function is called by OEMLaunch and disables MMU, invalidates I, D ; cache and TLB. U ...…

查看全部问答>

Zstack学习经验点滴:zstack的数据接收和发送

zstack的数据接收和发送。对于接收方需要作两件事情: 1)注册一个端点,通过在XXXXApp_Init()函数里调用afRegister()来实现。如果需要处理ZDO消息和案件消息还得调用相应的注册函数ZDO_RegisterForZDOMsg,RegisterForKeys。 2)注册了端点和消 ...…

查看全部问答>

【讨论】关于模拟串口和图形液晶

大家好,我在设计关于430方案时,遇到了点资源使用问题。想请问各路高手(司马长风请来啊): 1.使用TIME_A模拟串口最高能达到9600的波特率吗?我看了例程倒是可行,但不知能否在这个速率下胜任长时间大流量的通信任务? 2.4270只有一个TIME_ ...…

查看全部问答>

超值条形音箱参考设计套件

模拟、SPDIF 或 USB 输入和系统使立体声输出功率高达 30W ,参考设计的核心是立体声音频编解码器 PCM3070,配有的 miniDSP 可实现定制音频处理 , 控制器是MSP430F2xxx,喜欢玩音频的坛友有新东西了 …

查看全部问答>

关于OSTimeDly(ticks)函数

请问大家,我的一个时钟节拍为10ms,我再任务中用OSTimeDly(50)来等待500ms,可是在实际运行的时候,我发现有时OSTimeDly(50)函数根本没有待,就像执行一条语句一样就过去了,有时又是正确的等待了500ms。是什么原因呢?难道是我的UCOS-II有问题?…

查看全部问答>