CH582,CH592,CH57x芯片死机通过看门狗中断打印PC指针使用示例
简介:
看门狗超时时长与单片机系统主频有关,且看门狗功能是一个8位的递增计数器,一次计数时钟周期为131072/Fsys,
因此,例如:
1、主频如果是32M,看门狗最大时间 (131072/32000000)×255=1.04448s
2、主频如果是60M,看门狗最大时间 (131072/60000000)×255=0.557056s
#include "CH58x_common.h"
/********************************************************************* * @fn DebugInit * * @brief 调试初始化 * * @return none */ void DebugInit(void) { GPIOA_SetBits(GPIO_Pin_9); GPIOA_ModeCfg(GPIO_Pin_8, GPIO_ModeIN_PU); GPIOA_ModeCfg(GPIO_Pin_9, GPIO_ModeOut_PP_5mA); UART1_DefInit(); } void WWDG_Config(void) { WWDG_SetCounter(0);//喂狗处理 WWDG_ITCfg(ENABLE);//配置看门狗中断 WWDG_ResetCfg(DISABLE);//看门狗中断生效后不复位,如果需要产生看门狗中断后立即复位,则参数填写ENABLE PFIC_EnableIRQ(WDOG_BAT_IRQn);//使能看门狗中断生效 } void TEST_Code(void) { WWDG_SetCounter(0);//喂狗处理 static uint32_t count=0; count++; while(count>32000);//写一个死循环,产生现象后读取PC指针,去.lst文件中查看跑死在哪个地方 } /********************************************************************* * @fn main * * @brief 主函数 * * @return none */ int main() { SetSysClock(CLK_SOURCE_PLL_60MHz); /* 配置串口调试 */ DebugInit(); PRINT("Start @ChipID=%02X\n", R8_CHIP_ID); WWDG_Config(); while(1) { TEST_Code(); } } uint32_t __get_MEPC(void); uint32_t __get_MTVAL(void); uint32_t __get_MCAUSE(void); __attribute__((interrupt("WCH-Interrupt-fast"))) __attribute__((section(".highcode"))) void WDOG_BAT_IRQHandler(void) { GPIOA_ModeCfg(1<<9, GPIO_ModeOut_PP_5mA);//串口IO引脚配置 UART1_DefInit(); UART1_BaudRateCfg(115200); //串口初始化,波特率配置 uint32_t mepc = __get_MEPC(); uint32_t mtval = __get_MTVAL(); uint32_t mcause = __get_MCAUSE(); while(1)
{
printf("mepc = %08x\nmtval = %08x\nmcause = %08x\n",mepc,mtval,mcause); //异常后,如果没有喂狗,看门狗中断后,打印pc指针
}
}
跑死后查看PC指针停在了0x00000bbe,前去.lst文件中查找对应函数