STM32独立看门狗(宠物狗)

发布者:电竞狂人最新更新时间:2022-01-15 来源: eefocus关键字:STM32  独立看门狗 手机看文章 扫描二维码
随时随地手机看文章

一、前言

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


······看门狗( watchdog timer),是一个定时器电路, 一般有一个输入,叫喂狗(kicking the dog or service the dog),一个输出到MCU的RST端,MCU正常工作的时候,每隔一段时间输出一个信号到喂狗端,给WDT清零,重载计数值,如果超过规定的时间不喂狗,(一般在程序跑飞时,不在程序正常的状态),WDT 定时超过,就会给出一个复位信号到MCU,使MCU复位. 防止MCU死机.

看门狗的作用就是防止程序发生死循环,或者说程序跑飞。


······STM32 有两个看门狗,一个是独立看门狗另外一个是窗口看门狗, 独立看门狗号称宠物狗,窗口看门狗号称警犬。


······独立看门狗的初始化:初始化看门狗的超时时间,如果在规定的超时时间得不到程序“喂”操作,独立看门狗就立即复位系统。


独立看门狗的喂操作:重新刷新它的计数值,如果计数值减到0的时候,就立即复位CPU,为了防止复位CPU,程序必须每隔一段时间在它减到0之前刷新它的计数值,如此循环。


一、独立看门狗

独立看门狗 (IWDG) 由其专用低速时钟 (LSI) 驱动,因此即便在主时钟发生故障时仍然保持工作状态。


“独立”,这个独立表现在该看门狗有自己的振荡硬件电路,不用依靠于PPL分频的时钟信号,能够独立运行,所以当主时钟受到干扰的时候,独立看门狗还是仍然可以继续工作,如果没有正常喂狗,则会复位CPU。


用到32KHZ,收到的干扰较小,且能够 降低功耗。


1、主要特性

.自由运行递减计数器

.时钟由独立 RC 振荡器提供(可在待机和停止模式下运行)

.当递减计数器值达到 0x000 时产生复位(如果看门狗已激活)


如果要防止看门拘导致CPU复位,在计数值减到0之前,重载计数值就可以,这个动作“喂狗”!


注意:一般避免在while里面喂狗,怕不及时,一般放在定时器中断里面进行喂狗,但是定时器喂狗的计数时间要小于看门狗的计数时间。

在这里插入图片描述

2、功能说明


当通过对关键字寄存器 (IWDG_KR) 写入值 0xCCCC 启动独立看门狗时,计数器开始从复位值 0xFFF 递减计数。当计数器计数到终值 (0x000) 时会产生一个复位信号(IWDG 复位)。

任何时候将关键字 0xAAAA 写到 IWWDG_KR 寄存器中, IWDG_RLR 的值就会被重载到计数器,从而避免产生看门狗复位。


3、框图


STM32F4xx英文参考手册.pdf 第700页


4、库函数接口


a.解锁独立看够寄存器保护,对IWDG->KR写入0x5555。


@brief Enables or disables write access to IWDG_PR and IWDG_RLR registers.

@param IWDG_WriteAccess: new state of write access to IWDG_PR and IWDG_RLR registers.

This parameter can be one of the following values:

@arg IWDG_WriteAccess_Enable: Enable write access to IWDG_PR and IWDG_RLR registers

@arg IWDG_WriteAccess_Disable: Disable write access to IWDG_PR and IWDG_RLR registers

void IWDG_WriteAccessCmd(uint16_t IWDG_WriteAccess)

{

  /* Check the parameters */

  assert_param(IS_IWDG_WRITE_ACCESS(IWDG_WriteAccess));

  IWDG->KR = IWDG_WriteAccess;

}


b.设置独立看门狗分频值


This parameter can be one of the following values:

@arg IWDG_Prescaler_4: IWDG prescaler set to 4

@arg IWDG_Prescaler_8: IWDG prescaler set to 8

@arg IWDG_Prescaler_16: IWDG prescaler set to 16

