以單片機(jī)為核心的微型計(jì)算機(jī)系統(tǒng)中,單片機(jī)經(jīng)常會(huì)受到來自外界電磁場(chǎng)的干擾。
造成程序跑飛,只是程序的正常運(yùn)行狀態(tài)被打斷而進(jìn)入死循環(huán),從而使單片機(jī)控制的系統(tǒng)無法正常工作。看門狗就是一種專門用于檢測(cè)單片機(jī)程序運(yùn)行狀態(tài)的硬件結(jié)構(gòu)。
stm32也是如此。
stm32 的獨(dú)立看門狗由內(nèi)部專門的40Khz低速時(shí)鐘驅(qū)動(dòng),即使主時(shí)鐘發(fā)生故障時(shí),它也仍然有效。這里需要注意的是獨(dú)立看門狗的時(shí)鐘是一個(gè)內(nèi)部時(shí)鐘,所以不是準(zhǔn)確的40Khz,而是在30~60Khz之間的一個(gè)可變化的時(shí)鐘,看門狗的時(shí)鐘對(duì)時(shí)間的要求不是很精確,所以時(shí)鐘有偏差可以接受。
本例直接操作寄存器實(shí)現(xiàn)驗(yàn)證獨(dú)立看門狗的復(fù)位功能,設(shè)定一個(gè)800ms的喂狗時(shí)間,在主函數(shù)中實(shí)現(xiàn)LED閃爍,如果設(shè)定一個(gè)1s的延時(shí),則觸發(fā)獨(dú)立看門狗復(fù)位,LED常亮。
庫(kù)函數(shù)實(shí)現(xiàn)當(dāng)外部中斷發(fā)生(按下PA0按鍵),長(zhǎng)時(shí)間不喂狗,引發(fā)獨(dú)立看門狗復(fù)位時(shí),向外用串口輸出復(fù)位提示。
直接操作寄存器
使用獨(dú)立看門狗,需要了解一下寄存器:
鍵值寄存器:(IWDG_KR)
低16位有效的寄存器,只寫寄存器,讀出值恒為0x0000.
軟件必須以一定的間隔寫入0xAAAA,否則,當(dāng)計(jì)數(shù)器為0時(shí),看門狗會(huì)產(chǎn)生復(fù)位。
寫入0x5555表示允許訪問IWDG_PR和IWDG_RLR寄存器。
寫入0xCCCC,啟動(dòng)看門狗工作。
預(yù)分頻寄存器:(IWDG_PR)
第三位有效寄存器,用于設(shè)置看門狗的分頻系數(shù),最低為4,最高位256.
通過設(shè)置PR[2:0]:位來選擇計(jì)數(shù)器時(shí)鐘的預(yù)分頻因子。要改變預(yù)分頻因子,IWDG_SR寄存器的PVU位必須為0。
000: 預(yù)分頻因子=4 100: 預(yù)分頻因子=64
001: 預(yù)分頻因子=8 101: 預(yù)分頻因子=128
010: 預(yù)分頻因子=16 110: 預(yù)分頻因子=256
011: 預(yù)分頻因子=32 111: 預(yù)分頻因子=256
重裝載寄存器:(IWDG_RLR)
低12位有效,RL[11:0]。用于定義看門狗計(jì)數(shù)器的重裝載值。
每當(dāng)向IWDG_KR寄存器寫入0xAAAA時(shí),重裝載值會(huì)被傳送到計(jì)數(shù)器中。隨后計(jì)數(shù)器從這個(gè)值開始遞減計(jì)數(shù)。看門狗超時(shí)周期可通過此重裝載值和時(shí)鐘預(yù)分頻值來計(jì)算。 只有當(dāng)IWDG_SR寄存器中的RVU位為0時(shí),才能對(duì)此寄存器進(jìn)行修改。
狀態(tài)寄存器:(IWDG_SR)
只有低兩位有效。都由硬件置’1’和 清’0’。
RVU[1]: 看門狗計(jì)數(shù)器重裝載值更新
PVU[0]: 看門狗預(yù)分頻值更新
代碼如下: (system.h 和 stm32f10x_it.h 等相關(guān)代碼參照 stm32 直接操作寄存器開發(fā)環(huán)境配置)
User/main.c
01 #include <stm32f10x_lib.h>
02 #include "system.h"
03 #include "wdg.h"
04
05 #define LED1 PAout(4)
06 #define LED2 PAout(5)
07
08 void Gpio_Init(void);
09
10 int main(void)
11 {
12
13 Rcc_Init(9); //系統(tǒng)時(shí)鐘設(shè)置
14
15 Gpio_Init();
16
17 Iwdg_Init(3,1000); //設(shè)定為800ms內(nèi)喂狗
18
19 while(1){
20
21 LED1 = !LED1;
22
23 delay(100000); //延時(shí)100ms后喂狗,LED閃爍
24
25 //delay(1000000); //延時(shí)1000ms,引發(fā)獨(dú)立看門狗復(fù)位,LED不閃爍
26
27 Iwdg_Feed(); //喂狗
28
29 }
30
31 }
32
33
34 void Gpio_Init(void)
35 {
36 RCC->APB2ENR|=1<<2; //使能PORTA時(shí)鐘
37
38 GPIOA->CRL&=0x0000FFFF; // PA0~3設(shè)置為浮空輸入,PA4~7設(shè)置為推挽輸出
39 GPIOA->CRL|=0x33334444;
40
41 }
Library/wdg.c (此文件包含了獨(dú)立看門狗和窗口看門狗的驅(qū)動(dòng)函數(shù))
01 #include <stm32f10x_lib.h>
02 #include "wdg.h"
03
04 /********************************************
05 *
06 *本文件包含窗口看門狗和獨(dú)立看門口的相關(guān)函數(shù)
07 *
08 *********************************************/
09
10 u8 Wwdg_Cnt = 0x7F; //計(jì)數(shù)器值,默認(rèn)為最大值127
11
12 //獨(dú)立看門狗初始化
13 //參數(shù)說明:
14 // pre:分頻數(shù)(0~7),相應(yīng)分頻因子為4*(2^pre)
15 // rlr:低12位有[11:0]
16 // 喂狗時(shí)間計(jì)算: T = (4*(2^pre)*rlr)/40;(ms)
17 void Iwdg_Init(u8 pre,u16 rlr)
18 {
19 IWDG ->KR = 0x5555; //使能對(duì)PR RLR寄存器的寫操作
20 IWDG ->PR = pre; //設(shè)置分頻數(shù)
21 IWDG ->RLR = rlr; //設(shè)定重裝值
22 IWDG ->KR = 0xAAAA; //裝載RLR值到看門狗計(jì)數(shù)器,即喂狗
23 IWDG ->KR = 0xCCCC; //啟動(dòng)看門狗
24 }
25
26 //獨(dú)立看門狗喂狗
27 void Iwdg_Feed()
28 {
29 IWDG -> KR = 0xAAAA; //喂狗
30 }
31
32 //窗口看門狗初始化
33 //參數(shù)說明:
34 // cnt 計(jì)數(shù)器的值,最大 127,0x7F
35 // w_cnt 窗口值,最大 127,0x7F
36 // pre 預(yù)分頻器的時(shí)基值,低兩位有效;實(shí)際時(shí)鐘為: PLCK1/4096/2^pre
37 //需要再主函數(shù)中開啟中斷 WWDG_IRQChannel
38 //設(shè)定喂狗時(shí)間范圍必須在:(WWDG時(shí)鐘為PCLK1,36Mhz)
39 // Tmax =(4096*2^pre*(cnt-63)/36) (us)
40 // Tmin =(4096*2^pre*(cnt-w_cnt)/36) (us)
41 //超出次時(shí)間喂狗復(fù)位
42
43 void Wwdg_Init(u8 cnt,u8 w_cnt,u8 pre)
44 {
45 u8 Cnt_Max = 0x7f; //計(jì)數(shù)器最大值
46
47 Wwdg_Cnt = Cnt_Max&cnt; //設(shè)定計(jì)數(shù)器的值,防止溢出
48
49 RCC->APB1ENR |= 1<<11;
50
51 WWDG -> CFR |= pre <<7; //設(shè)定預(yù)分頻器的時(shí)基,實(shí)際分頻值我
52 WWDG -> CFR |= 1<<9; //使能中斷
53
54 WWDG -> CFR &= 0xFF80; //初始化低七位,即窗口值清0
55 WWDG -> CFR |= w_cnt; // 設(shè)定窗口值
56
57 WWDG -> CR |= Wwdg_Cnt|(1<<7); //設(shè)定計(jì)數(shù)器值,并激活開門狗
58
59 }
60
61 //窗口看門狗喂狗
62
63 void Wwdg_Feed()
64 {
65 WWDG->CR |= (Wwdg_Cnt&0x7F);
66
67 }
Library/wdg.h
1 #include <stm32f10x_lib.h>
2
3 void Iwdg_Init(u8 pre,u16 rlr);
4 void Iwdg_Feed(void);
5
6 void Wwdg_Init(u8 cnt,u8 w_cnt,u8);
7 void Wwdg_Feed(void);
需要注意的是 獨(dú)立看門狗沒有響應(yīng)的中斷。
庫(kù)函數(shù)操作
main.c
view source
print?
001 #include "stm32f10x.h"
002 #include "stdio.h"
003
004 #define PRINTF_ON 1
005
006 void RCC_Configuration(void);
007 void GPIO_Configuration(void);
008 void NVIC_Configuration(void);
009 void USART_Configuration(void);
010 void IWDG_Configuration(void);
011 void EXTI_Configuration(void);
012
013 vu32 DelayTime;
014
015 int main(void)
016 {
017 RCC_Configuration();
018 GPIO_Configuration();
019 NVIC_Configuration();
020 USART_Configuration();
021 EXTI_Configuration();
022 IWDG_Configuration();
023
024 while(1){
025 if(RCC_GetFlagStatus(RCC_FLAG_IWDGRST) == SET)
026 {
027 printf("\r\n The Stm32 has been reset by IWDG .\r\n");
028 RCC_ClearFlag();
029 }
030
031 //do sth. here
032 DelayTime = 100000;
033 while(--DelayTime);
034 // 延時(shí)17ms
035
036 IWDG_ReloadCounter(); //80ms不喂狗復(fù)位
037 GPIO_WriteBit(GPIOA,GPIO_Pin_4,(BitAction)(1- GPIO_ReadOutputDataBit(GPIOA,GPIO_Pin_4)));
038 }
039 }
040
041 void EXTI_Configuration(void)
042 {
043 EXTI_InitTypeDef EXTI_InitStructure;
044
045 EXTI_InitStructure.EXTI_Line = EXTI_Line0;
046 EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;
047 EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;
048 EXTI_InitStructure.EXTI_LineCmd = ENABLE;
049 EXTI_Init(&EXTI_InitStructure);
050
051 GPIO_EXTILineConfig(GPIO_PortSourceGPIOA,GPIO_PinSource0);
052
053 }
054
055 void GPIO_Configuration(void)
056 {
057 GPIO_InitTypeDef GPIO_InitStructure;
058 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0;
059 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
060 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
061 GPIO_Init(GPIOA , &GPIO_InitStructure);
062
063 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
064 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
065 GPIO_Init(GPIOA , &GPIO_InitStructure);
066
067
068 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
069 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
070 GPIO_Init(GPIOA , &GPIO_InitStructure);
071
072 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
073 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
074 GPIO_Init(GPIOA , &GPIO_InitStructure);
075
076 }
077
078 void IWDG_Configuration(void)
079 {
080 RCC_LSICmd(ENABLE); //打開LSI
081 while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY)==RESET);
082
083 IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);
084 IWDG_SetPrescaler(IWDG_Prescaler_32);
085 IWDG_SetReload(100); //80ms ,max 0xFFF 0~4095
086 IWDG_ReloadCounter();
087 IWDG_Enable();
088 }
089
090
091 void RCC_Configuration(void)
092 {
093 /* 定義枚舉類型變量 HSEStartUpStatus */
094 ErrorStatus HSEStartUpStatus;
095
096 /* 復(fù)位系統(tǒng)時(shí)鐘設(shè)置*/
097 RCC_DeInit();
098 /* 開啟HSE*/
099 RCC_HSEConfig(RCC_HSE_ON);
100 /* 等待HSE起振并穩(wěn)定*/
101 HSEStartUpStatus = RCC_WaitForHSEStartUp();
102 /* 判斷HSE起是否振成功,是則進(jìn)入if()內(nèi)部 */
103 if(HSEStartUpStatus == SUCCESS)
104 {
105 /* 選擇HCLK(AHB)時(shí)鐘源為SYSCLK 1分頻 */
106 RCC_HCLKConfig(RCC_SYSCLK_Div1);
107 /* 選擇PCLK2時(shí)鐘源為 HCLK(AHB) 1分頻 */
108 RCC_PCLK2Config(RCC_HCLK_Div1);
109 /* 選擇PCLK1時(shí)鐘源為 HCLK(AHB) 2分頻 */
110 RCC_PCLK1Config(RCC_HCLK_Div2);
111 /* 設(shè)置FLASH延時(shí)周期數(shù)為2 */
112 FLASH_SetLatency(FLASH_Latency_2);
113 /* 使能FLASH預(yù)取緩存 */
114 FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
115 /* 選擇鎖相環(huán)(PLL)時(shí)鐘源為HSE 1分頻,倍頻數(shù)為9,則PLL輸出頻率為 8MHz * 9 = 72MHz */
116 RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
117 /* 使能PLL */
118 RCC_PLLCmd(ENABLE);
119 /* 等待PLL輸出穩(wěn)定 */
120 while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
121 /* 選擇SYSCLK時(shí)鐘源為PLL */
122 RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
123 /* 等待PLL成為SYSCLK時(shí)鐘源 */
124 while(RCC_GetSYSCLKSource() != 0x08);
125 }
126 /* 打開APB2總線上的GPIOA時(shí)鐘*/
127 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1, ENABLE);
128
129 //RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP|RCC_APB1Periph_WWDG, ENABLE);
130
131 }
132
133
134 void USART_Configuration(void)
135 {
136 USART_InitTypeDef USART_InitStructure;
137 USART_ClockInitTypeDef USART_ClockInitStructure;
138
139 USART_ClockInitStructure.USART_Clock = USART_Clock_Disable;
140 USART_ClockInitStructure.USART_CPOL = USART_CPOL_Low;
141 USART_ClockInitStructure.USART_CPHA = USART_CPHA_2Edge;
142 USART_ClockInitStructure.USART_LastBit = USART_LastBit_Disable;
143 USART_ClockInit(USART1 , &USART_ClockInitStructure);
144
145 USART_InitStructure.USART_BaudRate = 9600;
146 USART_InitStructure.USART_WordLength = USART_WordLength_8b;
147 USART_InitStructure.USART_StopBits = USART_StopBits_1;
148 USART_InitStructure.USART_Parity = USART_Parity_No;
149 USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
150 USART_InitStructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
151 USART_Init(USART1,&USART_InitStructure);
152
153 USART_Cmd(USART1,ENABLE);
154 }
155
156
157 void NVIC_Configuration(void)
158 {
159 NVIC_InitTypeDef NVIC_InitStructure;
160
161
162 NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQn;
163 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
164 NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
165 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
166 NVIC_Init(&NVIC_InitStructure);
167
168
169 }
170
171
172 #if PRINTF_ON
173
174 int fputc(int ch,FILE *f)
175 {
176 USART_SendData(USART1,(u8) ch);
177 while(USART_GetFlagStatus(USART1,USART_FLAG_TC) == RESET);
178 return ch;
179 }
180
181 #endif