一、独立看门狗 STM32 的独立看门狗由内部专门的 40Khz 低速时钟驱动,即使主时钟发生故障,它也仍然有效。 看门狗的原理:单片机系统在外界的干扰下会出现程序跑飞的现象导致出现死循环,看门狗电路就是为了避免这种情况的发生。看门狗的作用就是在一定时间内(通过定时计数器实现)没有接收喂狗信号(表示 MCU 已经挂了),便实现处理器的自动复位重启(发送复位信号) 。 在键值寄存器(IWDG_KR)中写入 0xCCCC,开始启用独立看门狗;此时计数器开始从其复位值 0xFFF 递减计数。当计数器计数到末尾 0x000 时,会产生一个复位信号(IWDG_RESET)。无论何时,只要键寄存器 IWDG_KR 中被写入 0xAAAA, IWDG_RLR 中的值就会被重新加载到计数器中从而避免产生看门狗复位 。 IWDG_PR 和 IWDG_RLR 寄存器具有写保护功能。要修改这两个寄存器的值,必须先向IWDG_KR 寄存器中写入 0x5555。将其他值写入这个寄存器将会打乱操作顺序,寄存器将重新被保护。重装载操作(即写入 0xAAAA)也会启动写保护功能。 只要对以上三个寄存器进行相应的设置,我们就可以启动 STM32 的独立看门狗,启动过程可以按如下步骤实现(独立看门狗相关的库函数和定义分布在文件 stm32f10x_iwdg.h 和stm32f10x_iwdg.c 中) : 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 和重装载值就可以知道看门狗的喂狗时间 (也就是看门狗溢出时间) ,该时间的计算方式为: Tout=((4×2^prer) ×rlr) /40 其中 Tout 为看门狗溢出时间(单位为 ms) ;prer 为看门狗时钟预分频值(IWDG_PR 值),范围为 0~7;rlr 为看门狗的重装载值(IWDG_RLR 的值) ; 比如我们设定 prer 值为 4, rlr 值为 625,那么就可以得到 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 的话,就不要去打开它,免得麻烦。 - /**
8 H4 I6 Q. v2 d) e7 f - * 初始化独立看门狗
4 E& Z `4 E: F4 i6 l9 p; l - * prer:分频数:0~7(只有低 3 位有效!)' x$ |6 R6 S9 i7 i1 o
- * 分频因子=4*2^prer.但最大值只能是 256!/ y7 l& ?& O2 e- h, B0 Y/ H `# n
- * rlr:重装载寄存器值:低 11 位有效.# n0 D% b, Y$ ]4 d. a
- * 时间计算(大概):Tout=((4*2^prer)*rlr)/40 (ms).% e6 V5 p6 V# K$ d) ~7 E
- */# n& G7 o( ~+ ?
- void IWDG_Init(u8 prer,u16 rlr)! b" `& G I% l8 G/ A
- {
# S2 Z3 L9 ^0 s r) { [ - IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable); /* 使能对寄存器IWDG_PR和IWDG_RLR的写操作*/0 { `- y0 b! x
- IWDG_SetPrescaler(prer); /*设置IWDG预分频值:设置IWDG预分频值*/2 l. p! z0 S$ W& ^
- IWDG_SetReload(rlr); /*设置IWDG重装载值*/0 T, z) v% l& N
- IWDG_ReloadCounter(); /*按照IWDG重装载寄存器的值重装载IWDG计数器*/
' v- ~0 `/ N% W. Y - IWDG_Enable(); /*使能IWDG*/4 I7 a5 z ~7 y, [( J* Q0 D
- }) [5 H& q0 J3 Z. O4 T
- ' C$ |! M% X4 |- |& Z7 c, h
- /**
: u& O7 @- h3 f3 b& p5 d - * 喂独立看门狗
. q# G. @% D8 b. b" g0 f3 F) g - */' e R7 ^+ }6 G" l2 d' e
- void IWDG_Feed(void)
6 v% R! e U- p+ {' n) j$ ^ - {" H" \1 y1 [2 j
- IWDG_ReloadCounter(); /*reload*/
1 P2 N( a9 l3 d! |2 H" h* h) @ - }2 v& ]7 a9 K: ^( @$ R4 i
B: K6 D$ p$ g9 L- /**8 f: ?* \0 [4 `0 ^1 l7 ]3 ~9 X! L
- *main函数
. X3 k" `2 z/ \+ I+ J5 x - */1 v: l- Z4 c9 K8 D" X: q
- void main(void)) b) t. m4 l, m; I3 i
- {9 p5 [$ a" w% K$ T$ R& R6 E8 h
- NVIC_Configuration();//优先级配置) R) G' T0 @4 K' K: B
- IWDG_Init(4,625);//初始化独立看门狗,分频数为64,重装载值为625,溢出时间计算为:64*625/40=1000ms=1s
7 w |$ w0 S3 C# U; Z- C* u' W, V - while(1)
; s& E) G( l& v3 i9 t# u - {
! q( W6 F) v6 W. I3 i3 h' q2 T2 l - delay_ms(500);//0.5秒喂一次狗7 O) U' T% T3 e- C0 |' P, \
- IWDG_Feed();//喂狗
! f/ l X! c/ R) M - }
/ x [7 Q! V# z- o8 D- P! r( v$ I - }
复制代码 4 k+ L: R4 d2 d# R0 e+ w( }+ ^4 o
二、窗口看门狗 窗口看门狗(WWDG)通常被用来监测由外部干扰或不可预见的逻辑条件造成的应用程序背离正常的运行序列而产生的软件故障。除非递减计数器的值在 T6 位 (WWDG->CR 的第六位)变成 0 前被刷新,看门狗电路在达到预置的时间周期时,会产生一个 MCU 复位。在递减计数器达到窗口配置寄存器(WWDG->CFR)数值之前,如果 7 位的递减计数器数值(在控制寄存器中)被刷新, 那么也将产生一个 MCU 复位。这表明递减计数器需要在一个有限的时间窗口中被刷新。 小总结: 1、有个7位递减计数器(WWDG->CR),就这个计数器和窗口计数器(WWDG->CFR)决定什么时候喂狗。狗喂早了,复位——“早”体现在 计数器值(tr)>窗口值(wr),也就是计数器值还没有减到窗口值以下; 2、当 0x40 < 计数器值(tr) < 窗口值(wr) 时,这时候最适合喂狗了,也只有在这时候喂狗才合适; 3、当 计数器的值 从0x40变到0x3F的时候,将产生看门狗复位;当然在要产生复位的前一段时间,如果开启了提前唤醒中断,那么就会进入中断,在中断函数里,我们需要及时喂狗,否则会产生复位; 4、据网上资料介绍,在这个中断里面一般不进行喂狗,一般是系统去世前的“遗嘱”,比如存储重要的数据等。这个就需要根据个人需要设计。 % V" R2 f! i5 M: O2 W
库函数中用中断的方式来喂狗的方法,窗口看门狗库函数相关源码和定义分布在文件stm32f10x_wwdg.c 文件和头文件 stm32f10x_wwdg.h 中。步骤如下: 1)使能 WWDG 时钟 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); 该函数既设置了计数器初始值,同时使能了窗口看门狗。 5)编写中断服务函数 在最后,还是要编写窗口看门狗的中断服务函数,通过该函数来喂狗,喂狗要快,否则当窗口看门狗计数器值减到 0X3F 的时候,就会引起软复位了。在中断服务函数里面也要将状态寄存器的 EWIF 位清空。 完成了以上 5 个步骤之后,我们就可以使用 STM32 的窗口看门狗了。 - static u8 WWDG_CNT=0x7f; /*保存WWDG计数器的设置值,默认为最大. */
) U, ?; H3 x2 X+ W+ ^ - 0 b6 p5 d' d& H2 z* D& G
- /**/ D( Q. R3 A1 o, h: N+ u9 p
- * 初始化窗口看门狗 E- {6 Q# E: Y j: |
- * tr :T[6:0],计数器值
" v4 K+ q, A: V* m% |; x - * wr :W[6:0],窗口值+ r8 Z2 X1 _1 @2 D6 Y0 C
- * fprer:分频系数(WDGTB),仅最低2位有效! I4 I. h2 s: C5 N5 y
- * Fwwdg=PCLK1/(4096*2^fprer)., n3 V8 U ?+ v h) h7 n, A
- */
5 q0 F5 S( c0 _# E - void WWDG_Init(u8 tr,u8 wr,u32 fprer) w! M1 {' U, _/ w9 G
- {: j) ~7 ?6 T4 A" ]# B
- RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE); /*WWDG时钟使能*/2 x. J; v8 C, ~& J1 J- [
- WWDG_SetPrescaler(fprer); /*设置IWDG预分频值*/7 S3 x5 E4 G) z/ X1 \4 P
- WWDG_SetWindowValue(wr); /*设置窗口值*/, A/ y( w+ S/ u4 G4 ~1 b& K
- WWDG_CNT=tr&WWDG_CNT; /* 初始化WWDG_CNT. */8 c1 z6 R1 J! I
- WWDG_Enable(WWDG_CNT); /*使能看门狗 , 设置 counter . */
! k; M0 m$ o- P5 X _ }* S2 t; r% { - WWDG_ClearFlag(); /*清除提前唤醒中断标志位*/9 X* }. u( P7 A, H, p
- WWDG_NVIC_Init();/* 初始化窗口看门狗 NVIC */* \4 ^0 r/ W5 y7 ]- ]( I7 E
- WWDG_EnableIT(); /* 开启窗口看门狗中断 */& ^6 j; B" B3 }( J: r- @
- }
( S; L4 I s$ `2 m4 m$ P - 3 k# G4 A) \" p6 q, |! f7 `/ b
- /**' s! K/ ~# m1 K3 X1 i
- * 窗口看门狗中断服务程序
0 m3 n7 v, D+ ?# Y. {7 ? - */ Y, J& K" L9 N9 ^) c+ C7 b* ?& [6 t8 k
- void WWDG_NVIC_Init(void)# p) g& L$ P' o) r2 m2 ~% W
- {
( E& b- f* Y( B' p- p& @9 A/ | - NVIC_InitTypeDef NVIC_InitStructure;
; S( [/ Q! I' `4 y) P - NVIC_InitStructure.NVIC_IRQChannel = WWDG_IRQn; /*WWDG中断*/
/ [3 y4 E$ T: P' ~, I) | - /* 抢占2,子优先级3 */
! G7 ~& D! H0 q! P - NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2;, F* f/ L. C& F% b
- NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3;/ F- G2 d# m8 y1 S9 m
- NVIC_Init(&NVIC_InitStructure);/* NVIC初始化*/
5 U7 M. z( U, C! t - }/ E. `2 t2 R9 B/ F$ l, l1 w
- - E' p1 D& S, S4 V
- /**" ~. d/ J. e: ~ |
- * 重设置WWDG计数器的值2 I2 w) e/ S8 D; }/ [- w
- */
- @7 F3 r S5 c - void WWDG_Set_Counter(u8 cnt)
4 H, c0 Q: d# L3 V3 b. o - {5 n. W0 _$ i$ O- H; k3 u, c( N
- WWDG_Enable(cnt); /*使能看门狗 , 设置 counter . */
; h5 i& b. A. y% _- A - }
9 r; a- g+ v: D3 z - : V% @" W+ p; }. Z
- /**
" P, Y7 E: N) {4 @. `0 w - * 看门狗中断服务程序
" g$ R$ n3 `5 Y9 w: e - */
2 b7 g0 I6 c7 \ - void WWDG_IRQHandler(void)# X3 @; X# h7 r/ T
- {
/ y& p9 H% X3 q6 G1 r - WWDG_Set_Counter(WWDG_CNT);
8 T8 a; }- s Q G; {/ Y0 W8 } - WWDG_ClearFlag(); /*清除提前唤醒中断标志位*/2 _7 b6 S' |6 n( J! @6 x6 X# ?
- LED1 = ~LED1; /*LED状态翻转 */' z( ^3 \, b5 L X- {
- }
复制代码 * O) _- C1 W# K
9 `" ?1 M T8 P2 x( }/ _
小总结,一般工程都会使用两个看门狗,一个是独立看门狗,主要用于在代码跑飞之后复位使用,一个是窗口看门狗,主要用于在复位前对于一些重要数据进行保存。 " E& c% K0 k2 a u e. w: K
|