@arg IWDG_Prescaler_32: IWDG prescaler set to 32

@arg IWDG_Prescaler_64: IWDG prescaler set to 64

@arg IWDG_Prescaler_128: IWDG prescaler set to 128

@arg IWDG_Prescaler_256: IWDG prescaler set to 256

void IWDG_SetPrescaler(uint8_t IWDG_Prescaler)

1

c.设置独立看门狗重载值


@brief Sets IWDG Reload value.

@param Reload: specifies the IWDG Reload value.

This parameter must be a number between 0 and 0x0FFF.

void IWDG_SetReload(uint16_t Reload)

1

d.独立看门狗重载计数值


@brief Reloads IWDG counter with value defined in the reload register

(write access to IWDG_PR and IWDG_RLR registers disabled).

void IWDG_ReloadCounter(void)

1

e.检查是否看门狗复位


为什么需要检查是否看门狗复位呢?用于记录当前系统工作可靠性,方便工程师了解。


@brief Checks whether the specified RCC flag is set or not.

@param RCC_FLAG: specifies the flag to check.

This parameter can be one of the following values:

@arg RCC_FLAG_HSIRDY: HSI oscillator clock ready

@arg RCC_FLAG_HSERDY: HSE oscillator clock ready

@arg RCC_FLAG_PLLRDY: main PLL clock ready

@arg RCC_FLAG_PLLI2SRDY: PLLI2S clock ready

@arg RCC_FLAG_PLLSAIRDY: PLLSAI clock ready (only for STM32F42xxx/43xxx devices)

@arg RCC_FLAG_LSERDY: LSE oscillator clock ready

@arg RCC_FLAG_LSIRDY: LSI oscillator clock ready

@arg RCC_FLAG_BORRST: POR/PDR or BOR reset

@arg RCC_FLAG_PINRST: Pin reset

@arg RCC_FLAG_PORRST: POR/PDR reset

@arg RCC_FLAG_SFTRST: Software reset

@arg RCC_FLAG_IWDGRST: Independent Watchdog reset

@arg RCC_FLAG_WWDGRST: Window Watchdog reset

@arg RCC_FLAG_LPWRRST: Low Power reset

@retval The new state of RCC_FLAG (SET or RESET).

FlagStatus RCC_GetFlagStatus(uint8_t RCC_FLAG)


5、计算超时时间


STM32F4xx中文参考手册.pdf 第495 表85


例如:当前独立看门狗输入时钟源为32KHz,若再经过256分频,此时独立看门狗时钟=32KHz/256=125Hz,表示的是125减到0的时候,一秒钟到达。

同时独立看门狗设置重载值为125,则溢出时间为1S;设置重载值为250,则溢出时间为2S。


#练习

按键实现独立看门狗喂狗。(以STM32F429为例)

bsp_iwdg.c文件


#include "./iwdg/bsp_iwdg.h"   

#include "./led/bsp_led.h"



void iwdg_config(void)

{

  /* 检查是否由独立看门狗导致的复位,如果发现经常由看门狗导致的复位,那么要检查软硬件问题 */

   if (RCC_GetFlagStatus(RCC_FLAG_IWDGRST) != RESET)

   {

    /* IWDGRST flag set */

     /* 亮红灯 */

LED_RED;

    /* Clear reset flags,清除标志位 */

     RCC_ClearFlag();

   }

  else

  {

   /* IWDGRST flag is not set */

    /* 亮蓝灯 */

    LED_BLUE;

  }


   /* 独立看门狗寄存器是受到保护的,现在进行解锁动作*/

IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);


//设置看门狗的时钟 40000/64=625HZ  表示的意思就是计数值从625减到1的时候,1秒钟的到达

IWDG_SetPrescaler(IWDG_Prescaler_64);


   /* 设置重载数值,即超时时间 ,值区间在0-0XFFF,即0-4095

获得1s的超时时间,即需要满足关系式:40 000 /64 / x =1s

x= 625 ,x也在0-4095的区间内,所以满足需求

如果要设置2s为超时时间,那么40000/64/y = 2s   y=625*2=1250

最大的超时时间:4095/625 = 6.552s

    */   

  IWDG_SetReload(625);

 

