一、前言
在早期的MCU中是沒(méi)有看門(mén)狗這種東西的,所以產(chǎn)品就很容易出現(xiàn)死機(jī),跑飛的情況。為了避免這種情況的出現(xiàn),后期的MCU都集成了看門(mén)狗的功能。但是目前看門(mén)狗發(fā)展到今天基本上分為兩大類(lèi):獨(dú)立看門(mén)狗和窗口看門(mén)狗。
獨(dú)立看門(mén)狗:使用的是外部時(shí)鐘,即使主頻不工作了,看門(mén)狗也能正常工作。只要在到達(dá)喂狗時(shí)間的上限前喂狗即表示程序是正常的,這點(diǎn)和窗口看門(mén)狗是有區(qū)別的。另外獨(dú)立看門(mén)狗是獨(dú)立于整個(gè)系統(tǒng)之外的,這也是獨(dú)立看門(mén)狗名字的由來(lái),他有自己獨(dú)立的時(shí)鐘,不受整個(gè)系統(tǒng)的影響,所以獨(dú)立看門(mén)狗主要用來(lái)監(jiān)控硬件上的錯(cuò)誤。
窗口看門(mén)狗:使用芯片內(nèi)部時(shí)鐘。喂狗的時(shí)間既有上限又有下限,即喂狗太早或者太晚都不行,比如我要求你在0.8s到0.9s內(nèi)完成喂狗動(dòng)作,如果你在0.8s之前或者在0.9s之后喂狗都是不可以的,都會(huì)認(rèn)為MCU出現(xiàn)了異常,從而復(fù)位MCU。窗口看門(mén)狗是系統(tǒng)內(nèi)部故障探測(cè)器,如果系統(tǒng)時(shí)鐘出現(xiàn)了錯(cuò)誤,那么窗口看門(mén)狗也就失去了作用,主要用于監(jiān)視軟件的錯(cuò)誤。
二、獨(dú)立看門(mén)狗
從上面的簡(jiǎn)單對(duì)于相信大家對(duì)于獨(dú)立看門(mén)狗已經(jīng)有些了解了,這部分就詳細(xì)的給大家講解一下獨(dú)立看門(mén)狗,以及獨(dú)立看門(mén)狗的實(shí)現(xiàn)原理。
在了解獨(dú)立看門(mén)狗之前我想大家還是需要先了解一下看門(mén)狗到底是來(lái)干什么的,在由單片機(jī)構(gòu)成的微機(jī)系統(tǒng)中,由于單片機(jī)工作常常會(huì)受到來(lái)自外界電磁場(chǎng)干擾導(dǎo)致程序跑飛,陷入死循環(huán)——即程序正常運(yùn)行被打斷,系統(tǒng)無(wú)法繼續(xù)工作。
這種情況下會(huì)造成系統(tǒng)陷入停滯狀態(tài),發(fā)生不可預(yù)料的后果。因此出于對(duì)單片機(jī)運(yùn)行狀態(tài)進(jìn)行實(shí)時(shí)監(jiān)測(cè)的考慮,產(chǎn)生了一種專(zhuān)門(mén)用于監(jiān)測(cè)單片機(jī)程序運(yùn)行狀態(tài)的模塊或芯片,稱(chēng)為看門(mén)狗。
這里以大家熟悉的STM32為例給大家講解一下獨(dú)立看門(mén)狗的配置以及工作過(guò)程。STM32F10xxx內(nèi)置兩個(gè)看門(mén)狗:獨(dú)立看門(mén)狗和窗口看門(mén)狗,提供了更高的安全性、時(shí)間的精確性和使用的靈活性。
在這里插入圖片描述
STM32中的獨(dú)立看門(mén)狗時(shí)通過(guò)向鍵值寄存器(IWDG_KR)寫(xiě)入0xCCCC來(lái)進(jìn)行配置的,當(dāng)開(kāi)啟了獨(dú)立看門(mén)狗之后其計(jì)數(shù)器就開(kāi)始從0xFFF遞減計(jì)數(shù)。當(dāng)計(jì)數(shù)器計(jì)數(shù)到末尾0x000時(shí),會(huì)產(chǎn)生一個(gè)復(fù)位信號(hào)(IWDG_RESET)。無(wú)論何時(shí),只要鍵寄存器IWDG_KR中被寫(xiě)入0xAAAA,IWDG_RLR中的值就會(huì)被重新加載到計(jì)數(shù)器中從而避免產(chǎn)生看門(mén)狗復(fù)位。
IWDG_PR和IWDG_RLR寄存器具有寫(xiě)保護(hù)功能。要修改這兩個(gè)寄存器的值,必須先向IWDG_KR寄存器中寫(xiě)入0x5555。將其他值寫(xiě)入這個(gè)寄存器將會(huì)打亂操作順序,寄存器將重新被保護(hù)。重裝載操作(即寫(xiě)入0xAAAA)也會(huì)啟動(dòng)寫(xiě)保護(hù)功能。
知道了上面配置的基本原則之后我們就可以開(kāi)始配置我們的看門(mén)狗了,具體配置過(guò)程及配置代碼如下所示:
取消寄存器寫(xiě)保護(hù);
設(shè)置獨(dú)立看門(mén)狗的與分頻系數(shù),確定時(shí)鐘;
設(shè)置看門(mén)狗重裝載值;
使能看門(mén)狗;
應(yīng)用程序喂狗;
配置代碼如下所示:
/** *初始化獨(dú)立看門(mén)狗 *prer:分頻數(shù):0~7(只有低3位有效!) *分頻因子=4*2^prer.但最大值只能是256! *rlr:重裝載寄存器值:低11位有效. *時(shí)間計(jì)算(大概):Tout=((4*2^prer)*rlr)/40(ms). */ voidIWDG_Init(u8prer,u16rlr) { IWDG_WriteAccessCmd(IWDG_WriteAccess_Enable);/*使能對(duì)寄存器IWDG_PR和IWDG_RLR的寫(xiě)操作*/ IWDG_SetPrescaler(prer);/*設(shè)置IWDG預(yù)分頻值:設(shè)置IWDG預(yù)分頻值*/ IWDG_SetReload(rlr);/*設(shè)置IWDG重裝載值*/ IWDG_ReloadCounter();/*按照IWDG重裝載寄存器的值重裝載IWDG計(jì)數(shù)器*/ IWDG_Enable();/*使能IWDG*/ } /** *喂獨(dú)立看門(mén)狗 */ voidIWDG_Feed(void) { IWDG_ReloadCounter();/*reload*/ } /** *main函數(shù) */ voidmain(void) { NVIC_Configuration();//優(yōu)先級(jí)配置 IWDG_Init(4,625);//初始化獨(dú)立看門(mén)狗,分頻數(shù)為64,重裝載值為625,溢出時(shí)間計(jì)算為:64*625/40=1000ms=1s while(1) { delay_ms(500);//0.5秒喂一次狗 IWDG_Feed();//喂狗 } }
對(duì)于溢出時(shí)間的計(jì)算大家可以按照下面的公式計(jì)算:Tout=((4×2^prer) ×rlr) /40 (M3)
獨(dú)立看門(mén)狗所用到的庫(kù)函數(shù):
voidWWDG_DeInit(void); voidWWDG_SetPrescaler(uint32_tWWDG_Prescaler); voidWWDG_SetWindowValue(uint8_tWindowValue); voidWWDG_EnableIT(void); voidWWDG_SetCounter(uint8_tCounter); voidWWDG_Enable(uint8_tCounter); FlagStatusWWDG_GetFlagStatus(void); voidWWDG_ClearFlag(void);
三、窗口看門(mén)狗
窗口看門(mén)狗(WWDG)通常被用來(lái)監(jiān)測(cè)由外部干擾或不可預(yù)見(jiàn)的邏輯條件造成的應(yīng)用程序背離正常的運(yùn)行序列而產(chǎn)生的軟件故障。除非遞減計(jì)數(shù)器的值在 T6 位 (WWDG->CR 的第六位)變成 0 前被刷新,看門(mén)狗電路在達(dá)到預(yù)置的時(shí)間周期時(shí),會(huì)產(chǎn)生一個(gè) MCU 復(fù)位。
在遞減計(jì)數(shù)器達(dá)到窗口配置寄存器(WWDG->CFR)數(shù)值之前,如果 7 位的遞減計(jì)數(shù)器數(shù)值(在控制寄存器中)被刷新,那么也將產(chǎn)生一個(gè) MCU 復(fù)位。這表明遞減計(jì)數(shù)器需要在一個(gè)有限的時(shí)間窗口中被刷新。
但是在使用窗口看門(mén)狗的時(shí)候需要注意寫(xiě)入WWDG_CR 寄存器時(shí),始終將 1 寫(xiě)入 T6 位,以避免生成立即復(fù)位。
下面來(lái)看一下窗口看門(mén)狗的配置步驟以及配置代碼;
使能 WWDG 時(shí)鐘
設(shè)置窗口值和分頻數(shù)
開(kāi)啟 WWDG 中斷并分組
設(shè)置計(jì)數(shù)器初始值并使能看門(mén)狗
窗體看門(mén)狗需要用到的庫(kù)函數(shù);
voidRCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG,ENABLE);//WWDG時(shí)鐘使能 voidWWDG_SetWindowValue(uint8_tWindowValue);//設(shè)置窗口值的函數(shù) voidWWDG_SetPrescaler(uint32_tWWDG_Prescaler);//設(shè)置分頻數(shù)的函數(shù) voidWWDG_EnableIT();//開(kāi)啟窗口看門(mén)狗中斷 voidWWDG_Enable(uint8_tCounter);//設(shè)置計(jì)數(shù)器初始值并使能看門(mén)狗
注意:在編寫(xiě)中斷服務(wù)函數(shù)時(shí)喂狗一定要快,因?yàn)榇翱诳撮T(mén)狗的時(shí)效性比較強(qiáng)
窗口看門(mén)狗的代碼如下:
.c
#ifndef__WDG_H #define__WDG_H #include"sys.h" //獨(dú)立看門(mén)狗 voidIWDG_Init(u8prer,u16rlr); voidIWDG_Feed(void); //窗口看門(mén)狗 voidWWDG_Init(u8tr,u8wr,u32fprer);//初始化WWDG voidWWDG_Set_Counter(u8cnt);//設(shè)置WWDG的計(jì)數(shù)器 voidWWDG_NVIC_Init(void); #endif
.h
#include"wdg.h" #include"led.h" //窗口看門(mén)狗 //保存WWDG計(jì)數(shù)器的設(shè)置值,默認(rèn)為最大. u8WWDG_CNT=0x7f; //初始化窗口看門(mén)狗 //tr:T[6:0],計(jì)數(shù)器值 //wr:W[6:0],窗口值 //fprer:分頻系數(shù)(WDGTB),僅最低2位有效 //Fwwdg=PCLK1/(4096*2^fprer). voidWWDG_Init(u8tr,u8wr,u32fprer) { RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG,ENABLE);//WWDG時(shí)鐘使能 WWDG_CNT=tr&WWDG_CNT;//初始化WWDG_CNT. WWDG_SetPrescaler(fprer);設(shè)置IWDG預(yù)分頻值 WWDG_SetWindowValue(wr);//設(shè)置窗口值 WWDG_Enable(WWDG_CNT);//使能看門(mén)狗,設(shè)置counter. WWDG_ClearFlag();//清除提前喚醒中斷標(biāo)志位 WWDG_NVIC_Init();//初始化窗口看門(mén)狗NVIC WWDG_EnableIT();//開(kāi)啟窗口看門(mén)狗中斷 } //重設(shè)置WWDG計(jì)數(shù)器的值 voidWWDG_Set_Counter(u8cnt) { WWDG_Enable(cnt);//使能看門(mén)狗,設(shè)置counter. } //窗口看門(mén)狗中斷服務(wù)程序 voidWWDG_NVIC_Init() { NVIC_InitTypeDefNVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel=WWDG_IRQn;//WWDG中斷 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=2;//搶占2,子優(yōu)先級(jí)3,組2 NVIC_InitStructure.NVIC_IRQChannelSubPriority=3;//搶占2,子優(yōu)先級(jí)3,組2 NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE; NVIC_Init(&NVIC_InitStructure);//NVIC初始化 } voidWWDG_IRQHandler(void) { WWDG_SetCounter(WWDG_CNT);//當(dāng)禁掉此句后,窗口看門(mén)狗將產(chǎn)生復(fù)位 WWDG_ClearFlag();//清除提前喚醒中斷標(biāo)志位 LED1=!LED1;//LED狀態(tài)翻轉(zhuǎn) }
main.c
#include"led.h" #include"delay.h" #include"key.h" #include"sys.h" #include"usart.h" #include"wdg.h" intmain(void) { delay_init();//延時(shí)函數(shù)初始化 NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//設(shè)置中斷優(yōu)先級(jí)分組為組2:2位搶占優(yōu)先級(jí),2位響應(yīng)優(yōu)先級(jí) uart_init(115200);//串口初始化為115200 LED_Init(); KEY_Init();//按鍵初始化 LED0=0; delay_ms(500); WWDG_Init(0X7F,0X5F,WWDG_Prescaler_8);//計(jì)數(shù)器值為7f,窗口寄存器為5f,分頻數(shù)為8 while(1) { LED0=1; } }
在main函數(shù)里通過(guò) LED0 來(lái)指示 STM32 是否被復(fù)位了,如果被復(fù)位了就會(huì)點(diǎn)亮 500ms。LED0 用來(lái)指示中斷喂狗,每次中斷喂狗翻轉(zhuǎn)一次。
四、結(jié)語(yǔ)
到這里今天的講解內(nèi)容就結(jié)束了,不知道你對(duì)于看門(mén)狗以及看門(mén)狗的使用有沒(méi)有理解,如果覺(jué)得有不理解的地方歡迎大家在下方評(píng)論區(qū)留言一起討論!
-
mcu
+關(guān)注
關(guān)注
146文章
16885瀏覽量
349914 -
看門(mén)狗
+關(guān)注
關(guān)注
10文章
558瀏覽量
70707 -
STM32
+關(guān)注
關(guān)注
2264文章
10854瀏覽量
354289
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論