1)实验平台:正点原子APM32E103最小系统板
2)平台购买地址:https://detail.tmall.com/item.htm?id=609294757420
3)全套实验源码+手册+视频下载地址: http://www.openedv.com/docs/boards/xiaoxitongban
本章介绍APM32E103窗口看门狗(WWDT)的使用,窗口看门狗与独立看门狗一样能够帮助CPU在进入错误状态或程序跑飞时进行复位,不过窗口看门狗相对于独立看门狗限制了“喂狗”的最小间隔,若两次“喂狗”的间隔太短,一样会产生复位。通过本章的学习,读者将学习到WWDT的使用。
本章分为如下几个小节:
15.1 硬件设计
15.2 程序设计
15.3 下载验证
15.1 硬件设计
15.1.1 例程功能
#include "apm32e10x.h"
#include "apm32e10x_wwdt.h"
void example_fun(void)
{
/* 使能WWDT */
WWDT_Enable(0x7F);
}
②:配置WWDT预分频系数
该函数用于配置WWDT的预分频系数,预分频系数决定了WWDT计数的快慢,其函数原型如下所示:
void WWDT_ConfigTimebase(WWDT_TIME_BASE_T timeBase);
该函数的形参描述,如下表所示:
形参 描述
timeBase WWDT的时基预分频值
例如:WWDT_TIME_BASE_1、WWDT_TIME_BASE_2等(在apm32e10x _wwdt.h文件中有定义)
表15.2.1.3 函数WWDT_ConfigTimebase()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表15.2.1.4 函数WWDT_ConfigTimebase()返回值描述
该函数的使用示例,如下所示:
#include "apm32e10x.h"
#include "apm32e10x_wwdt.h"
void example_fun(void)
{
/* 配置WWDT的预分频系数
* WWDT的计数时钟为(PCLK1/4096)/8
*/
WWDT_ConfigTimebase(WWDT_TIME_BASE_8);
}
③:配置WWDT窗口值
该函数用于配置WWDT的窗口值,在WWDT的计数值减少到窗口值之前进行“喂狗”,将为导致WWDT复位,其函数原型如下所示:
void WWDT_ConfigWindowData(uint8_t windowData);
该函数的形参描述,如下表所示:
形参 描述
windowData WWDT的窗口值
表15.2.1.5 函数WWDT_ConfigWindowData()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表15.2.1.6 函数WWDT_ConfigWindowData()返回值描述
该函数的使用示例,如下所示:
#include "apm32e10x.h"
#include "apm32e10x_wwdt.h"
void example_fun(void)
{
/* 配置WWDT窗口值 */
WWDT_ConfigWindowData(0x5F);
}
④:使能WWDT提前唤醒中断
该函数用于使能WWDT的提前唤醒中断,其函数原型如下所示:
void WWDT_EnableEWI(void);
该函数的形参描述,如下表所示:
形参 描述
无 无
表15.2.1.7 函数WWDT_EnableEWI()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表15.2.1.8 函数WWDT_EnableEWI()返回值描述
该函数的使用示例,如下所示:
#include "apm32e10x.h"
#include "apm32e10x_wwdt.h"
void example_fun(void)
{
/* 使能WWDT提前唤醒中断 */
WWDT_EnableEWI();
}
⑤:配置WWDT中断
请见第12.2.3小节中配置中断的相关内容。
⑥:配置WWDT的计数值
该函数用于配置WWDT的计数值,也就是所谓的“喂狗”,其函数原型如下:
void WWDT_ConfigCounter(uint8_t counter);
该函数的形参描述,如下表所示:
形参 描述
counter WWDT的计数值
表15.2.1.9 函数WWDT_ConfigCounter()形参描述
该函数的返回值描述,如下表所示:
返回值 描述
无 无
表15.2.1.10 函数WWDT_ConfigCounter()返回值描述
该函数的使用示例,如下表所示:
#include "apm32e10x.h"
#include "apm32e10x_wwdt.h"
void example_fun(void)
{
/* 配置WWDT的计数值(喂狗) */
WWDT_ConfigCounter(0x7F);
}
15.2.2 看门狗驱动
本章实验的看门狗驱动主要负责向应用层提供WWDT的初始化函数,并实现WWDT的提前唤醒中断服务函数,在WWDT的提前唤醒中断服务函数中执行“喂狗”操作。本章实验中,看门狗驱动的驱动代码包括wdt.c和wdt.h两个文件。
看门狗驱动中WWDT的初始化函数,如下所示:
```c
/**
* @brief 初始化窗口看门狗
* @param tr: T[5:0],计数器值
* @param tw: W[6:0],窗口值
* @param fprer:分频系数(WDGTB),范围:WWDT_TIME_BASE_1~WWDT_TIME_BASE_8,
表示2^WDGTB分频
* Twwdt=TPCLK1 * 2^(WTB) * (T[5:0]+1). 一般PCLK1 = 60MHz
最大-最小超时时间参考《APM32E103xCxE用户手册 V1.3》 第十八章
* @retval 无
*/
void wwdt_init(uint8_t tr, uint8_t tw, WWDT_TIME_BASE_T fprer)
{
RCM_EnableAPB1PeriphClock(RCM_APB1_PERIPH_WWDT);/* 打开时钟 */
wwdt_cnt = tr; /* 初始化wwdt_cnt传进参数tr*/
WWDT_ConfigTimebase(fprer); /* 设置WWDT分频系数 */
WWDT_ConfigWindowData(tw); /* 窗口值 */
WWDT_Enable(tr); /* 设置计数器的值,使能WWDT */
WWDT_ClearFlag(); /* 清楚唤醒中断标志位 */
wwdt_nvic_init(); /* 窗口看门狗中断初始化 */
WWDT_EnableEWI(); /* 使能窗口看门狗中断 */
}
从上面的代码中可以看出,WWDT的初始化函数中除了使能WWDT的时钟外,还分别执行了使能WWDT、配置WWDT的预分频系数和窗口值、使能WWDT的提前唤醒中断和使能WWDT中断的操作。随后,在WWDT计数值减少到0x40时,将会产生提前唤醒中断。
```c
/**
* @brief 窗口看门狗中断服务程序
* @param 无
* @retval 无
*/
void WWDT_IRQHandler(void)
{
if (WWDT_ReadFlag() != RESET)
{
WWDT_ConfigCounter(wwdt_cnt);/* 刷新递减计数器的值,当计数器值小于窗口值喂狗 */
LED1_TOGGLE(); /* LED1闪烁 */
WWDT_ClearFlag(); /* 清楚唤醒中断标志位 */
}
}
从上面的代码中可以看出,在窗口看门狗的提前唤醒中断服务函数中,对WWDT进行了喂狗操作,同时LED1的亮灭状态发生了翻转。通过代码设置LED1亮灭状态翻转的目的在于方便观察“喂狗”这一过程。
15.2.3 实验应用代码
本实验的应用代码,如下所示:
int main(void)
{
NVIC_ConfigPriorityGroup(NVIC_PRIORITY_GROUP_4); /* 设置中断优先级分组为组4 */
sys_apm32_clock_init(15); /* 配置系统时钟 */
delay_init(120); /* 初始化延时功能 */
usart_init(115200); /* 初始化串口 */
led_init(); /* 初始化LED */
LED0(0); /* 先点亮LED0 */
delay_ms(500); /* 延时500ms */
wwdt_init(0x7F, 0x5F, WWDT_TIME_BASE_8); /* 初始化窗口看门狗 */
while (1)
{
LED0(1); /* 关闭LED0 */
}
}
从上面的代码中可以看出,初始化LED后,LED0会被点亮500毫秒,随后初始化窗口看门狗并熄灭LED0,若窗口看门狗没有正常被“喂狗”,那么程序复位,可以看到LED0再次亮起和熄灭,如果窗口看门狗被正常“喂狗”,则可以看到LED1快速闪烁。
15.3 下载验证
在完成编译和烧录操作后,可以看到板子上的LED0在上电时,亮起并熄灭一次,随后LED1不断闪烁,说明WWDT“喂狗”正常。