/* Reload IWDG counter,重载计数值,就是喂狗,可以写个feed_dog函数封装起来 */

IWDG_ReloadCounter();


   /* Enable IWDG ,使能看门狗*/

  IWDG_Enable();

 

}


也可以封装起来,如下所示

/*

 * 设置 IWDG 的超时时间

 * Tout = prv/40 * rlv (s)

 *      prv可以是[4,8,16,32,64,128,256]

 * 独立看门狗使用LSI作为时钟。

 * LSI 的频率一般在 30~60KHZ 之间,根据温度和工作场合会有一定的漂移,我

 * 们一般取 40KHZ,所以独立看门狗的定时时间并一定非常精确,只适用于对时间精度

 * 要求比较低的场合。

 *

 * rlv:预分频器值,取值范围为:0-0XFFF

 * 函数调用举例:

 * IWDG_Config(IWDG_Prescaler_64 ,625);  // IWDG 1s 超时溢出 

 * (64/40)*625 = 1s

 */

void IWDG_Config(uint8_t prv ,uint16_t rlv)

{

// 使能 预分频寄存器PR和重装载寄存器RLR可写

IWDG_WriteAccessCmd( IWDG_WriteAccess_Enable );

// 设置预分频器值

IWDG_SetPrescaler( prv );

// 设置重装载寄存器值

IWDG_SetReload( rlv );

// 把重装载寄存器的值放到计数器中

IWDG_ReloadCounter();

// 使能 IWDG

IWDG_Enable();

}


// 喂狗

void IWDG_Feed(void)

{

// 把重装载寄存器的值放到计数器中,喂狗,防止IWDG复位

// 当计数器的值减到0的时候会产生系统复位

IWDG_ReloadCounter();

}


bsp_iwdg.h


#ifndef __IWDG_H

#define __IWDG_H

#include "stm32f4xx.h"

void IWDG_Config(uint8_t prv ,uint16_t rlv);

void IWDG_Feed(void);

void iwdg_config(void);

#endif /* __IWDG_H */


bsp_led.c


#include "./led/bsp_led.h"   


void LED_GPIO_Config(void)

{

/*定义一个GPIO_InitTypeDef类型的结构体*/

GPIO_InitTypeDef GPIO_InitStructure;


/*开启LED相关的GPIO外设时钟*/

RCC_AHB1PeriphClockCmd ( LED1_GPIO_CLK|LED2_GPIO_CLK|LED3_GPIO_CLK|LED4_GPIO_CLK, ENABLE); 


/*选择要控制的GPIO引脚*/    

GPIO_InitStructure.GPIO_Pin = LED1_PIN;


/*设置引脚模式为输出模式*/

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;   

    

    /*设置引脚的输出类型为推挽输出*/

    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;

    

    /*设置引脚为上拉模式,默认LED亮*/

    GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;


/*设置引脚速率为50MHz */   

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_2MHz; 


/*调用库函数,使用上面配置的GPIO_InitStructure初始化GPIO*/

GPIO_Init(LED1_GPIO_PORT, &GPIO_InitStructure);

    

    /*选择要控制的GPIO引脚*/    

GPIO_InitStructure.GPIO_Pin = LED2_PIN;

    GPIO_Init(LED2_GPIO_PORT, &GPIO_InitStructure);

    

    /*选择要控制的GPIO引脚*/    

GPIO_InitStructure.GPIO_Pin = LED3_PIN;

    GPIO_Init(LED3_GPIO_PORT, &GPIO_InitStructure);

/*选择要控制的GPIO引脚*/    

GPIO_InitStructure.GPIO_Pin = LED4_PIN;

    GPIO_Init(LED4_GPIO_PORT, &GPIO_InitStructure);

/*关闭RGB灯*/

LED_RGBOFF;

/*指示灯默认开启*/

LED4(ON);

}


