max32630-fthr: add hardware_fault handler to dump state, prepare for console input

This commit is contained in:
Matthias Ringwald 2017-08-18 10:05:38 +02:00
parent ec2bbb878f
commit 7dabc299c3

View File

@ -43,6 +43,10 @@
#include "spim.h"
#include "max14690n.h"
#define UART_ERRORS (MXC_F_UART_INTEN_RX_FIFO_OVERFLOW | \
MXC_F_UART_INTEN_RX_FRAMING_ERR | \
MXC_F_UART_INTEN_RX_PARITY_ERR)
/***** Global Variables *****/
// LEDs
@ -88,6 +92,53 @@ void mxc_assert(const char *expr, const char *file, int line)
while (1);
}
/******************************************************************************/
int Console_Init(void)
{
int err;
if ((err = UART_Init(MXC_UART_GET_UART(CONSOLE_UART), &console_uart_cfg, &console_sys_cfg)) != E_NO_ERROR) {
MXC_ASSERT_FAIL();
return err;
}
// Setup the interrupt
NVIC_ClearPendingIRQ(MXC_UART_GET_IRQ(CONSOLE_UART));
NVIC_DisableIRQ(MXC_UART_GET_IRQ(CONSOLE_UART));
NVIC_SetPriority(MXC_UART_GET_IRQ(CONSOLE_UART), 1);
NVIC_EnableIRQ(MXC_UART_GET_IRQ(CONSOLE_UART));
MXC_UART1->inten |= (MXC_F_UART_INTEN_RX_FIFO_NOT_EMPTY | UART_ERRORS);
return E_NO_ERROR;
}
/******************************************************************************/
int Console_PrepForSleep(void)
{
fflush(stdout);
return UART_PrepForSleep(MXC_UART_GET_UART(CONSOLE_UART));
}
void UART0_IRQHandler(void)
{
UART_Handler(MXC_UART_GET_UART(0));
}
void UART1_IRQHandler(void)
{
UART_Handler(MXC_UART_GET_UART(1));
}
void UART2_IRQHandler(void)
{
UART_Handler(MXC_UART_GET_UART(2));
}
void UART3_IRQHandler(void)
{
UART_Handler(MXC_UART_GET_UART(3));
}
/******************************************************************************/
int Board_Init(void)
{
@ -122,22 +173,163 @@ int Board_Init(void)
return E_NO_ERROR;
}
/******************************************************************************/
int Console_Init(void)
#ifdef CRASH_DUMP
void FaultISR_C(uint32_t *hardfault_args)
{
int err;
unsigned int stacked_r0;
unsigned int stacked_r1;
unsigned int stacked_r2;
unsigned int stacked_r3;
unsigned int stacked_r12;
unsigned int stacked_lr;
unsigned int stacked_pc;
unsigned int stacked_psr;
if ((err = UART_Init(MXC_UART_GET_UART(CONSOLE_UART), &console_uart_cfg, &console_sys_cfg)) != E_NO_ERROR) {
MXC_ASSERT_FAIL();
return err;
}
volatile unsigned char mmsr;
volatile unsigned char bfsr;
volatile unsigned short ufsr;
volatile unsigned int hfsr;
return E_NO_ERROR;
stacked_r0 = ((unsigned int) hardfault_args[0]);
stacked_r1 = ((unsigned int) hardfault_args[1]);
stacked_r2 = ((unsigned int) hardfault_args[2]);
stacked_r3 = ((unsigned int) hardfault_args[3]);
stacked_r12 = ((unsigned int) hardfault_args[4]);
stacked_lr = ((unsigned int) hardfault_args[5]);
stacked_pc = ((unsigned int) hardfault_args[6]);
stacked_psr = ((unsigned int) hardfault_args[7]);
printf("\n\n[Hard fault handler - all numbers in hex]\n");
printf("R0 = 0x%08x\n", stacked_r0);
printf("R1 = 0x%08x\n", stacked_r1);
printf("R2 = 0x%08x\n", stacked_r2);
printf("R3 = 0x%08x\n", stacked_r3);
printf("R12 = 0x%08x\n", stacked_r12);
printf("LR [R14] = 0x%08x subroutine call return address\n", stacked_lr);
printf("PC [R15] = 0x%08x program counter address\n", stacked_pc);
printf("PSR = 0x%08x\n", stacked_psr);
printf("MMAR = 0x%08x memory manage fault address\n", (*((volatile unsigned int *) (0xE000ED34))));
printf("BFAR = 0x%08x bus fault address\n", (*((volatile unsigned int *) (0xE000ED38))));
/***********************************************************************************************
* Memory Management Fault Status Register: (0xE000ED28)
* Bit Name Description
* 7 MMARVALID MMAR is valid (0x40)
* 4 MSTKERR Stacking error (0x10)
* 3 MUNSTKERR Unstacking error (0x8)
* 1 DACCVIOL Data access violation (0x2)
* 0 IACCVIOL Instruction access violation (0x1)
***********************************************************************************************/
mmsr = (*((volatile unsigned char *) (0xE000ED28)));
printf("MMSR = 0x%02x ", mmsr);
if (mmsr & 0x40)
printf("MMARVALID: MMAR is valid ");
if (mmsr & 0x10)
printf("MSTKERR: Stacking error\n");
else if (mmsr & 0x8)
printf("MUNSTKERR: Unstacking error\n");
else if (mmsr & 0x2)
printf("DACCVIOL: Data access violation\n");
else if (mmsr & 0x1)
printf("IACCVIOL: Instruction access violation\n");
else
printf("\n");
/***********************************************************************************************
* Bus Fault Status Register: (0xE000ED28)
* Bit Name Description
* 7 BFARVALID BFAR is valid (0x80)
* 4 STKERR Stacking error (0x10)
* 3 UNSTKERR Unstacking error (0x8)
* 2 IMPREISERR Imprecise data access violation (0x4)
* 1 PRECISERR Precise data access violation (0x2)
* 0 IBUSERR Instruction access violation (0x1)
***********************************************************************************************/
bfsr = (*((volatile unsigned char *) (0xE000ED29)));
printf("BFSR = 0x%02x ", bfsr);
if (bfsr & 0x80)
printf("BFARVALID: BFAR is valid ");
if (bfsr & 0x10)
printf("STKERR: Stacking error\n");
else if (bfsr & 0x8)
printf("UNSTKERR: Unstacking error\n");
else if (bfsr & 0x4)
printf("IMPREISERR: Imprecise data access violation\n");
else if (bfsr & 0x2)
printf("PRECISERR: Precise data access violation\n");
else if (bfsr & 0x1)
printf("IBUSERR: Instruction access violation\n");
else
printf("\n");
/***********************************************************************************************
* Usage Fault Status Register: (0xE000ED2A)
* Bit Name Description
* 9 DIVBYZERO Divide by zero will take place (0x200)
* 8 UNALIGNED Unaligned access will take place (0x100)
* 3 NOCP Attempt to execute a coprocessor instruction (0x8)
* 2 INVPC Attempt to do exception with bad value (0x4)
* 1 INVSTATE Attempt to switch to invalid state (0x2)
* 0 UNDEFINSTR Attempt to execute an undefined instruction (0x1)
***********************************************************************************************/
ufsr = (*((volatile unsigned short *) (0xE000ED2A)));
printf("UFSR = 0x%04x ", ufsr);
if (ufsr & 0x200)
printf("DIVBYZERO: Divide by zero will take place\n");
else if (ufsr & 0x100)
printf("UNALIGNED: Unaligned access will take place\n");
else if (ufsr & 0x8)
printf("NOCP: Attempt to execute a coprocessor instruction\n");
else if (ufsr & 0x4)
printf("INVPC: Attempt to do exception with bad value\n");
else if (ufsr & 0x2)
printf("INVSTATE: Attempt to switch to invalid state\n");
else if (ufsr & 0x1)
printf("UNDEFINSTR: Attempt to execute an undefined instruction\n");
else
printf("\n");
/***********************************************************************************************
* Usage Fault Status Register: (0xE000ED2A)
* Bit Name Description
* 31 DEBUGEVT Hard fault caused by debug event (0x8000_0000)
* 30 FORCED Hard fault caused by bus/memory management/usage fault (0x4000_0000)
* 1 VECTBL Hard fault caused by failed vector fetch (0x1)
***********************************************************************************************/
hfsr = (*((volatile unsigned int *) (0xE000ED2C)));
printf("HFSR = 0x%08x ", hfsr);
if (hfsr & 0x80000000)
printf("DEBUGEVT: Hard fault caused by debug event\n");
else if (hfsr & 0x40000000)
printf("FORCED: Hard fault caused by bus/memory management/usage fault\n");
else if (hfsr & 0x1)
printf("VECTBL: Hard fault caused by failed vector fetch\n");
else
printf("\n");
printf ("AFSR = 0x%08x\n", (*((volatile unsigned int *)(0xE000ED3C))));
printf ("SCB_SHCSR = %x\n", SCB->SHCSR);
while (1) ; /* Spin so we can use a debugger to anlayzer the situation */
}
/******************************************************************************/
int Console_PrepForSleep(void)
#else /* ENABLE_CRASH_DUMP */
void FaultISR_C(uint32_t *hardfault_args)
{
fflush(stdout);
return UART_PrepForSleep(MXC_UART_GET_UART(CONSOLE_UART));
/* spin so we can use a debugger to anlayze the situation */
while(1);
/* reset the system */
//NVIC_SystemReset();
}
#endif /* CRASH_DUMP */
void HardFault_Handler(void)
{
printf("HardFault_Handler! (main)\n");
__asm(
" TST LR, #4\n"
" ITE EQ \n"
" MRSEQ R0, MSP \n"
" MRSNE R0, PSP \n"
" B FaultISR_C \n");
}