【STM32】窗口看门狗程序

00. 目录

01. 窗口看门狗简介

窗口看门狗(WWDG)通常被用来监测由外部干扰或不可预见的逻辑条件造成的应用程序背离正常的运行序列而产生的软件故障。除非递减计数器的值在 T6 位(WWDG->CR 的第六位)变成 0 前被刷新,看门狗电路在达到预置的时间周期时,会产生一个 MCU 复位。在递减计数器达到窗口配置寄存器(WWDG->CFR)数值之前,如果 7 位的递减计数器数值(在控制寄存器中)被刷新, 那么也将产生一个 MCU 复位。这表明递减计数器需要在一个有限的时间窗口中被刷新。他们的关系可以用图 12.1.1 来说明:
在这里插入图片描述

图 12.1.1 中,T[6:0]就是 WWDG_CR 的低七位,W[6:0]即是 WWDG->CFR 的低七位。T[6:0]就是窗口看门狗的计数器,而 W[6:0]则是窗口看门狗的上窗口,下窗口值是固定的(0X40)。当窗口看门狗的计数器在上窗口值之外被刷新,或者低于下窗口值都会产生复位。

上窗口值(W[6:0])是由用户自己设定的,根据实际要求来设计窗口值,但是一定要确保窗口值大于 0X40,否则窗口就不存在了。

窗口看门狗的超时公式如下:

Twwdg=(4096×2^WDGTB×(T[5:0]+1)) /Fpclk1;
其中:
Twwdg:WWDG 超时时间(单位为 ms)
Fpclk1:APB1 的时钟频率(单位为 Khz)
WDGTB:WWDG 的预分频系数
T[5:0]:窗口看门狗的计数器低 6 位

02. 硬件设计

本实验用到的硬件资源有:

1) 指示灯 DS0 和 DS1

2) 窗口看门狗

窗口看门狗属于 STM32F4 的内部资源,只需要软件设置好即可正常工作。

03. 编程步骤

1) 使能 WWDG 时钟
WWDG 不同于 IWDG,IWDG 有自己独立的 32Khz 时钟,不存在使能问题。而 WWDG使用的是 PCLK1 的时钟,需要先使能时钟。

RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); // WWDG 时钟使能

2) 设置窗口值和分频数
设置窗口值的函数是:

void WWDG_SetWindowValue(uint8_t WindowValue)

这个函数就一个入口参数为窗口值,很容易理解。

设置分频数的函数是:

void WWDG_SetPrescaler(uint32_t WWDG_Prescaler)

这个函数同样只有一个入口参数就是分频值。

3) 开启 WWDG 中断并分组

开启 WWDG 中断的函数为:

WWDG_EnableIT(); //开启窗口看门狗中断

接下来是进行中断优先级配置,这里就不重复了,使用 NVIC_Init()函数即可。

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

这一步在库函数里面是通过一个函数实现的:

void WWDG_Enable(uint8_t Counter)

该函数既设置了计数器初始值,同时使能了窗口看门狗。

这里还需要说明一下,库函数还提供了一个独立的设置计数器值的函数为:

void WWDG_SetCounter(uint8_t Counter);

5) 编写中断服务函数

在最后,还是要编写窗口看门狗的中断服务函数,通过该函数来喂狗,喂狗要快,否则当窗口看门狗计数器值减到 0X3F 的时候,就会引起软复位了。在中断服务函数里面也要将状态寄存器的 EWIF 位清空。

04. 程序示例

wwdg.h

#ifndef __WWDG_H__
#define __WWDG_H__

#include "sys.h"

//初始化函数
void WWDG_Init(u8 tr,u8 wr,u32 fprer);

//中断处理函数
void WWDG_IRQHandler(void);

#endif /*__WWDG_H__*/

wwdg.c

#include "wwdg.h"
#include "led.h"

//保存WWDG计数器的设置值,默认为最大. 
u8 WWDG_CNT=0X7F;

//初始化函数
//tr计数器的值
//wr窗口值
//fprer 预分频值
void WWDG_Init(u8 tr,u8 wr,u32 fprer)
{
	NVIC_InitTypeDef NVIC_InitStruct;
	
	//使能WWDG时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE);
	
	//设置预分频
	WWDG_SetPrescaler(fprer);
	
	//设置窗口值
	WWDG_SetWindowValue(wr & WWDG_CNT);
	
	//使能看门狗
	WWDG_Enable(WWDG_CNT);
	
	//设置中断
	NVIC_InitStruct.NVIC_IRQChannel = WWDG_IRQn;
	NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 2;
	NVIC_InitStruct.NVIC_IRQChannelSubPriority = 2;
	
	NVIC_Init(&NVIC_InitStruct);
	
	//清除提前唤醒中断标志位
	WWDG_ClearFlag();
	
	//开启提前唤醒中断
	WWDG_EnableIT();
	
}

//中断处理函数
void WWDG_IRQHandler(void)
{
	//清除提前唤醒中断标志位
	WWDG_ClearFlag();
	
	WWDG_SetCounter(WWDG_CNT);
	
	LED2 = !LED2;

}

main.c

#include "sys.h"
#include "delay.h"
#include "usart.h"
#include "led.h"
#include "beep.h"
#include "key.h"
#include "exti.h"
#include "iwdg.h"
#include "wwdg.h"

int main(void)
{ 
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	
	delay_init(168);
	LED_Init();
	KEY_Init();

	LED1 = 0;
	LED2 = 0;
	
	delay_ms(300);

	WWDG_Init(0x7F,0X5F,WWDG_Prescaler_8); 	//计数器值为7f,窗口寄存器为5f,分频数为8	 
	
	while(1)
	{
		LED1 = 1;
	}
	
}

05. 测试结果

可以看到 DS0 亮一下之后熄灭,紧接着 DS1 开始不停的闪烁。每秒钟闪烁 20 次左右。

06. 附录

6.1 【STM32】STM32系列教程汇总

网址:【STM32】STM32系列教程汇总

07. 声明

该教程参考了正点原子的《STM32 F4 开发指南》

猜你喜欢

转载自blog.csdn.net/dengjin20104042056/article/details/108209860