STM32中的独立看门狗和窗口看门狗是什么

发布者:Yuexiang888最新更新时间:2024-04-03 来源: elecfans关键字:STM32  独立看门狗  窗口看门狗 手机看文章 扫描二维码
随时随地手机看文章

一、前言

在早期的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 位,以避免生成立即复位。

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

  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();//初始化窗口看门狗 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 用来指示中断喂狗,每次中断喂狗翻转一次。


四、结语

到这里今天的讲解内容就结束了,不知道你对于看门狗以及看门狗的使用有没有理解


关键字:STM32  独立看门狗  窗口看门狗 引用地址:STM32中的独立看门狗和窗口看门狗是什么

上一篇:在家没有硬件开发板怎么调试STM32?
下一篇:STM32入门学习笔记之EEPROM存储实验3

推荐阅读最新更新时间:2024-11-07 23:40

STM32_USB之完全双缓存(包括发送和接收) -- 更新中断处理
STM32的USB双缓存接收代码其实已经可以在ST提供的USB示例代码中找到,只要稍加修改,就可以得到将近1MB的数据接收性能。虽然Datasheet中说明USB发送也同样可以使用双缓存,但并没有示例代码,由于为了测试性能,自己做了一个,测试中没有发现问题,虽然对性能的提升不如在USB接收上实现双缓存那么多。 注意: FreeUserBuffer的作用是切换当前的USB缓存。 1.接收双缓冲: EPX_OUT_Callback中,此代码只是在ST的示例程序的基础上稍加修改,并且不是偶写的,而是一个网友测试的: if(GetENDPOINT(ENDP3) & EP_DTOG_TX) { FreeUserBuff
[单片机]
iar下的stm32启动代码分析
使用的芯片是 STM32F103VET,编译器使用 IAR ARM V5.5 设置头文件查找路径,例如: $PROJ_DIR$.. $PROJ_DIR$......LibrariesCMSISCM3CoreSupport $PROJ_DIR$......LibrariesCMSISCM3DeviceSupportSTSTM32F10x $PROJ_DIR$......LibrariesSTM32F10x_StdPeriph_Driverinc 预定义的symbol 为,HD为high desity 的意思 USE_STDPERIPH_DRIVER STM32F10X_HD 有两个符号是系统默认的,看名字就应该知道什么了
[单片机]
STM32 影子寄存器
01、概述 在定时器框图中,有个小细节,有些寄存器下有个阴影 有这些阴影的表示这些寄存器存在影子寄存器。 在图例中也有对影子寄存器的说明: 根据控制位,在发生U事件后,预装载寄存器内容转移到有效寄存器。这也就是对影子寄存器的说明。 有阴影的寄存器(AutoReloadRegister),表示在物理上这个寄存器对应2个寄存器,一个是程序员可以写入或读出的寄存器,称为preloadregister(预装载寄存器),另一个是程序员看不见的、但在操作中真正起作用的寄存器,称为shadowregister(影子寄存器)。 这里有3个寄存器名称 AutoReloadRegister。 preloadregister。 shadowre
[单片机]
<font color='red'>STM32</font> 影子寄存器
定时器周期计算公式
例如: TIM_TimeBaseStructure.TIM_Period = 10000-1; //当定时器从0计数到10000,即为10000次,为一个定时周期10khz TIM_TimeBaseStructure.TIM_Prescaler = 71; //设置预分频:1us/clk TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1 ; //设置时钟分频系数:不分频(是对外部时钟TIMXETR进行滤波的) TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //向上计数模式 T
[单片机]
定时器周期计算公式
STM32 USB DFU设备固件升级 工程讲解
说到STM32 USB的UDF,其实就是我们常说的IAP(In Application Programming)在应用编程。IAP有很多方法,我之前就用过串口IAP,网络IAP。而这里我们使用的是USB IAP,就是通过USB更新代码。所以这里有必要线了解IAP。 IAP是In Application Programming的首字母缩写,IAP是用户自己的程序在运行过程中对User Flash的部分区域进行烧写,目的是为了在产品发布后可以方便地通过预留的通信口对产品中的固件程序进行更新升级。 通常在用户需要实现IAP功能时,即用户程序运行中作自身的更新操作,需要在设计固件程序时编写两个项目代码,第一个项目程序不执行正常的功能操作,
[单片机]
<font color='red'>STM32</font> USB DFU设备固件升级 工程讲解
基于STM32的SDIO用4位总线24MHZDMA模式操作SHDC卡
发现网上很多所谓的SDIO操作SHDC无意例外都是官方的那个烂玩意,完全没有修改过,所以很多时候根本无法初始化SHDC,我也在网上看到很多人关于这部分的疑问,虽然STM32的SDIO的确是可以这样操作。但是很佩服那群人,什么都没改就发上来,把哥我害惨了。。。。 经过查资料,追踪,最后运气可佳。我发现自己的金士顿4GSD卡(class4)不能初始化跟用4位总线dma操作的原因。。各位也可以上网去找别人的试试,很多人都说不能用4位总线操作,而且用1位总线也只能是在低速率以及开启流控的情况下。而且经常出错。而4位总线总是提示没有检测到起始位。 但是他们都只会问,都没有去想象为什么,我也是。。但是后来发现。STM32的SDIO是完全没问题
[单片机]
一文详解STM32的时钟系统
STM32的时钟树 时钟信号推动单片机内各个部分执行相应的指令,时钟就像人的心跳一样。 STM32本身十分复杂,外设非常多,任何外设都需要时钟才能启动,但并不是所有的外设都需要系统时钟那么高的频率,如果都用高速时钟势必造成浪费。同一个电路,时钟越快功耗越大、抗电磁干扰能力越弱。复杂的MCU采用多时钟源的方法来解决这些问题。如下图,是STM32的时钟系统框图。 如上图左边的部分,看到STM32有4个独立时钟源,HSI、HSE、LSI、LSE。 HSI是高速内部时钟,RC振荡器,频率为8MHz,精度不高。 HSE是高速外部时钟,可接石英/陶瓷谐振器,或者接外部时钟源,频率范围为4MHz~16MHz。 LSI是低速内部时钟,RC
[单片机]
一文详解<font color='red'>STM32</font>的时钟系统
STM32端口输入输出模式配置
STM32的IO口模式配置 根据数据手册提供的信息,stm32的io口一共有八种模式,他们分别是: 四种输入模式 上拉输入: 通过内部的上拉电阻将一个不确定的信号通过一个电阻拉到高电平。 下拉输入: 把电压拉到GND。与上拉原理相似。 浮空输入: 引脚内部什么都不接,处于浮空模式下,电平状态是不确定的。外部信号输入什么,IO口就是什么状态。 模拟输入: 接收到的是连续的模拟信号,一般用于AD转换。 四种输出模式 推挽输出: 可以输出高低电平,连接数字器件。在stm32中推挽电路由两个MOS管组成:输出高电平时P-MOS管导通,引脚联通VDD(3.3v)。输出低电平时N-MOS导通,引脚联通GND。**该方式既提高电路的负
[单片机]
小广播
设计资源 培训 开发板 精华推荐

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

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

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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