bsp_led.h


#ifndef __LED_H

#define __LED_H


#include "stm32f4xx.h"


//引脚定义

/*******************************************************/

//R 红色灯

#define LED1_PIN                  GPIO_Pin_10                 

#define LED1_GPIO_PORT            GPIOH                      

#define LED1_GPIO_CLK             RCC_AHB1Periph_GPIOH


//G 绿色灯

#define LED2_PIN                  GPIO_Pin_11                 

#define LED2_GPIO_PORT            GPIOH                      

#define LED2_GPIO_CLK             RCC_AHB1Periph_GPIOH


//B 蓝色灯

#define LED3_PIN                  GPIO_Pin_12                 

#define LED3_GPIO_PORT            GPIOH                       

#define LED3_GPIO_CLK             RCC_AHB1Periph_GPIOH


//小指示灯

#define LED4_PIN                  GPIO_Pin_11                 

#define LED4_GPIO_PORT            GPIOD                       

#define LED4_GPIO_CLK             RCC_AHB1Periph_GPIOD

/************************************************************/


/** 控制LED灯亮灭的宏,

* LED低电平亮,设置ON=0,OFF=1

* 若LED高电平亮,把宏设置成ON=1 ,OFF=0 即可

*/

#define ON  0

#define OFF 1


/* 带参宏,可以像内联函数一样使用 */

#define LED1(a) if (a)

GPIO_SetBits(LED1_GPIO_PORT,LED1_PIN);

else

GPIO_ResetBits(LED1_GPIO_PORT,LED1_PIN)


#define LED2(a) if (a)

GPIO_SetBits(LED2_GPIO_PORT,LED2_PIN);

else

GPIO_ResetBits(LED2_GPIO_PORT,LED2_PIN)


#define LED3(a) if (a)

GPIO_SetBits(LED3_GPIO_PORT,LED3_PIN);

else

GPIO_ResetBits(LED3_GPIO_PORT,LED3_PIN)

#define LED4(a) if (a)

GPIO_SetBits(LED4_GPIO_PORT,LED4_PIN);

else

GPIO_ResetBits(LED4_GPIO_PORT,LED4_PIN)

/* 直接操作寄存器的方法控制IO */

#define digitalHi(p,i) {p->BSRRL=i;}   //设置为高电平

#define digitalLo(p,i) {p->BSRRH=i;} //输出低电平

#define digitalToggle(p,i) {p->ODR ^=i;} //输出反转状态


/* 定义控制IO的宏 */

#define LED1_TOGGLE digitalToggle(LED1_GPIO_PORT,LED1_PIN)

#define LED1_OFF digitalHi(LED1_GPIO_PORT,LED1_PIN)

#define LED1_ON digitalLo(LED1_GPIO_PORT,LED1_PIN)

[1] [2]
关键字:STM32  独立看门狗 引用地址:STM32独立看门狗(宠物狗)

上一篇:STM32窗口看门狗(警犬)
下一篇:STM32定时器输出PWM

推荐阅读最新更新时间:2024-11-12 08:38

