Stm32 第二十九讲:独立看门狗


独立看门狗概述

为什么要看门狗?

在由单片机构成的微型计算机系统中,由于单片机的工作常常会受到来自外界电磁场的干扰, 造成程序的跑飞,而陷入死循环,程序的正常运行被打断,由单片机控制的系统无法继续工作,会造成整个系统陷入停滞状态,发生不可预料的后果, 所以出于对单片机运行状态进行实时监测的考虑, 便产生了一种专门用于监测单片机程序运行状态的模块或者芯片, 俗称“看门狗(watchdog)”。

看门狗解决的问题是什么?

在启动正常运行的时候,系统不能复位。
在系统跑飞(程序异常执行)的情况,系统复位,程序重新执行。

独立看门狗和窗口看门狗的区别

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

独立看门狗

独立看门狗(IWDG)由专用的低速时钟(LSI,40KHz)驱动,即使主时钟发生故障它仍然有效。
独立看门狗适合应用于需要看门狗作为一个在主程序之外能够完全独立工作,并且对时间精度要求低的场合。

窗口看门狗

窗口看门狗由从APB1时钟分频后得到时钟驱动。通过可配置的时间窗口来检测应用程序非正常的过迟或过早操作。
窗口看门狗最适合那些要求看门狗在精确计时窗口起作用的程序。

独立看门狗功能描述

  • 在键值寄存器(IWDG_KR)中写入0XCCCC,开始启用独立看门狗。此时计数器开始从其复位值0XFFF递减,当计数器值计数到尾值0X000时会产生一个复位信号(IWDG_RESET)。
  • 无论何时,只要在键值寄存器IWDG_KR中写入0XAAAA(通常说的喂狗),自动重装载寄存器IWDG_RLR的值就会重新加载到计数器,从而避免看门狗复位。
  • 如果程序异常,就无法正常喂狗,从而系统复位。

独立看门狗相关寄存器

独立看门狗框图

  • 键值寄存器IWDG_KR:0~15位有效
  • 预分频寄存器IWDG_PR:0~2位有效(具有写保护功能,要操作先取消写保护)
  • 重装载寄存器IWDG_RLR:0~11位有效(具有写保护功能,要操作先取消写保护)
  • 状态寄存器IWDG_SR:0~1位有效

键值寄存器IWDG_KR

独立看门狗_KR

预分频寄存器IWDG_PR

独立看门狗_PR

重装载寄存器IWDG_RLR

独立看门狗_RLR

状态寄存器IWDG_SR

独立看门狗_SR

独立看门狗超时时间

独立看门狗_超时时间

溢出时间计算: $Tout=((4\star2^{prer})\star rlr)/40$。
时钟频率$LSI=40K$,一个看门狗时钟周期就是最短超时时间
最长超时时间=(IWDG_RLR寄存器最大值)*看门狗时钟周期。

IWDG独立看门狗操作库函数

  • void IWDG_WriteAccessCmd(uint16_t IWDG_WriteAccess);//取消写保护:0X5555使能
  • void IWDG_SetPrescaler(uint8_t IWDG_Prescaler);//设置预分频系数:写PR
  • void IWDG_SetReload(uint16_t Reload);//设置重装载值:写RLR
  • void IWDG_ReloadCounter(void);//喂狗:写0xAAAA到KR
  • void IWDG_Enable(void);//使能看门狗:写0xCCCC到KR
  • FlagStatus IWDG_GetFlagStatus(uint16_t IWDG_FLAG);//状态:重装载/预分频更新

独立看门狗操作步骤

  1. 取消寄存器写保护。

    IWDG_WriteAccessCmd();

  2. 设置独立看门狗的预分频系数,确定时钟。

    IWDG_SetPrescaler();

  3. 设置看门狗重装载值,确定溢出时间。

    IWDG_SetReload();

  4. 使能看门狗。

    IWDG_Enable();

  5. 应用程序喂狗。

    IWDG_ReloadCounter();

独立看门狗实验

该程序是在按键输入实验的基础上进行的,现象是如果不按WK_UP按键,看门狗会1秒溢出一次,即程序一秒复位一次,按WK_UP按键可以喂狗,以防止程序复位。

by库函数

iwdg.h

  #ifndef __IWDG_H
  #define __IWDG_H
  #include "sys.h" //u8 u16定义在这里面

  void IWDG_Init(u8 prer,u16 rlr);

  #endif

iwdg.c

  #include "iwdg.h"

  void IWDG_Init(u8 prer,u16 rlr)
  {
      IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);//取消写保护
      IWDG_SetPrescaler(prer);//设置预分频系数
      IWDG_SetReload(rlr);//设置看门狗重装载值
      IWDG_ReloadCounter();//喂狗
      IWDG_Enable();//使能看门狗
  }

main.c

  #include "stm32f10x.h"
  #include "key.h"
  #include "beep.h"
  #include "led.h"
  #include "sys.h"
  #include "delay.h"
  #include "iwdg.h"

  int main(void)
  {     
      delay_init();//延时函数初始化
      KEY_Init();//KEY初始化函数
      BEEP_Init();//蜂鸣器初始化函数
      LED_Init();//LED初始函数

      delay_ms(200);
      LED0=0;

      IWDG_Init(4,625);//最大溢出时间是一秒

      while(1)
      {
          if(KEY_Scan(0)==WK_UP_PRESS)
          {
              IWDG_ReloadCounter();
          }
      }
  }

by寄存器

iwdg.h

  #ifndef __IWDG_H
  #define __IWDG_H
  #include "sys.h"

  void IWDG_Init(u8 prer,u16 rlr);
  void FEED(void);

  #endif

iwdg.c

  #include "iwdg.h"
  #include "sys.h"

  void IWDG_Init(u8 prer,u16 rlr)
  {
      IWDG->KR=0X5555;//取消写保护
      IWDG->PR=prer;//设置预分频系数
      IWDG->RLR=rlr;//设置重装载值
      IWDG->KR=0XCCCC;//使能看门狗
      IWDG->KR=0XAAAA;//喂狗
  }

  void FEED()//喂狗函数
  {
          IWDG->KR=0XAAAA;
  }

main.c

  #include "stm32f10x.h"
  #include "delay.h"
  #include "led.h"
  #include "beep.h"
  #include "key.h"
  #include "sys.h"
  #include "iwdg.h"

  int main(void)
  {
      Stm32_Clock_Init(9);
      delay_init(72);
      LED_Init();
      BEEP_Init();
      KEY_Init();
      IWDG_Init(4,625);//溢出时间1s
      delay_ms(200);
      LED0=0;

      while(1)
      {
          if(KEY_Scan(0)==WK_UP_PRESS)
          {
              FEED();
          }
      }
  }

文章作者: Mat Jenin
文章链接: http://matjenin.xyz
版权声明: 本博客所有文章除特別声明外,均采用 CC BY 4.0 许可协议。转载请注明来源 Mat Jenin !
  目录