12.1 独立看门狗
12.1.1 独立看门狗工作原理
独立看门狗用通俗一点的话来解释就是一个 12 位的递减计数器,当计数器的值从某个值一直减到 0 的时候,系统就会产生一个复位信号,即 IWDG_RESET。如果在计数没减到 0 之前,刷新了计数器的值的话,那么就不会产生复位信号,这个动作就是我们经常说的喂狗。看门狗功能由 VDD 电压域供电,在停止模式和待机模式下仍能工作。
独立看门狗由内部专门的 40Khz 低速时钟驱动,即使主时钟发生故障,它也仍然有效。这里需要注意独立看门狗的时钟是一个内部 RC 时钟,所以并不是准确的 40Khz,而是在 30~60Khz 之间的一个可变化的时钟,只是我们在估算的时候,以 40Khz 的频率来计算,看门狗对时间的要求不是很精确,所以,时钟有些偏差,都是可以接受的。
图1独立看门狗框图 【注】看门狗功能处于VDD供电区,即在停机和待机模式时仍能正常工作。
表1看门狗超时时间(40kHz的输入时钟(LSI))
【注】这些时间是按照40kHz时钟给出。实际上, MCU内部的RC频率会在30kHz到60kHz之间变化。此外,即使RC振荡器的频率是精确的,确切的时序仍然依赖于APB接口时钟与RC振荡器时钟之间的相位差,因此总会有一个完整的RC周期是不确定的。通过对LSI进行校准可获得相对精确的看门狗超时时间。
12.1.2 独立看门狗的寄存器描述
首先是键值寄存器 IWDG_KR,该寄存器的各位描述如图2所示。
图2键寄存器(IWDG_KR) 在键值寄存器(IWDG_KR)中写入 0xCCCC,开始启用独立看门狗;此时计数器开始从其复位值 0xFFF 递减计数。当计数器计数到末尾 0x000 时,会产生一个复位信号(IWDG_RESET)。无论何时,只要键寄存器 IWDG_KR 中被写入 0xAAAA, IWDG_RLR 中的值就会被重新加载到计数器中从而避免产生看门狗复位 。
IWDG_PR 和 IWDG_RLR 寄存器具有写保护功能。要修改这两个寄存器的值,必须先向IWDG_KR 寄存器中写入 0x5555。 将其他值写入这个寄存器将会打乱操作顺序,寄存器将重新被保护。重装载操作(即写入 0xAAAA)也会启动写保护功能。预分频寄存器(IWDG_PR)用来设置看门狗时钟的分频系数。 重装载寄存器用来保存重装载到计数器中的值,该寄存器也是一个 32位寄存器,但是只有低 12 位是有效的。
图3预分频寄存器(IWDG_PR)
图4重装载寄存器(IWDG_RLR) 12.1.3独立看门狗的具体代码分析
独立看门狗一般用来检测和解决由程序引起的故障,比如一个程序正常运行的时间是50ms,在运行完这个段程序之后紧接着进行喂狗,我们设置独立看门狗的定时溢出时间为60ms,比我们需要监控的程序 50ms 多一点,如果超过 60ms 还没有喂狗,那就说明我们监控的程序出故障了,跑飞了,那么就会产生系统复位,让程序重新运行。
独立看门的启动过程可以按如下步骤实现。
启动过程可以按如下步骤实现。
1)取消寄存器写保护(向 IWDG_KR 写入 0X5555)
通过这步,我们取消 IWDG_PR 和 IWDG_RLR 的写保护,使后面可以操作这两个寄存器,设置 IWDG_PR 和 IWDG_RLR 的值。 这在库函数中的实现函数是:
IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable); 这个函数非常简单, 顾名思义就是开启/取消写保护,也就是使能/失能写权限。
2)设置独立看门狗的预分频系数和重装载值
设置看门狗的分频系数的函数是:
void IWDG_SetPrescaler(uint8_t IWDG_Prescaler); //设置 IWDG 预分频值 设置看门狗的重装载值的函数是:
void IWDG_SetReload(uint16_t Reload); //设置 IWDG 重装载值 设置好看门狗的分频系数 prer 和重装载值就可以知道看门狗的喂狗时间(也就是看门狗溢出时间),该时间的计算方式为:
T o u t = ( ( 4 × 2 p r e r ) × r l r ) / 40 Tout=((4×2^prer) ×rlr) /40 Tout=((4×2prer)×rlr)/40
其中 Tout 为看门狗溢出时间(单位为 ms); prer 为看门狗时钟预分频值(IWDG_PR 值),范围为 0~7; rlr 为看门狗的重装载值(IWDG_RLR 的值);
比如我们设定 prer 值为 4, rlr 值为 625,那么就可以得到 T o u t = 64 × 625 / 40 = 1000 m s Tout=64×625/40=1000ms Tout=64×625/40=1000ms,这样,看门狗的溢出时间就是 1s, 只要你在一秒钟之内,有一次写入 0XAAAA 到 IWDG_KR,就不会导致看门狗复位(当然写入多次也是可以的)。这里需要提醒大家的是,看门狗的时钟不是准确的 40Khz,所以在喂狗的时候,最好不要太晚了,否则,有可能发生看门狗复位。
3)重载计数值喂狗(向 IWDG_KR 写入 0XAAAA)
库函数里面重载计数值的函数是:
IWDG_ReloadCounter(); //按照 IWDG 重装载寄存器的值重装载 IWDG 计数器 通过这句,将使 STM32 重新加载 IWDG_RLR 的值到看门狗计数器里面。 即实现独立看门狗的喂狗操作。
4) 启动看门狗(向 IWDG_KR 写入 0XCCCC)
库函数里面启动独立看门狗的函数是:
IWDG_Enable(); //使能 IWDG 通过这句,来启动 STM32 的看门狗。 注意 IWDG 在一旦启用,就不能再被关闭!想要关闭,只能重启,并且重启之后不能打开 IWDG,否则问题依旧,所以在这里提醒大家,如果不用 IWDG 的话,就不要去打开它,免得麻烦。
主函数如下:
/** * @brief main * @param None * @retval None */ int main(void) { /*启动系统滴答定时器*/ SysTick_Init(); /*串口初始化*/ USART1_Config(); /*独立看门狗初始化*/ IWDG_Configuration(); printf(“独立看门狗rn”); while(1) { //喂狗 IWDG_ReloadCounter(); printf(“r喂狗rn”); Delay_ms(800); } } 主函数很简单,初始化独立看门狗后,在主循环里不断喂狗即可。
12.1.4 独立看门狗实现现象
编译无误,打开串口,现象如下:
当注释掉IWDG_ReloadCounter(); 语句后,板子就会不断重启,因为没有喂狗就导致板子不断复位。
12.2 窗口看门狗
12.2.1 窗口看门狗工作原理
窗口看门狗通常被用来监测,由外部干扰或不可预见的逻辑条件造成的应用程序背离正常的运行序列而产生的软件故障。除非递减计数器的值在T6位变成0前被刷新,看门狗威廉希尔官方网站
在达到预置的时间周期时,会产生一个MCU复位。在递减计数器达到窗口寄存器数值之前,如果7位的递减计数器数值(在控制寄存器中)被刷新, 那么也将产生一个MCU复位。这表明递减计数器需要在一个有限的时间窗口中被刷新。
图5看门狗框图
图6看门狗时序图 上窗口时间: T m i n = 4096 ∗ ( 2 W D G T B ) ∗ ( T − W + 1 ) / 36 ( u s ) T_{min} = 4096 * (2^{WDGTB})*(T - W + 1)/36 (us) Tmin=4096∗(2WDGTB)∗(T−W+1)/36(us)
下窗口时间: T m a x = 4096 ∗ ( 2 W D G T B ) ∗ ( T − 0 x 40 + 1 ) / 36 ( u s ) T_{max} = 4096 * (2^{WDGTB})*(T - 0x40 + 1)/36 (us) Tmax=4096∗(2WDGTB)∗(T−0x40+1)/36(us)
12.2.2 窗口看门狗的寄存器描述
首先介绍控制寄存器(WWDG_CR),该寄存器的各位描述如图12-7所示。
图7控制寄存器WWDG_CR 可以看出,这里我们的 WWDG_CR 只有低八位有效, T[6: 0]用来存储看门狗的计数器值,随时更新的,每个窗口看门狗计数周期( 4096 × 2 W D G T B 4096×2^ {WDGTB} 4096×2WDGTB)减 1。当该计数器的值从 0X40 变为 0X3F 的时候,将产生看门狗复位。
WDGA 位则是看门狗的激活位,该位由软件置 1,以启动看门狗,并且一定要注意的是该位一旦设置,就只能在硬件复位后才能清零了。
窗口看门狗的第二个寄存器是配置寄存器(WWDG_CFR),该寄存器的各位及其描述如图12-8所示。
图8配置寄存器WWDG_CFR 该寄存器中的 EWI 是提前唤醒中断,也就是在快要产生复位的前一段时间(T[6:0]=0X40) 来提醒我们,需要进行喂狗了,否则将复位!因此,我们一般用该位来设置中断,当窗口看门狗的计数器值减到 0X40 的时候,如果该位设置,并开启了中断,则会产生中断,我们可以在中断里面向 WWDG_CR 重新写入计数器的值,来达到喂狗的目的。
注意这里在进入中断后, 必须在不大于 1 个窗口看门狗计数周期的时间(在 PCLK1 频率为 36M 且 WDGTB 为 0 的条件下,该时间为 113us)内重新写 WWDG_CR,否则,看门狗将产生复位!
最后我们要介绍的是状态寄存器(WWDG_SR),该寄存器用来记录当前是否有提前唤醒的标志。该寄存器仅有位 0 有效,其他都是保留位。当计数器值达到 40h 时,此位由硬件置 1。它必须通过软件写 0 来清除。对此位写 1 无效。 即使中断未被使能, 在计数器的值达到0X40的时候, 此位也会被置 1。
图9状态寄存器WWDG_SR 12.2.3窗口看门狗具体代码分析
启用 STM32 的窗口看门狗如下步骤。
1)使能 WWDG 时钟
WWDG 不同于 IWDG, IWDG 有自己独立的 40Khz 时钟,不存在使能问题。而 WWDG使用的是 PCLK1 的时钟,需要先使能时钟。 方法是:
RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); // WWDG 时钟使能 2)设置窗口值和分频数
设置窗口值的函数是:
void WWDG_SetWindowValue(uint8_t WindowValue); 这个函数的入口参数 WindowValue 用来设置看门狗的上窗口值。
设置分频数的函数是:
void WWDG_SetPrescaler(uint32_t WWDG_Prescaler); 这个函数同样只有一个入口参数,用来设置看门狗的分频值。
3)开启 WWDG 中断并分组
开启 WWDG 中断的函数为:
WWDG_EnableIT(); //开启窗口看门狗中断 接下来是进行中断优先级配置,这里就不重复了。
4)设置计数器初始值并使能看门狗
这一步在库函数里面是通过一个函数实现的:
void WWDG_Enable(uint8_t Counter); 该函数既设置了计数器初始值,同时使能了窗口看门狗。
5)编写中断服务函数
在最后,还是要编写窗口看门狗的中断服务函数,通过该函数来喂狗,喂狗要快,否则当窗口看门狗计数器值减到 0X3F 的时候,就会引起软复位了。在中断服务函数里面也要将状态寄存器的 EWIF 位清空。
【小贴士】当然啦,最好不要在中断服务程序中喂狗,标准的写法是在中断服务程序中做重要的操作,也就是复位前的操作,喂狗在主函数中进行。
12.2.4 窗口看门狗实现现象
编译无误,打开串口,现象如下:
12.1 独立看门狗
12.1.1 独立看门狗工作原理
独立看门狗用通俗一点的话来解释就是一个 12 位的递减计数器,当计数器的值从某个值一直减到 0 的时候,系统就会产生一个复位信号,即 IWDG_RESET。如果在计数没减到 0 之前,刷新了计数器的值的话,那么就不会产生复位信号,这个动作就是我们经常说的喂狗。看门狗功能由 VDD 电压域供电,在停止模式和待机模式下仍能工作。
独立看门狗由内部专门的 40Khz 低速时钟驱动,即使主时钟发生故障,它也仍然有效。这里需要注意独立看门狗的时钟是一个内部 RC 时钟,所以并不是准确的 40Khz,而是在 30~60Khz 之间的一个可变化的时钟,只是我们在估算的时候,以 40Khz 的频率来计算,看门狗对时间的要求不是很精确,所以,时钟有些偏差,都是可以接受的。
图1独立看门狗框图 【注】看门狗功能处于VDD供电区,即在停机和待机模式时仍能正常工作。
表1看门狗超时时间(40kHz的输入时钟(LSI))
【注】这些时间是按照40kHz时钟给出。实际上, MCU内部的RC频率会在30kHz到60kHz之间变化。此外,即使RC振荡器的频率是精确的,确切的时序仍然依赖于APB接口时钟与RC振荡器时钟之间的相位差,因此总会有一个完整的RC周期是不确定的。通过对LSI进行校准可获得相对精确的看门狗超时时间。
12.1.2 独立看门狗的寄存器描述
首先是键值寄存器 IWDG_KR,该寄存器的各位描述如图2所示。
图2键寄存器(IWDG_KR) 在键值寄存器(IWDG_KR)中写入 0xCCCC,开始启用独立看门狗;此时计数器开始从其复位值 0xFFF 递减计数。当计数器计数到末尾 0x000 时,会产生一个复位信号(IWDG_RESET)。无论何时,只要键寄存器 IWDG_KR 中被写入 0xAAAA, IWDG_RLR 中的值就会被重新加载到计数器中从而避免产生看门狗复位 。
IWDG_PR 和 IWDG_RLR 寄存器具有写保护功能。要修改这两个寄存器的值,必须先向IWDG_KR 寄存器中写入 0x5555。 将其他值写入这个寄存器将会打乱操作顺序,寄存器将重新被保护。重装载操作(即写入 0xAAAA)也会启动写保护功能。预分频寄存器(IWDG_PR)用来设置看门狗时钟的分频系数。 重装载寄存器用来保存重装载到计数器中的值,该寄存器也是一个 32位寄存器,但是只有低 12 位是有效的。
图3预分频寄存器(IWDG_PR)
图4重装载寄存器(IWDG_RLR) 12.1.3独立看门狗的具体代码分析
独立看门狗一般用来检测和解决由程序引起的故障,比如一个程序正常运行的时间是50ms,在运行完这个段程序之后紧接着进行喂狗,我们设置独立看门狗的定时溢出时间为60ms,比我们需要监控的程序 50ms 多一点,如果超过 60ms 还没有喂狗,那就说明我们监控的程序出故障了,跑飞了,那么就会产生系统复位,让程序重新运行。
独立看门的启动过程可以按如下步骤实现。
启动过程可以按如下步骤实现。
1)取消寄存器写保护(向 IWDG_KR 写入 0X5555)
通过这步,我们取消 IWDG_PR 和 IWDG_RLR 的写保护,使后面可以操作这两个寄存器,设置 IWDG_PR 和 IWDG_RLR 的值。 这在库函数中的实现函数是:
IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable); 这个函数非常简单, 顾名思义就是开启/取消写保护,也就是使能/失能写权限。
2)设置独立看门狗的预分频系数和重装载值
设置看门狗的分频系数的函数是:
void IWDG_SetPrescaler(uint8_t IWDG_Prescaler); //设置 IWDG 预分频值 设置看门狗的重装载值的函数是:
void IWDG_SetReload(uint16_t Reload); //设置 IWDG 重装载值 设置好看门狗的分频系数 prer 和重装载值就可以知道看门狗的喂狗时间(也就是看门狗溢出时间),该时间的计算方式为:
T o u t = ( ( 4 × 2 p r e r ) × r l r ) / 40 Tout=((4×2^prer) ×rlr) /40 Tout=((4×2prer)×rlr)/40
其中 Tout 为看门狗溢出时间(单位为 ms); prer 为看门狗时钟预分频值(IWDG_PR 值),范围为 0~7; rlr 为看门狗的重装载值(IWDG_RLR 的值);
比如我们设定 prer 值为 4, rlr 值为 625,那么就可以得到 T o u t = 64 × 625 / 40 = 1000 m s Tout=64×625/40=1000ms Tout=64×625/40=1000ms,这样,看门狗的溢出时间就是 1s, 只要你在一秒钟之内,有一次写入 0XAAAA 到 IWDG_KR,就不会导致看门狗复位(当然写入多次也是可以的)。这里需要提醒大家的是,看门狗的时钟不是准确的 40Khz,所以在喂狗的时候,最好不要太晚了,否则,有可能发生看门狗复位。
3)重载计数值喂狗(向 IWDG_KR 写入 0XAAAA)
库函数里面重载计数值的函数是:
IWDG_ReloadCounter(); //按照 IWDG 重装载寄存器的值重装载 IWDG 计数器 通过这句,将使 STM32 重新加载 IWDG_RLR 的值到看门狗计数器里面。 即实现独立看门狗的喂狗操作。
4) 启动看门狗(向 IWDG_KR 写入 0XCCCC)
库函数里面启动独立看门狗的函数是:
IWDG_Enable(); //使能 IWDG 通过这句,来启动 STM32 的看门狗。 注意 IWDG 在一旦启用,就不能再被关闭!想要关闭,只能重启,并且重启之后不能打开 IWDG,否则问题依旧,所以在这里提醒大家,如果不用 IWDG 的话,就不要去打开它,免得麻烦。
主函数如下:
/** * @brief main * @param None * @retval None */ int main(void) { /*启动系统滴答定时器*/ SysTick_Init(); /*串口初始化*/ USART1_Config(); /*独立看门狗初始化*/ IWDG_Configuration(); printf(“独立看门狗rn”); while(1) { //喂狗 IWDG_ReloadCounter(); printf(“r喂狗rn”); Delay_ms(800); } } 主函数很简单,初始化独立看门狗后,在主循环里不断喂狗即可。
12.1.4 独立看门狗实现现象
编译无误,打开串口,现象如下:
当注释掉IWDG_ReloadCounter(); 语句后,板子就会不断重启,因为没有喂狗就导致板子不断复位。
12.2 窗口看门狗
12.2.1 窗口看门狗工作原理
窗口看门狗通常被用来监测,由外部干扰或不可预见的逻辑条件造成的应用程序背离正常的运行序列而产生的软件故障。除非递减计数器的值在T6位变成0前被刷新,看门狗威廉希尔官方网站
在达到预置的时间周期时,会产生一个MCU复位。在递减计数器达到窗口寄存器数值之前,如果7位的递减计数器数值(在控制寄存器中)被刷新, 那么也将产生一个MCU复位。这表明递减计数器需要在一个有限的时间窗口中被刷新。
图5看门狗框图
图6看门狗时序图 上窗口时间: T m i n = 4096 ∗ ( 2 W D G T B ) ∗ ( T − W + 1 ) / 36 ( u s ) T_{min} = 4096 * (2^{WDGTB})*(T - W + 1)/36 (us) Tmin=4096∗(2WDGTB)∗(T−W+1)/36(us)
下窗口时间: T m a x = 4096 ∗ ( 2 W D G T B ) ∗ ( T − 0 x 40 + 1 ) / 36 ( u s ) T_{max} = 4096 * (2^{WDGTB})*(T - 0x40 + 1)/36 (us) Tmax=4096∗(2WDGTB)∗(T−0x40+1)/36(us)
12.2.2 窗口看门狗的寄存器描述
首先介绍控制寄存器(WWDG_CR),该寄存器的各位描述如图12-7所示。
图7控制寄存器WWDG_CR 可以看出,这里我们的 WWDG_CR 只有低八位有效, T[6: 0]用来存储看门狗的计数器值,随时更新的,每个窗口看门狗计数周期( 4096 × 2 W D G T B 4096×2^ {WDGTB} 4096×2WDGTB)减 1。当该计数器的值从 0X40 变为 0X3F 的时候,将产生看门狗复位。
WDGA 位则是看门狗的激活位,该位由软件置 1,以启动看门狗,并且一定要注意的是该位一旦设置,就只能在硬件复位后才能清零了。
窗口看门狗的第二个寄存器是配置寄存器(WWDG_CFR),该寄存器的各位及其描述如图12-8所示。
图8配置寄存器WWDG_CFR 该寄存器中的 EWI 是提前唤醒中断,也就是在快要产生复位的前一段时间(T[6:0]=0X40) 来提醒我们,需要进行喂狗了,否则将复位!因此,我们一般用该位来设置中断,当窗口看门狗的计数器值减到 0X40 的时候,如果该位设置,并开启了中断,则会产生中断,我们可以在中断里面向 WWDG_CR 重新写入计数器的值,来达到喂狗的目的。
注意这里在进入中断后, 必须在不大于 1 个窗口看门狗计数周期的时间(在 PCLK1 频率为 36M 且 WDGTB 为 0 的条件下,该时间为 113us)内重新写 WWDG_CR,否则,看门狗将产生复位!
最后我们要介绍的是状态寄存器(WWDG_SR),该寄存器用来记录当前是否有提前唤醒的标志。该寄存器仅有位 0 有效,其他都是保留位。当计数器值达到 40h 时,此位由硬件置 1。它必须通过软件写 0 来清除。对此位写 1 无效。 即使中断未被使能, 在计数器的值达到0X40的时候, 此位也会被置 1。
图9状态寄存器WWDG_SR 12.2.3窗口看门狗具体代码分析
启用 STM32 的窗口看门狗如下步骤。
1)使能 WWDG 时钟
WWDG 不同于 IWDG, IWDG 有自己独立的 40Khz 时钟,不存在使能问题。而 WWDG使用的是 PCLK1 的时钟,需要先使能时钟。 方法是:
RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); // WWDG 时钟使能 2)设置窗口值和分频数
设置窗口值的函数是:
void WWDG_SetWindowValue(uint8_t WindowValue); 这个函数的入口参数 WindowValue 用来设置看门狗的上窗口值。
设置分频数的函数是:
void WWDG_SetPrescaler(uint32_t WWDG_Prescaler); 这个函数同样只有一个入口参数,用来设置看门狗的分频值。
3)开启 WWDG 中断并分组
开启 WWDG 中断的函数为:
WWDG_EnableIT(); //开启窗口看门狗中断 接下来是进行中断优先级配置,这里就不重复了。
4)设置计数器初始值并使能看门狗
这一步在库函数里面是通过一个函数实现的:
void WWDG_Enable(uint8_t Counter); 该函数既设置了计数器初始值,同时使能了窗口看门狗。
5)编写中断服务函数
在最后,还是要编写窗口看门狗的中断服务函数,通过该函数来喂狗,喂狗要快,否则当窗口看门狗计数器值减到 0X3F 的时候,就会引起软复位了。在中断服务函数里面也要将状态寄存器的 EWIF 位清空。
【小贴士】当然啦,最好不要在中断服务程序中喂狗,标准的写法是在中断服务程序中做重要的操作,也就是复位前的操作,喂狗在主函数中进行。
12.2.4 窗口看门狗实现现象
编译无误,打开串口,现象如下:
举报