ck4j3isnybl64029377708.png
" k0 C, G! H* W) ?+ Q
$ t$ V/ C& A9 B1 h4 v, g
看门狗基础:& @7 ]6 V9 k1 R8 V! X. k
STM32微控制器上的看门狗主要有两种类型:独立看门狗(IWDG)和窗口看门狗(WWDG),这两者都是用于监控系统运行状态的机制,但它们在实现和应用上有一些区别:2 m$ E/ q. {; r7 Y3 B
独立看门狗(IWDG):IWDG是一个定时器,其计数器在启用后开始递增。在程序中,你需要定期喂狗(通过向IWDG的寄存器写入特定的值),以防止看门狗超时。否则,如果超过了预定的时间,系统将被认为是失效的,IWDG将生成复位信号,导致系统重启。- _' z/ Z, G- |
! ]* J; v/ z$ Z' Q; Q; Z窗口看门狗(WWDG):3 I1 A6 N: |1 X7 ?2 b3 ^5 m# Q
WWDG是一种更灵活的看门狗类型,允许设置时间窗口。WWDG的计数器在启用后开始递增,但与IWDG不同,WWDG的计数器可以在一个特定的窗口内进行喂狗。这个时间窗口由上限和下限值确定。如果在窗口内喂狗,系统被认为是正常的。如果超出窗口或在窗口内未喂狗,WWDG将生成复位信号,导致系统重启。# k" N: [( i: k6 H) v; i1 G* M
在STM32微控制器中,启动看门狗有几种方式,具体取决于使用的是独立看门狗(IWDG)还是窗口看门狗(WWDG)。
4 g( J- g; x3 F! {启动独立看门狗(IWDG)的方式:通过写入寄存器:配置IWDG的预分频器和重装载寄存器,并在启用IWDG后,定期写入IWDG_KR中的特定值(例如0xAAAA),以喂狗。HAL_IWDG_Refresh(&hiwdg); // 喂狗通过STM32CubeMX和HAL库:在CubeMX中启用IWDG并生成代码,HAL库提供了相应的API来初始化和刷新IWDG。
# Z% a6 n, o v启动窗口看门狗(WWDG)的方式:通过写入寄存器:配置WWDG的计数器窗口上下限值,使能WWDG,并在程序中定期写入WWDG_CR中的特定值(例如0x7F),以喂狗。HAL_WWDG_Refresh(&hwwdg); // 喂狗% [& P* c! z$ p. f' b5 u" z4 `
看门狗相关寄存器:6 [7 M% f$ T5 M
[/ol]在STM32微控制器中,看门狗(IWDG和WWDG)的控制和配置主要通过特定的寄存器完成。以下是关键的看门狗寄存器:
8 ^/ B0 M1 l- ~4 d0 y独立看门狗寄存器(IWDG):/ J/ G/ V/ d0 i8 Y4 E6 r' d
IWDG_KR(0x40003000+0x00):写入特定的值(0xCCCC)来启动或重新加载独立看门狗。
3 u, A8 \8 m! ^& cIWDG_PR(0x40003000 + 0x04):
# u( M6 z. M, G/ u$ l- X/ X预分频器寄存器,用于设置IWDG的时钟源的分频系数。- _ r% o. e, \
IWDG_RLR(0x40003000 + 0x08):重装载寄存器,用于设置IWDG的超时值。当IWDG倒计数到0时,会生成复位信号。
: S( w7 U4 q( r- kIWDG_SR(0x40003000 + 0x0C):状态寄存器,用于指示IWDG的状态,如是否在运行。
8 K: A6 }* j ~* T0 ~8 ?窗口看门狗(WWDG):' U, V7 x: C1 z
WWDG_CR(0x40002C00 + 0x00):控制寄存器,包含使能WWDG的位和计数器窗口值的设置。1 n" ~; z6 n% _6 g3 u0 _
WWDG_CFR(0x40002C00 + 0x04):配置寄存器,包含WWDG的分频系数和计数器窗口的设置。+ C& e6 l: J- k9 y! W
WWDG_SR(0x40002C00 + 0x08):状态寄存器,用于指示WWDG的状态,如是否在运行。$ D# S8 g# k+ @5 H7 C: V7 k
WWDG_CR (0x40002C00 + 0x0C):窗口寄存器,用于设置WWDG的计数器窗口值。* ~6 M" J5 L/ ]
看门狗寄存器代码:#include "stm32f4xx.h"void IWDG_Configuration(void);void Delay(__IO uint32_t nCount);int main(void){ IWDG_Configuration(); while (1) { // 定期喂狗 IWDG->KR = 0xAAAA; // 你的主要代码 // ... // 延时,模拟主循环的执行时间 Delay(500000); }}void IWDG_Configuration(void){ // 使能IWDG时钟 RCC->APB1ENR |= RCC_APB1ENR_IWDGEN; // 设置IWDG的时钟源为LSI(低速内部时钟) RCC->CSR |= RCC_CSR_LSION; while ((RCC->CSR & RCC_CSR_LSIRDY) == 0); // 配置IWDG的预分频器和重装载寄存器 IWDG->KR = 0x5555; // 使能对IWDG_PR和IWDG_RLR寄存器的写操作 IWDG->PR = IWDG_PRESCALER_256; // 预分频器,配置IWDG的时钟 IWDG->RLR = 4095; // 超时时间,根据需求调整 // 启动独立看门狗 IWDG->KR = 0xCCCC;}void Delay(__IO uint32_t nCount){ while(nCount--) { }}
+ c! w+ @: r+ D* V' \& x" K8 ~看门狗HAL库代码:#include "main.h"#include "stm32f4xx_hal.h"IWDG_HandleTypeDef hiwdg;void Systemclock_Config(void);static void MX_GPIO_Init(void);static void MX_IWDG_Init(void);int main(void){ HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_IWDG_Init(); while (1) { // 定期喂狗 HAL_IWDG_Refresh(&hiwdg); // 你的主要代码 }}void SystemClock_Config(void){ RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSI; RCC_OscInitStruct.PLL.PLLM = 8; RCC_OscInitStruct.PLL.PLLN = 160; RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2; RCC_OscInitStruct.PLL.PLLQ = 4; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5) != HAL_OK) { Error_Handler(); }}void MX_IWDG_Init(void){ hiwdg.Instance = IWDG; hiwdg.Init.Prescaler = IWDG_PRESCALER_256; hiwdg.Init.Reload = 4095; if (HAL_IWDG_Init(&hiwdg) != HAL_OK) { Error_Handler(); }}void Error_Handler(void){ while (1) { // 处理错误 }}#ifdef USE_FULL_ASSERTvoid assert_failed(uint8_t *file, uint32_t line){ while (1) { // 处理断言失败的情况 }}#endif==========
) q% a9 N. b% m/ X& r7 d! y往期回顾:HAL库常用函数汇总【不间断更新】
$ r' |/ m0 U- z- P. N2 ]& k6 gST官方的STM32CubeMX培训文档; k; o1 T: Y H9 G1 J* _5 [
小白都看得懂的STM32的DMA知识
. q5 R z" }( ]9 V9 }' ZSTM32CubeMX的外部中断的使用
+ b5 N6 [0 j# i1 D# j+ gADC的低功耗和阻抗问题
1 j8 d+ w* `; y$ V==========5 j+ \9 }7 B7 n; |/ e* R
1 U- r( c$ l6 `2 A7 r
tzz1yix1zwe64029377808.png
8 u. y. O3 L4 ~2 Y9 | w; s! }, H5 v: u7 v
b5ixrkektmb64029377909.png
a& N5 ]5 J @- _# S
' _9 I& u$ ]9 l. o8 g; ^/ z
cphqitgmj3c64029378009.png
|