STM32看门狗–窗口看门狗

STM32看门狗–窗口看门狗stm32有两个看门狗,独立看门狗和窗口看门狗,其实两者的功能是类似的,只是喂狗的限制时间不同。 独立看门狗是限制喂狗时间在0-x内,x由你的相关寄存器决定。喂狗的时间不能过晚。窗口看门狗,所以称之为窗口就是因为其喂狗时间是一个有上下限的范围内,你可以通过设定相关寄存器,设定其上限时间和下限时间。喂狗的时间不能过早也不能过晚。图1&n…

大家好,又见面了,我是你们的朋友全栈君。

stm32有两个看门狗,独立看门狗和窗口看门狗,其实两者的功能是类似的,只是喂狗的限制时间不同。

 

独立看门狗是限制喂狗时间在0-x内,x由你的相关寄存器决定。喂狗的时间不能过晚。

窗口看门狗,所以称之为窗口就是因为其喂狗时间是一个有上下限的范围内,你可以通过设定相关寄存器,设定其上限时间和下限时间。喂狗的时间不能过早也不能过晚。

图1

 STM32看门狗--窗口看门狗

窗口看门狗的

上窗口就是配置寄存器WWDG->CFR里设定的W[6:0];

下窗口是固定的0x40;

当窗口看门狗的计数器在上窗口值之外,或是低于下窗口值都会产生复位。

图2

 

 STM32看门狗--窗口看门狗

&& 上窗口的值可以只有设定,7位二进制数最大只可以设定为127(0x7F),最小又必须大于下窗口的0x40,所以其取值范围为64~127(即:0x40~0x7F);

 

&& 配置寄存器WWDG->CFR中为计数器设定时钟分频系数,确定这个计数器可以定时的时间范围,从而确定窗口的时间范围。

 

&& 窗口看门狗的时钟来自于PCLK1,在时钟配置中,其频率为外部时钟经倍频器后的二分频时钟,即为36MHz,根据手册可以知道其定时时间计算方法:

      

上窗口时间:T_min = 4096 * (2^WDGTB)*(WWDG_CR[6:0] – WWDG_CFR[6:0])/36 (us)

 

 下窗口时间: T_max = 4096 * (2^WDGTB)*(WWDG_CR[6:0] – 0x40)/36 (us)

 

 

36Mhz下相关窗口看门狗的喂狗时间范围:

WDGTB(计数器分频值)最早喂狗时间/us最晚喂狗时间/ms

0                         113              7.28

1                         227              14.56

2                         455              29.12

3                         910              58.25

 

本例直接寄存器实现4种测试模式,测试窗口看门狗的复位原理。

MODE_1 0   // 在30ms时喂狗,在窗口范围内喂狗,LED灯闪烁

MODE_2 0   // 在10ms时喂狗,在窗口范围外导致复位,LED灯常亮

MODE_3 0   // 在100ms时喂狗,在窗口范围外喂狗导致复位

MODE_4 1   // 主函数不执行喂狗,开启提前唤醒中断,在WWDG中断函数中喂狗

库函数实现在提前唤醒中断中,喂狗操作,PA4口LED正常闪烁。当外部中断发生(按下PA0按键),长时间不喂狗,引发窗口看门狗复位。

 

直接操作寄存器

 

相关寄存器如下:

控制寄存器WWDG->CR:低8位有效。

第8位为WDGA,看门狗激活位。低7位[6:0]为看门狗计数器的计数值。

 

配置寄存器WWDG->CFR:低十位有效。

第10位为EWI:提醒唤起中断。此位置1,当计数器值到达0x40时,将产生中断。

第8,9位WDGTB:为计数器分频系数设置为:

00: CK计时器时钟(PCLK1除以4096)除以1     01: CK计时器时钟(PCLK1除以4096)除以2

10: CK计时器时钟(PCLK1除以4096)除以4     11: CK计时器时钟(PCLK1除以4096)除以8

低7位[6:0]:为窗口值。

 

int main(void)

