STM32学习笔记14—看门狗实验 14.1 STM32看门狗简介 在单片机构成的微型计算机系统中,单片机的工作可能会受到外界的电磁干扰或者程序运行的BUG导致程序指针错误,或者其他错误导致的死循环,引发整个系统陷入停滞状态,所以需要一个与系统独立的定时器来监控单片机的运行状态,这个定时器在系统正常运转的时候,不停的刷新定时器的计数器,例如隔一段时间给这个定时器的计数器写100,然后在定时器减运算到0之前再一次写入100,这样,就保证了定时器不计数到0,也就意味着通过判断这个定时器是否计数到0来判断系统是否陷入死机状态,实现这种功能的定时器就称为看门狗,不停的刷新计数器值的行为就称为“喂狗”,一般计数器计数到0后会直接对单片机进行复位,用于避免系统陷入死循环。 STM32内部有两种看门狗模块,一种是窗口看门狗WWDG,另一种是独立看门狗IWDG,STM32的独立看门狗由内部专门的40Khz低速时钟驱动,即使主时钟发生故障,它也仍然有效。独立看门狗的时钟是一个内部RC时钟,所以并不是准确的40Khz,而是在30~60Khz之间的一个可变化的时钟,只是我们在估算的时候,以40Khz的频率来计算。窗口看门狗通常被用来监测由外部干扰或不可预见的逻辑条件造成的应用程序背离正常的运行序列而产生的软件故障。除非递减计数器的值在T6位变成0前被刷新,看门狗电路在达到预置的时间周期时,会产生一个MCU复位。在递减计数器达到窗口配置寄存器数值之前,如果7位的递减计数器数值在控制寄存器中被刷新,那么也将产生一个MCU复位。这表明递减计数器需要在一个有限的时间窗口中被刷新。 14.2 独立看门狗相关寄存器 14.2.1 键值寄存器IWDG_KR 该寄存器属于只写寄存器,读取的值为0x0000,软件必须以一定间隔写入0xAAAA,否则,当计数器为0时,看门狗会产生复位; 写入0x5555表示允许访问IWDG_PR和IWDG_RLR寄存器; 写入0xCCCC表示启动看门狗工作,如果选择了硬件看门狗则不受此命令字限制。 14.2.2 预分频寄存器IWDG_PR Bit2~Bit 0:预分频因子 000:预分频因子=4 001:预分频因子=8 010:预分频因子=16 011:预分频因子=32 100:预分频因子=64 101:预分频因子=128 110:预分频因子=256 111:预分频因子=256 14.2.3 重装载寄存器IWDG_RLR Bit 11~Bit 0:看门狗计数器重装载值:每当向IWDG_KR寄存器写入0xAAAA时,重装载值会被传送到计数器中,随后计数器从这个值开始递减计数。 14.3 窗口看门狗相关寄存器 14.3.1 控制寄存器WWDG_CR Bit 7:激活位:,此位由软件置1,但仅能由硬件在复位后清0。当WDGA=1时,看门狗可以产生复位 0:禁止看门狗 1:启用看门狗 Bit 6~Bit 0:7位计数器,存储看门狗的计数器值。每(4096x2WDGTB)个PCLK1周期减1。当T6变成0产生看门狗复位 14.3.2 配置寄存器WWDG_CFR Bit 9:提前唤醒中断,此位若置1,则当计数器值达到40h,即产生中断,此中断只能由硬件在复位后清除 Bit 8:预分频器时基 00:CK计时器时钟不分频 01:CK计时器时钟2分频 10:CK计时器时钟4分频 11:CK计时器时钟8分频 Bit6~Bit 5:7位窗口值,用来与递减计数器进行比较用的窗口值 14.3.3 状态寄存器WWDG_SR Bit 0:提前唤醒中断标志,当计数器值达到40h时,此位由硬件置1。它必须通过软件写0来清除。若中断未被使能,此位也会被置1 14.4 实验例程 14.4.1 独立看门狗实验 功能:如果看门狗没有复位,接在PB5上的LED常亮,如果PA0的按键按下,就喂狗,只要按键不停的按,看门狗就一直不会产生复位,保持DS0的常亮,一旦超过看门狗定溢出时间,那么将导致程序重启,这将导致DS0熄灭一次。 (1)创建wdg.h文件输入以下代码。 - #ifndef _WDG_H_6 t& L+ m8 ]7 E$ S
- #define _WDG_H_1 s+ g) C) j9 \. W; S$ _
- * Y, }6 F" {: i( m4 S; k0 [* s
- #include "sys.h"
- c* i& z' E+ e* k$ ] - /*********************************************************************************************************7 B. g* n9 t( F% v
- 函 数 列 表
2 K9 O0 T. H& ~) N$ I - *********************************************************************************************************/! X8 d: c+ v# V+ `; r6 w8 h+ d
- void IWDG_Init( u8 prer, u16 rlr ) ; //独立看门狗初始化& g6 F* B8 K: u' i9 {: V" a& V
, v; h$ t4 b* R4 Q, r5 \- #endif
复制代码
, s K* S/ V: G3 g(2)创建wdg.c文件输入以下代码。 - #include "wdg.h"
+ [# R9 k, P2 E& d9 n t/ d$ o - /***************************************************) O6 k5 w7 z9 H8 f x' _4 d- {
- Name :IWDG_Init. p9 f7 N$ E( ?& Y5 c/ B
- Function :独立看门狗初始化
6 l& M4 {8 m8 D* V& H% ?7 m - Paramater :* j& r7 o# v6 f% p
- prer:分频数:0~73 P4 W. b( J& k6 F4 ]* }* A
- rlr:重装载寄存器值5 F$ U4 {$ x& [4 W& L
- Return :None
' D; U% k3 N( {2 C R. b, v - ***************************************************/2 R% O' n8 k: a7 g- V' S) l
- void IWDG_Init( u8 prer, u16 rlr )
6 b2 G9 Y( G8 p+ Y8 R* G - {
8 F0 f! E- ^1 `/ a8 }: @. c% e5 D/ i - IWDG->KR = 0x5555 ; //使能对IWDG->PR和IWDG->RLR的写. \# o- x9 V w2 | N8 h
- IWDG->PR = prer ; //设置分频系数
$ [( F$ H. ^" f1 |- Y( o& v. o& W - IWDG->RLR = rlr ; //从加载寄存器 IWDG->RLR
) v+ u) J; `3 \+ n2 _& S - IWDG->KR = 0xAAAA ; //更新计数器
( F7 k0 o8 _# B5 Q - IWDG->KR = 0xCCCC ; //使能看门狗4 r' J5 k) g B6 e" L; k
- }' q/ i2 Q1 b: k9 M" ]/ ?
复制代码
5 N1 _9 @: i: S$ x(3)在1.c文件中输入以下代码。 - #include "sys.h"
" I' ^2 r) s& `0 | - #include "delay.h"
) G" u+ o. H5 v& D2 y: o0 O% L - #include "usart1.h"- e. H3 n9 b; t/ B0 R1 W$ U
- #include "wdg.h"' A0 Q% j8 D8 w; d0 X2 k! q0 K
- /***************************************************
9 a X1 ~1 A2 Y% ?5 W' E8 Q - Name :LED_Init
3 {2 I. }& j, ~1 _6 ]2 j E - Function :LED初始化
0 t3 Q+ ?9 F8 D* C- O; y5 |; i, H - Parameter :None
( r- } T( ?2 c1 _7 f/ Z n+ ] - Return :None
3 \; `6 s/ ~* x9 t% h9 C; V - ***************************************************/
8 t* O! L2 @- Q! R0 i. r" r - #define LED PBout( 5 ) //定义LED端口
' s( x" A9 q/ }+ w4 m6 m - void LED_Init()
7 v0 ]! ]5 a9 O) {/ u( ~ - {/ s9 L" B$ P* Z" h" J7 t
- RCC->APB2ENR |= 1<<3 ;
' O$ G5 _: R( u1 s6 N0 E) N - GPIOB->CRL &= 0xFF0FFFFF ;" Y2 L3 K! w9 v1 o1 V: H& y
- GPIOB->CRL |= 0x00300000 ;
, A1 N1 d$ M, ~$ o) @ l - LED = 1 ;1 s* b: _' i, L. E8 N% a' t9 H* k
- }2 P3 U) J; ^3 P
- /***************************************************+ P1 w4 c' G8 y d4 {
- Name :KEY_Init, X3 @7 z7 r0 r+ L7 S1 {
- Function :KEY初始化
+ n; |/ R' _$ ]; I" z2 l3 G - Parameter :None
+ q) |: |2 i& {( _ - Return :None' a" o, o) Q8 b6 T
- ***************************************************/# q/ d) c4 ~+ r" S/ p; `
- #define KEY PAin( 0 ) //定义按键端口0 A1 P) q. s( ~# ]6 ?4 P
- void KEY_Init()( o) Z& e4 A) W9 n+ h/ ]$ y! ]
- {
0 p6 Z1 [5 Z0 K3 K) d - RCC->APB2ENR |= 1<<2 ;
$ V: E# A: d1 Y8 f! C+ ^' c. Q: H - GPIOA->CRL &= 0xFFFFFFF0 ;3 H) J# z* k% i$ u2 V# ~
- GPIOA->CRL |= 0x00000008;7 u9 w9 V/ C& ^5 i9 }! x9 X3 Z
- }
+ S6 p9 Y6 s9 j* h( v - /***************************************************
; Y6 C$ K- c( F7 K5 d6 l - Name :main( J9 K! ]- Y/ u! c7 F4 \
- Function :主函数
: Y6 m9 E5 a j6 S$ F - Parameter :None# q/ z& b: j! A
- Return :None
( N! z* |1 a, c& c8 C& D" W - ***************************************************/: d8 F: |0 m2 \9 o
- int main()* ~% u$ `5 G/ C8 D( t& ?. J
- {8 T, P* [- Q6 o- N
- STM32_Clock_Init( 9 ) ; //STM32时钟初始化$ ^8 o* Y$ `1 I- u0 \- [
- SysTick_Init( 72 ) ; //SysTick初始化
6 W6 h. m7 F$ S) o. ?' F - USART1_Init( 72, 115200 ) ; //初始化串口1波特率1152009 A* W0 `$ ?: `8 t% X
- LED_Init() ; //LED初始化$ o4 g& M% F/ f% t3 h; n8 T
- KEY_Init() ; //按键初始化 }& I y& {2 L! H. j0 R
- delay_ms( 500 ) ; //延时500ms,让人可以看到DS0灭的状态. x" z, P' K1 S2 v0 K
- IWDG_Init( 4, 625 ) ; //与分频数为64,重载值为625,溢出时间为1s
, E( c8 Q- N; e3 H% B - LED = 0 ; //点亮DS0
; Q( U, ^9 ^& T; D$ B2 t - while( 1 )
/ O! l! M' \' B0 ~ - {
4 d* j! F2 g& Q( K; a. C J% { ` - if( KEY==1 )
) G0 t6 Z6 w( k1 b - {
- s1 Q# `# U5 |' s - delay_ms( 10 ) ;$ }( D) z# T# Y. f7 m" _" M
- if( KEY==1 )
( J2 Y5 K4 W! ~7 f5 H - {* q: ?; |) e! P* q4 ~+ L: ^
- IWDG->KR = 0xAAAA ; //喂狗
7 n3 r7 p& w& o: W" _ - }8 h+ V' Y3 t' m
- }- M1 Q. y2 i+ |) q1 V+ ^8 e( {2 P, ~2 G8 n
- delay_ms( 20 ) ;# K* e* K4 i+ ~ x+ N# u v, v( p; X6 ^
- }* W% a v; Z% ?" U
- }7 R' F9 x4 {9 k0 y2 \
复制代码
0 k1 Z1 b# |& W: L14.4.2 窗口看门狗实验 功能:程序一运行使得接在PB5上的LED1亮300ms后关闭,进入死循环。等待WWDG中断的到来,在中断里面,喂狗,并对PE5上的LED2进行翻转操作。可以看到LED2不停的闪烁,LED1只在刚启动的时候闪一下。 (1)在上一个实验的wdg.h文件的函数列表区域添加以下代码。 - void WWDG_Init( u8 tr, u8 wr, u8 fprer ) ; //窗口看门狗初始化
复制代码 ' l' Q4 P$ J0 b* Y7 b3 e
(2)在上一个实验的wdg.c文件末尾添加以下代码。 - /***************************************************
0 A' u& f/ m7 V* s' Q7 V - Name :WWDG_IRQHandler
7 `- y8 ]: M) B: f3 f+ U - Function :窗口看门狗中断服务程序
( N5 Z& \3 ~; E4 Y& e2 w% d - Paramater :None+ R; m' L# x+ {
- Return :None
- R t# s5 k; T: p, X0 R3 P5 d - ***************************************************/
- Y. g- B9 B% [8 ~ - void WWDG_IRQHandler()
& {4 {, |! W2 Z0 w - {3 x- u$ V& @6 Y- a9 r: l" E- l
- WWDG->CR = 0x7F ; //重设置7位计数器
$ d: T9 S5 x$ B7 ~+ K' k - WWDG->SR = 0x00 ; //清除提前唤醒中断标志位4 A# c6 ?7 n8 D9 n, i5 Z: K
- LED2 != LED2 ;
% }8 B0 o! e$ @6 j( n - }
/ g6 h1 ^+ V3 k" h X& [ - /***************************************************; A- o2 y7 Y: l ^; M
- Name :WWDG_Init
" f5 m& }0 i# q/ T5 P6 l - Function :窗口看门狗初始化) s; l7 v+ G8 B$ h$ j7 J# c, {' k/ Y% ]
- Paramater :
/ o D* [1 g! J# O - tr:计数器值
+ Q. L1 r+ N9 `' H* b% H: R" N - wr:窗口值% T3 P$ Y d3 z. z: f
- fprer:分频系数. b/ v* q+ Q* w% t" X8 x9 ]9 J
- Return :None
/ T. u `6 x/ J W# n& d8 L - ***************************************************/' o8 P h3 \+ `3 h6 O2 U
- void WWDG_Init( u8 tr, u8 wr, u8 fprer )( Y1 K1 D& ~( j7 a
- {7 y* O' S: y% D/ g3 U
- RCC->APB1ENR |= 1<<11 ; //使能wwdg时钟
& L; g* U. |5 S1 p& V - WWDG->CFR |= fprer<<7 ; //PCLK1/4096再除2^fprer' g( n5 g( O# S- t T
- WWDG->CFR &= 0xFF80 ;
* S* G1 t L. F4 n% }+ L) N4 y - WWDG->CFR |= wr ; //设定窗口值
( ]6 u: P0 v+ S9 k/ X: P$ Y* M3 y - WWDG->CR |= tr&0x7F ; //设定计数器值
' i1 C4 p: Y5 r! P - WWDG->CR |= 1<<7 ; //开启看门狗
6 o! T) }9 z6 b4 X - NVIC_Init( 2, 3, WWDG_IRQn, 2 ) ; //抢占2,子优先级3,组2- c" v9 V7 M4 Q/ B" Y) x
- WWDG->SR = 0x00 ; //清除提前唤醒中断标志位0 w4 H5 D7 `1 O6 ^/ h
- WWDG->CFR |= 1<<9 ; //使能提前唤醒中断) ^1 b, g# x; D3 C5 C
- }<span style="background-color: rgb(255, 255, 255);"> </span>
复制代码注:由于在中断服务函数中引用了LED2,所以需要添加头文件#include “led.h”。 . y$ {% M% A6 [0 S
(3)创建led.h文件,并输入以下代码。 - #ifndef _LED_H_. p3 N1 r) G) f: ]( p
- #define _LED_H_( L! u( o0 e9 k6 T6 z8 m) V- x
l1 f0 [9 M/ I6 n) I! a5 M- #include "sys.h"
$ W& W. F/ `3 ^5 m* a* P; S e; c - /*********************************************************************************************************4 _5 ` M. Y/ G) b5 P' g! Q4 v
- 硬 件 端 口$ ^/ D H/ y1 j) V
- *********************************************************************************************************/
' v% A. I- I; I# V9 v1 F - #define LED1 PBout( 5 ) //定义LED1端口
' j, h; y* J/ N. \% l - #define LED2 PEout( 5 ) //定义LED2端口( m' J/ p& ~5 e5 g6 c0 S
- /*********************************************************************************************************
3 X! h5 X' d% | G9 Y - 函 数 列 表, y( Q: `( n/ N e6 @1 R5 B
- *********************************************************************************************************/
% V" B( N4 _" J4 }0 e7 J9 N - void LED_Init( void ) ; //LED初始化, P: N R4 N3 @6 X
+ ~4 Q7 K' L" Y1 r* Q$ b; J2 V* j- #endif
复制代码
' W- j9 }8 T6 g) f(4)创建led.c文件,并输入以下代码。 - #include "led.h"; t& a% O# g; q* p N
- . p) E4 y% W4 d+ n1 r5 T0 ?
- /***************************************************1 k; d. ]' c" O+ D! ^* [
- Name :LED_Init, N) q( a" M$ q2 o$ I- M& ?
- Function :LED初始化8 {+ z4 W$ ^. h! p, t- U
- Paramater :None
! Q: D/ B! w/ e - Return :None
$ P& s! G! B- `4 T/ j - ***************************************************/5 w, E) e9 a) }
- void LED_Init()
+ u( U$ R6 |. _. ?# d3 s, o - {
4 E% L' I# O5 u# E) C - RCC->APB2ENR |= 1<<3 ;
9 T1 y( B3 U) a$ H - GPIOB->CRL &= 0xFF0FFFFF ;
4 ?6 S+ I; J2 Z2 G, E2 v5 b - GPIOB->CRL |= 0x00300000 ;
3 y) n: F& Y4 G! k/ |% U -
) h# G1 U4 ^9 n7 ?+ I" I) s - RCC->APB2ENR |= 1<<6 ; z% X4 p) {: [8 M; N9 K
- GPIOE->CRL &= 0xFF0FFFFF ;
/ B8 L+ [. w; C, w4 B7 C" l( P - GPIOE->CRL |= 0x00300000 ;4 w' f3 R0 N" Q* b
- - Q2 S( s+ {7 E, f2 w
- LED1 = 1 ;6 `2 ^+ P) g e9 } I
- LED2 = 1 ;, X, l6 z/ r$ Q3 j$ l) S6 K
- }
复制代码 7 m6 a# R8 N- F2 |: w, ]& h
(5)在1.c文件中输入以下代码。 - #include "sys.h"
) m# F* }; q" n) U# Y$ W; s - #include "delay.h"
7 D0 b' F) y1 V( y$ p3 ~ - #include "usart1.h"
( ~0 o7 ?3 B' C - #include "led.h"$ S' i1 O. l" n
- #include "wdg.h"0 e) G3 N8 W/ K- \# k& l! p4 K2 C
- /***************************************************
$ u) {1 g- b. u) Z7 _& f! Z - Name :main
! w0 v6 a7 r* K - Function :主函数. Q; s5 r4 Y( _6 A6 J( N% F. _ d* q
- Parameter :None
/ l4 ?3 [! M0 o5 R4 B - Return :None9 ]* y' x8 g; X: l o! T
- ***************************************************/4 _% D7 ^0 v/ R7 ]# z5 _8 \
- int main()" x. t" L! S6 s7 N( N5 d# K' q) V( R( N
- {0 }5 Z! _1 a5 B8 v. H( e( a4 `
- STM32_Clock_Init( 9 ) ; //STM32时钟初始化 v0 @6 e; O3 A7 h+ _6 W y' B
- SysTick_Init( 72 ) ; //SysTick初始化
# h" \9 O( U0 |. }9 B7 t" u0 Y - USART1_Init( 72, 115200 ) ; //初始化串口1波特率115200
9 c1 @& j. b. o0 ~& q - LED_Init() ; //LED初始化
! z2 H6 `+ N6 o: Q" u6 i3 L0 Y+ t) K7 J - LED1 = 0 ; //点亮DS0
6 \3 U% [: ]. R/ x& U- P - delay_ms( 300 ) ; //延时300ms,让人可以看到DS0亮的状态7 N6 G; M( H0 O4 y; E
- WWDG_Init( 0x7F, 0x5F, 3 ) ; //计数器值为7f,窗口寄存器为5f,分频数为8
1 o4 f6 S0 N' d8 k - while( 1 )& O8 _: G) }) T$ `; X2 `
- {( ], Z' k5 W$ f t+ ?$ i4 N7 k4 N
- LED1 = 1 ; //熄灭LED1
% }/ s: T! Y5 c- S; ? - }
5 J O$ A {$ f( R% W4 U. q - }
5 p) D% ~2 ?7 D# ~, u
复制代码 . |1 n' a4 C. g; v k# @
14.5 为何STM32要同时存在窗口看门狗与独立看门狗 14.5.1 独立看门狗的使用条件 (1)程序跑飞 (2)出现死循环 (3)睡眠与休眠不合理 (4)外部主晶振损坏 (5)需要重新复位,且不保留任何数据 14.5.2 窗口看门狗使用条件 (1)软件逻辑出现错误 (2)死机或者死循环 (3)软件执行不按预期效果执行 (4)软件需要重新复位,但是保留所有数据 14.5.3 两者的区别 (1)独立看门狗使用内部专用40kHz低速时钟 窗口看门狗则使用PCLK1的时钟 (2)独立看门狗没有中断,超时直接复位 窗口看门狗有中断,超时可以在中断服务函数中操作或者喂狗 (3)独立看门狗一般用于避免程序跑飞或者死循环 窗口看门狗则是为了避免程序不按照预先设定的逻辑执行 (4)独立看门狗是12位递减操作 窗口看门狗则是6位递减操作 文章出处: 滑小稽笔记 " `5 n) G1 @. ]+ ~9 p6 e b
|