电子产业一站式赋能平台

PCB联盟网

搜索
查看: 280|回复: 0
收起左侧

STM32的看门狗原理和示例代码

[复制链接]

397

主题

397

帖子

2463

积分

三级会员

Rank: 3Rank: 3

积分
2463
发表于 2023-11-30 21:00:00 | 显示全部楼层 |阅读模式

ck4j3isnybl64029377708.png

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

    tzz1yix1zwe64029377808.png

    8 u. y. O3 L4 ~2 Y9 |  w; s! }, H5 v: u7 v

    b5ixrkektmb64029377909.png

    b5ixrkektmb64029377909.png
      a& N5 ]5 J  @- _# S
    ' _9 I& u$ ]9 l. o8 g; ^/ z

    cphqitgmj3c64029378009.png

    cphqitgmj3c64029378009.png
  • 回复

    使用道具 举报

    发表回复

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则


    联系客服 关注微信 下载APP 返回顶部 返回列表