STM32的看门狗有俩个 WWDG和IWDG 两者最大的区别就是IWDG只有一个喂食下限而WWDG顾名思义窗户必须在一个范围内喂食才能保证不会触发复位
一.IWDG(独立看门狗)
1) 取消寄存器写保护( 向 IWDG_KR 写入 0X5555)通过这步,我们取消 IWDG_PR 和 IWDG_RLR 的写保护,使后面可以操作这两个寄存器,
设置 IWDG_PR 和 IWDG_RLR 的值。 这在库函数中的实现函数是:
IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);
1
这个函数非常简单, 顾名思义就是开启/取消写保护,也就是使能/失能写权限。
2) 设置独立看门狗的预分频系数和重装载值,
void IWDG_SetPrescaler(uint8_t IWDG_Prescaler); //分频系数
IWDG_Prescaler_4 设置 IWDG 预分频值为 4
IWDG_Prescaler_8 设置 IWDG 预分频值为 8
IWDG_Prescaler_16 设置 IWDG 预分频值为 16
IWDG_Prescaler_32 设置 IWDG 预分频值为 32
IWDG_Prescaler_64 设置 IWDG 预分频值为 64
IWDG_Prescaler_128 设置 IWDG 预分频值为 128
IWDG_Prescaler_256 设置 IWDG 预分频值为 256
void IWDG_SetReload(uint16_t Reload); //重装载值0-0x0fff
下面是俩个函数的资料
设置好看门狗的分频系数 prer 和重装载值就可以知道看门狗的喂狗时间,该时间的计算方式为:
Tout=((4×prer) ×rlr) /40
其中 Tout 为看门狗溢出时间(单位为 ms); prer 为看门狗时钟预分频值( IWDG_PR 值); rlr 为看门狗的重装载值( IWDG_RLR 的值);
比如我们设定 prer 值为 16, rlr 值为 625,那么就可以得到 Tout=64×625/40=1000ms,这样,看门狗的溢出时间就是 1s。这里需要提醒大家的是,看门狗的时钟不是准确的 40Khz,所以在喂狗的时候,最好不要太晚了,否则,有可能发生看门狗复位。
3) 重载计数值喂狗( 向 IWDG_KR 写入 0XAAAA)库函数里面重载计数值的函数是:
IWDG_ReloadCounter(); //按照 IWDG 重装载寄存器的值重装载 IWDG 计数器
4) 启动看门狗(向 IWDG_KR 写入 0XCCCC)库函数里面启动独立看门狗的函数是:
IWDG_Enable(); //使能 IWDG
注意 IWDG 在一旦启用,就不能再被关闭!想要关闭,只能重启,并且重启之后不能打开 IWDG,否则问题依旧,所以在这里提醒大家,如果不
用 IWDG 的话,就不要去打开它,免得麻烦。
一般示例如下
#include "wdg.h"
void IWDG_Init(u8 prer,u16 rlr)
{
IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable); //①使能写操作
IWDG_SetPrescaler(prer); //②设置 IWDG 预分频值:设置 IWDG 预分频值
IWDG_SetReload(rlr); //②设置 IWDG 重装载值
IWDG_ReloadCounter(); //③按照 IWDG 重装载寄存器的值重装载 IWDG 计数器
IWDG_Enable(); //④使能 IWDG
}
//喂独立看门狗
void IWDG_Feed(void)
{
IWDG_ReloadCounter();//reload
}
二.窗户看门狗
WWDG 窗户看门狗 Window Watch DoG
在自己的中断程序喂狗
1.可编程自由运行的递减计数器
2.复位条件: (1). 当计数器的数值从0x40减到0x3F时(2) .当刷新看门狗时计数器的数值大于窗口上限值时(窗口的值可以进行设定最大为0x7f 当喂狗时会比较窗户寄存器和递减计数器的值如果递减计数器的值大于窗口寄存器时,会发生置位就相当于设置了一个上限喂狗的时间必须在上限以下才行 这也体现了窗口这个词的含义)
3.若允许中断,当递减计数器等于0x40时可以产生中断,使递减计数器值被重装以避免复位
1) 使能 WWDG 时钟
WWDG 不同于 IWDG, IWDG 有自己独立的 40Khz 时钟,不存在使能问题。而 WWDG使用的是 PCLK1 的时钟,需要先使能时钟。 方法是:
RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); // WWDG 时钟使能
2) 设置窗口值函数是:
void WWDG_SetWindowValue(uint8_t WindowValue);//设置看门狗的上窗口值值在0x40-0x7f
设置分频数的函数是:
void WWDG_SetPrescaler(uint32_t WWDG_Prescaler);//设置看门狗的分频值
WWDG_Prescaler_1 WWDG 计数器时钟为( PCLK/4096) /1
WWDG_Prescaler_2 WWDG 计数器时钟为( PCLK/4096) /2
WWDG_Prescaler_4 WWDG 计数器时钟为( PCLK/4096) /4
WWDG_Prescaler_8 WWDG 计数器时钟为( PCLK/4096) /8
3) 开启 WWDG 中断并分组,开启 WWDG 中断的函数为:
WWDG_EnableIT(); //开启窗口看门狗中断
再用NVIC_Init()函数即可。
4) 设置计数器初始值并使能看门狗
void WWDG_Enable(uint8_t Counter);
该参数取值必须在 0x40 与 0x7F 之间
WWDG 一旦被使能就不能被失能
该函数既设置了计数器初始值,同时使能了窗口看门狗。
5) 编写中断服务函数
用来保存重要的信息
示例
void WWDG_Init(u8 wr,u32 fprer)
{
RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); // WWDG 时钟使能
WWDG_SetPrescaler(fprer); //设置 IWDG 预分频值
WWDG_SetWindowValue(wr); //设置窗口值
WWDG_Enable(0x7f); //使能看门狗,设置 counter
WWDG_ClearFlag(); //清除提前唤醒中断标志位
WWDG_NVIC_Init(); //初始化窗口看门狗 NVIC
WWDG_EnableIT(); //开启窗口看门狗中断
}
//窗口看门狗中断服务程序
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 状态翻转
}
关于喂狗时间:
WWDG与IWDG的主要区别是有一个窗口控制,WWDG的中断是给你最后一次喂狗的机会,通常这个中断不是让你执行喂狗操作的;一般进到这个中断时表示你在其它地方安排的喂狗操作不能奏效,而发生这种现象时,肯定是系统有问题了,或者是程序有Bug,或者是碰到了干扰,在这种情况下,这个中断是为了让你的程序在发生真正的看门狗复位前,有一个紧急处理的机会,如保存重要的数据,或做系统刹车等操作。
由此看出,简单地在WWDG中断喂狗,既没有发挥WWDG相对于IWDG的优势,又因为在中断中喂狗而为以后的产品留下了隐患。
几点思考:
第一、我们可以发现即使设定了最大值,WWDG最大计时仅仅有58ms,我们在比较大的程序中也没必要运行一小段就添加一个喂狗程序,想使其定时5S或10S的时间再复位系统应该怎样处理呢?通过实验我找到一种方法,就是在中断函数中再做一个额外计数器,如果计数器没有达到设定值,就重新加载喂狗定时器初值,同时使设定值加1,当计数器达到设定值时,就不加载喂狗定时器初值,这时看门狗定时器就会从从0x40减到0x3F产生系统复位。使用这个方法可以将定时时间拓展到 58ms*额外计数器设定值,定个几十秒都不是问题
第二,当额外计数器达到设定值时,此时说明程序没有及时复位这个额外计数器,软件或硬件发生了错误,将时系统复位,我们需要存储一些运行过程中的变量,仅仅有不到1ms的时间(从0x40减到0x3F最长大概为910us)怎么够用呢?
这样就先写Wwdg_Feed(0x7F)重新加定时器初值,再对我们的存储函数进行改造,多添加一些Wwdg_Feed(0x7F)函数,使其不至于再减到0x40,存储工作都做好之后,不再喂狗,那么再次发生中断后不再喂狗就会复位系统了。
第三,如果发生复位,如何区分是上电复位还是看门狗复位呢?
在初始化WWDG时候,有一个RCC_GetFlagStatus(RCC_FLAG_WWDGRST)可以用于判断是否发生看门狗复位,如果是重新上电引起的复位这个值当然是系统默认值,如果是看门狗复位的话这个值就会发生变化,这样就可以针对这两种不同状态进行状态恢复。
理解
1、有个7位递减计数器(WWDG->CR),就这个计数器和窗口计数器(WWDG->CFR)决定什么时候喂狗。 狗喂早了,复位——“早”体现在 计数器值(tr)>窗口值(wr),也就是计数器值还没有减到窗口值以下
2、当 0x40 < 计数器值(tr) < 窗口值(wr) 时,这时候最适合喂狗了,也只有在这时候喂狗才合适;
3、当 计数器的值 从0x40变到0x3F的时候,将产生看门狗复位;当然在要产生复位的前一段时间,如果开启了提前唤醒中断,那么就会进入中断,在中断函数里,我们需要及时喂狗,否则会产生复位;
4、据网上资料介绍,在这个中断里面一般不进行喂狗,一般是系统去世前的“遗嘱”,比如存储重要的数据等。这个就需要根据个人需要设计。