在项目处于调试期间,Fault处理程序可能只是一个断点指令,调试器遇到这个指令后停止程序的运行。默认情况下,由于非硬Fault被禁能,所有发生的非Fault都会上访成硬Fault,因此只要在硬Fault处理程序中设置一个断点,就可以观察所有Fault信息。
当使用MDK-ARM的RealView编译器时,你可以用下面的C代码替代默认硬Fault处理程序,这段代码检测产品是否连接到一个调试器,只有在连接到一个调试器的情况下,才会执行断点指令。
void HardFault_Handler (void) {
if (CoreDebug ->DHCSR & 1) { // check C_DEBUGEN == 1 -> Debugger Connected
__breakpoint (0); // halt program execution here
}
while (1); // enter endless loop otherwise
}
说明一下,在这段代码中,关于这个CoreDebug->DHCSR也可以在core_cm3.h中找到;__breakpoint()函数是ARM编译器所支持的内部指令,这个函数的作用是在指令流中插入一个断点指令(BKTP)。详细可以查看编译器手册Compiler Reference Guide – Compiler-specific Features - __breakpoint。
为了使能除数为零以及未对齐内存访问产生Fault,应用程序初始化代码要设置SCB->CCR寄存器,下面的C代码清单用于使能除数为零以及未对齐内存访问产生Fault。
SCB ->CCR |= 0x18; /* enable div-by-0 and unaligned fault*/
SCB ->CCR |= 0x18; /* enable div-by-0 and unaligned fault*/
对于最终的应用程序,Fault处理程序或许会按照下面所说的实现:
下面的C代码清单可以用来使能用法、存储器管理和总线Fault:
SCB ->SHCSR |= 0x00007000; // enable Usage Fault, Bus Fault, and MMU Fault
在调试期间,最主要的是要弄清楚触发了哪类Fault,什么原因触发了Fault以及定位到触发Fault的代码,可以利用一个空闲串口当作调试用,将以上信息发给PC,通过串口调试助手接收这些Fault信息。
主要步骤如下:
补充一些基础知识,有利于理解下面的代码:
堆栈:
Cortex-M3的堆栈是使用“向下生长的满栈”模型,SP指针指向最后一个被压入堆栈的32位数值。在下一次压栈时,SP先自减4,再存入新的数值。POP操作正好相反,先弹出当前SP指针处的32位数值,再将SP的值增4.
Cortex-M3的异常/中断过程:
LR:在出入ISR的时候,LR保存一些在异常返回时用到的特殊位。
下面以硬Fault处理为例,介绍一下如何将Fault信息上报到PC的调试助手上。
1. 在程序初始化代码中,使能非硬Fault(使用Keil MDK编译器,必须包含core_cm3.h头文件)
static __INLINE void EnableFault(void)
{
SCB->SHCSR |= (SCB_SHCSR_USGFAULTENA_Msk|SCB_SHCSR_BUSFAULTENA_Msk|SCB_SHCSR_MEMFAULTENA_Msk);
}
2. 编写硬Fault处理程序
/*
* 截获硬Fault异常
* arg:堆栈指针
*/
void HardFaultHandle(unsigned int *arg)
{
unsigned int stacked_r0,stacked_r1,stacked_r2,stacked_r3,stacked_r12,stacked_lr,stacked_pc,stacked_psr;
stacked_r0=((unsigned long)arg[0]);
stacked_r1=((unsigned long)arg[1]);
stacked_r2=((unsigned long)arg[2]);
stacked_r3=((unsigned long)arg[3]);
stacked_r12=((unsigned long)arg[4]);
stacked_lr=((unsigned long)arg[5]);
stacked_pc=((unsigned long)arg[6]);
stacked_psr=((unsigned long)arg[7]);
PLC_DEBUGF(TEST_DEBUG,("致命错误:系统发生硬Fault!!\n"));
PLC_DEBUGF(TEST_DEBUG,("捕获错误发生时的环境,上报Fault状态寄存器:\n"));
PLC_DEBUGF(TEST_DEBUG,("R0= 0x%x\n",stacked_r0));
PLC_DEBUGF(TEST_DEBUG,("R1= 0x%x\n",stacked_r1));
PLC_DEBUGF(TEST_DEBUG,("R2= 0x%x\n",stacked_r2));
PLC_DEBUGF(TEST_DEBUG,("R3= 0x%x\n",stacked_r3));
PLC_DEBUGF(TEST_DEBUG,("R12= 0x%x\n",stacked_r12));
PLC_DEBUGF(TEST_DEBUG,("LR= 0x%x\n",stacked_lr));
PLC_DEBUGF(TEST_DEBUG,("PC= 0x%x\n",stacked_pc));
PLC_DEBUGF(TEST_DEBUG,("PSR= 0x%x\n",stacked_psr));
PLC_DEBUGF(TEST_DEBUG,("HFSR= 0x%x\n",HFSR));
PLC_DEBUGF(TEST_DEBUG,("BFSR= 0x%x\n",BFSR));
PLC_DEBUGF(TEST_DEBUG,("BFAR= 0x%x\n",BFAR));
PLC_DEBUGF(TEST_DEBUG,("MMSR= 0x%x\n",MMSR));
PLC_DEBUGF(TEST_DEBUG,("MMAR= 0x%x\n",MMAR));
PLC_DEBUGF(TEST_DEBUG,("UFSR= 0x%x\n",UFSR));
HFSR=0xFFFFFFFF;
BFSR=0xFF;
MMSR=0xFF;
UFSR=0xFFFF;
while(1);
}
3. 在启动代码中,将默认硬Fault处理程序更换为自己需要的Fault处理程序
HardFault_Handler\
PROC
IMPORT HardFaultHandle
TST LR,#4
ITE EQ
MRSEQ R0,MSP
MRSNE R0,PSP
B HardFaultHandle
ENDP