STM32 之二 HAL库详解 及 手动移植
HAL库结构   说到STM32的HAL库,就不得不提STM32CubeMX,其作为一个可视化的配置工具,对于开发者来说,确实大大节省了开发时间。STM32CubeMX就是以HAL库为基础的,且目前仅支持HAL库及LL库!首先看一下,官方给出的HAL库的包含结构: - **stm32f2xx.h**主要包含STM32同系列芯片的不同具体型号的定义,是否使用HAL库等的定义,接着,其会根据定义的芯片信号包含具体的芯片型号的头文件: #if defined(STM32F205xx) #include stm32f205xx.h #elif defined(STM32F215xx) #include stm32f215xx
[单片机]
STM32中USART的使用方法
USART作为一种标准接口在应用中十分常见。本文着重分析其作为UART的配置和应用方法。 1、STM32固件库使用外围设备的主要思路 在STM32中,外围设备的配置思路比较固定。首先是使能相关的时钟,一方面是设备本身的时钟,另一方面如果设备通过IO口输出还需要使能IO口的时钟;最后如果对应的IO口是复用功能的IO口,则还必须使能AFIO的时钟。 其次是配置GPIO,GPIO的各种属性由硬件手册的AFIO一章详细规定,较为简单。 接着相关设备需要如果需要使用中断功能,必须先配置中断优先级,后文详述。 然后是配置外围设备的相关属性,视具体设备而定,如果设备需要使用中断方式,必须使能相应设备的中断,之后需要使能相关设备。 最后如
[单片机]
stm32串口dma接收异步数据
本文采用第3种方式.在波特率576000下大数据包冲击证明可行. 源代码: //串口接收DMA缓存 #define UART_RX_LEN 128 extern uint8_t Uart_Rx ; //串口接收DMA缓存 uint8_t Uart_Rx = {0}; //---------------------串口功能配置--------------------- //打开串口对应的外设时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 , ENABLE); //串口发DMA配置 //启动DMA时钟 RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1
[单片机]
基于STM32实现串口的两个分案解析
首先总结一下串口232,422,485 串口232:可双向传输,全双工,最大速率20Kbps,负逻辑电平,-15V~-3V逻辑“1”,+3V~+15V逻辑“0”。 串口422:可双向传输,4线全双工,2线单工。 串口485:可双向传输,4线全双工,2线单工,最大速率10Mb/s,差分信号,发送端:+2V~+6V逻辑“1”,-2V~-6V逻辑“0”,接收端:+200mV逻辑“1”,-200mV逻辑“0”。 对于串口的实现有以两个方案: 方案一,和原子的《例说STM32》一样,首先接收,然后处理,没有消息验证处理,这样就会出现消息覆盖,消息出错后死机,无法明确区分命令,无法及时应答握手信号。方案二,借鉴uC/OSII的消息队列,进
[单片机]
基于<font color='red'>STM32</font>实现串口的两个分案解析
STM32单片机(3) 串口中断通信
注:使用普中科技开发板测试时,需要拔掉Boot1短接帽 两根下载线也要拿掉,重启 /******************************************************************************* * * 软件功能: 串口实验(软件延时方式) * *******************************************************************************/ #include stm32f10x.h #include string.h #include delay.h u8 uart1_
[单片机]
教你使用stm32接收串口的一帧数据
stm32支持接受单个数据或者一帧数据,若配置单个数据接收中断的话,会出现接收包丢包,数据不完整的情况!因此在stm32的串口中断中,还有一个IDLE中断,用来产生串口接受一帧数据而产生的中断,比如说串口发来n个数据,会产生n次接收中断和一次IDLE中断,因此方便使用者来接收主机发送或者从机返回的数据!若想发送一帧数据到串口请参照我博客中的另一篇文章——教你如何使用stm32发送一帧数据! 为了方便大家移植,提醒大家此次代码中的代码不需要任何修改,只需要复制粘贴即可! 第一步:首先配置串口中断! (我配置的是串口3,可根据自己的使用情况更改配置。需要更改的地方:时 钟RCC,GPIO_PIN,中断入口,将所 有的USAR
[单片机]
STM32在Kiel4中注意的问题
开发环境:KIEL4,芯片STM32F103VCT6。函数库VC3.5 1.配置开发环境。 2.Option for Target,主要是C/C++和debug两个部分设置。 3.变量和函数名自动补全以及中文乱码设置。 4.逻辑分析仪。信号不存在或者不能出图形。
[单片机]
<font color='red'>STM32</font>在Kiel4中注意的问题
STM32之RCC
STM32 RCC复位与时钟配置,我首先忽略掉复位,首先学习时钟配置,复位以后用到再学习 STM32有多个时钟源,分别是 HSI:上电默认启动,因精度不高所以先不采用,以后如果需要再使用 HSE:外部高速时钟,系统时钟一般采用它,经过PLL倍频作为系统同时钟 LSE:外部低速时钟,一般专门用于RTC,等到RTC模块时再使用 LSI:内部低速时钟,精度不高,一般用于IWDGCLK 时钟系统框图如下: STM32中各个模块都有自己的时钟,当使用相应的模块时首先记得把此模块时钟开启 本次学习使用标准固件库3.3.0 好了,看明白上图咱就开始吧: void RCC_Configurati
[单片机]
<font color='red'>STM32</font>之RCC

推荐帖子

关于ARM汇编中的一些问题
中括号中间再加条竖线是什么意思,比如:   这段表示什么意思?IMPORT|Image$$RO$$Limit|;EndofROMcode(=startofROMdata)IMPORT|Image$$RW$$Base|;BaseofRAMtoinitialiseIMPORT|Image$$ZI$$Base|;BaseandlimitofareaIMPORT|Image$$ZI$$Limit|;tozeroinitial
hantoufool ARM技术
各位大神谁有时间帮我看一样
/*********************************************程序功能:MCU控制蜂鸣器演奏歌曲《祝你平安》----------------------------------------------拨码开关设置:将BUZZER位拨至ON,其余位拨至OFF测试说明:聆听蜂鸣器“唱出”的乐曲既然是演奏乐曲对于一个音符应该包括两个部分一是声调二是持续时间,在这个程序中声调是用简单的延时-电平翻转来实现的,改变了延时的时间就改变了声调,而时间是通过计数比
shyatupc 微控制器 MCU
altera soc高速信号采集HPS读DDR速度太慢,求助
大家好,我现在正用socFPGA做高速信号采集的项目,FPGA驱动AD芯片每秒钟产生80Mbyte的数据,我要把这些数据写到SSD里,我试了两种方案:1:FPGA将数据写到FPGA端的DDR中,HPS通过AXIbrdges桥将数据读到linux中,但我发现读的速度很慢,只有50Mbyte每秒,linux再把数据写到SSD中,整个过程就更慢了,每秒只能写25Mbyte到SSD2:linux预留512M的空间给FPGA通过FPAGtoSDRAM写入数据,这512M内存不归linux管理
石玉 Altera SoC
关于ISE ERROR:Xst:528
具体错误如下:Xst:528-Multi-sourceinUnitdata_inonsignalref_pix1127;thissignalisconnectedtomultipledrivers.Xst:528-Multi-sourceinUnitdata_inonsignalref_pix1126;thissignalisconnectedtomultipledrivers.Xst:528-Multi-sour
超自然 FPGA/CPLD
该简易电路二极管钳位的原理是什么
如图,增加二极管后,信号将向上平移Vp-0.7V,Vp为输入信号峰值。用软件仿真,的确是该现象,但我不懂这个的原理是什么?为什么信号被抬高这么多?请高手解惑。该简易电路二极管钳位的原理是什么“二极管钳位的原理是什么”原理大概可以说是二极管两端正向电压达到0.7V后二极管导通,为电容C1充电到电容两端电压为(Vp-0.7V)。而电容没有放电通道。钳位输出电压在大于-0.7V的电压范围内,输入电压低于-0.7V时D1就导通假如二极管是理想的,0V导通时。先从信号的第一周开
天涯海角sr 模拟电子
i2c的问题
我想问各位,再设定好丛机的地址后,430是不是会自动发送呀,以及其读写信号是不是会根据I2CRX的高低自动设定。多谢!i2c的问题no我怎么看那个实例程序上并没有写这个得指令呀还没听说过会自动写.除非430自带IIC硬件接口.但也要启动后才行.我用的是430F169,它自带I2C接口。我已经启动了。看看I2C协议吧!带i2c的接口,只说明在发送和接受时,你不用管具体的操作。只需要读,写数据就行了。但是,你也要启动,控制传输顺序,停止等。
wbq315 微控制器 MCU
小广播
设计资源 培训 开发板 精华推荐

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

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

换一换 更多 相关热搜器件

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

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