{

#ifdef DEBUG

  debug();

#endif

 

  /* System Clocks Configuration ———————————————*/

  RCC_Configuration();

    

  /* GPIO configuration ——————————————————*/

  GPIO_Configuration();

 

  /*注释-hope:利用窗口看门狗的提前唤醒中断服务子程序来进行独立看门狗计数值的

       重载,每次重载一次计数值LED2就闪烁一次。*/

  /* Check if the system has resumed from WWDG reset ————————-*/

  if(RCC_GetFlagStatus(RCC_FLAG_WWDGRST) != RESET)

  { /* WWDGRST flag set */

    /* Turn on led connected to PC.06 */

    GPIO_WriteBit(GPIOC, GPIO_Pin_6, Bit_SET);

 

       /* Clear reset flags */

    RCC_ClearFlag();

  }

  else

  { /* WWDGRST flag is not set */

    /* Turn off led connected to PC.06 */

    GPIO_WriteBit(GPIOC, GPIO_Pin_6, Bit_RESET);

  }

 

  /* Configure EXTI Line9 to generate an interrupt on falling edge ———–*/

  EXTI_Configuration();

 

/* NVIC configuration ——————————————————–*/

  NVIC_Configuration();

 

/* WWDG configuration ——————————————————–*/

  /* Enable WWDG clock */

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_WWDG, ENABLE);

 

   /*   This parameter can be one of the following values:

       * @arg WWDG_Prescaler_1: WWDG counter clock = (PCLK1/4096)/1

       * @arg WWDG_Prescaler_2: WWDG counter clock = (PCLK1/4096)/2

       * @arg WWDG_Prescaler_4: WWDG counter clock = (PCLK1/4096)/4

       * @arg WWDG_Prescaler_8: WWDG counter clock = (PCLK1/4096)/8

 

/* WWDG clock counter = (PCLK1/4096)/8 = 244 Hz (~4 ms) —PCLK1=8MHZ */

 

  WWDG_SetPrescaler(WWDG_Prescaler_8);

 

  /* Set Window value to 0x41 —设置上窗口边界值*/

  WWDG_SetWindowValue(0x41);

 

  /* Enable WWDG and set counter value to 0x7F,  0x7f-0x41=64 –WWDG timeout = ~4 ms * 64 = 262 ms */

/*看门狗计数器的计数值0x7F—结合上面上窗口边界,计数值从ox7f递减到0x40*/

  WWDG_Enable(0x7F);

 

  /* Clear EWI flag */

  WWDG_ClearFlag();

 

  /* Enable EW interrupt */

  WWDG_EnableIT();

 

  while (1)

  {

  }

}

 

 

/******按键的优先级为0高于窗口看门狗的优先级1*****/

void NVIC_Configuration(void)

{

  NVIC_InitTypeDef NVIC_InitStructure;

 

  /* 2 bits for Preemption Priority and  2 bits for Sub Priority */

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);

 

  NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

 

  NVIC_InitStructure.NVIC_IRQChannel = WWDG_IRQChannel;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;

  NVIC_Init(&NVIC_InitStructure);

}

 

 

void EXTI_Configuration(void)

{

  EXTI_InitTypeDef EXTI_InitStructure;

 

  /* Connect EXTI Line9 to PB.9 */

  GPIO_EXTILineConfig(GPIO_PortSourceGPIOB, GPIO_PinSource9);

 

  /* Configure EXTI Line9 to generate an interrupt on falling edge */

  EXTI_ClearITPendingBit(EXTI_Line9);

  EXTI_InitStructure.EXTI_Line = EXTI_Line9;

  EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt;

  EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling; //下降沿有效

  EXTI_InitStructure.EXTI_LineCmd = ENABLE;

  EXTI_Init(&EXTI_InitStructure);

}

 

在Stm32f10x_it.c文件中:

 

void WWDG_IRQHandler(void)

{

  /* Update WWDG counter */

  WWDG_SetCounter(0x7F);

      

  /* Clear EWI flag */

  WWDG_ClearFlag();

 

  /* Toggle led connected to PC.07 */

  GPIO_WriteBit(GPIOC, GPIO_Pin_7, (BitAction)(1-GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_7)));

}

 

/********按键中断处理程序********/

//注意:这里中断函数里没有清中断标志的操作,这会导致窗口看门狗的提前唤醒中断

          无法进入,最终导致因无法重载看门狗计数器而系统复位;

void EXTI9_5_IRQHandler(void)

{

  if(EXTI_GetITStatus(EXTI_Line9) != RESET)

  {  

    /* Turn off led connected to PC.07 */

       GPIO_WriteBit(GPIOC, GPIO_Pin_7, Bit_RESET);

 

       /* As EXTI line9 pending bit is not cleared, the CPU will execute indefinitely

          this ISR and when the WWDG counter falls to 3Fh the WWDG reset occurs */

  }

}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/138929.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号