2014-06-11 14:23:27 +02:00
|
|
|
/*
|
2015-02-17 13:16:26 +01:00
|
|
|
* Copyright (C) 2014-2015 INRIA
|
|
|
|
* Copyright (C) 2015 Eistec AB
|
2014-06-11 14:23:27 +02:00
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2014-06-18 11:49:45 +02:00
|
|
|
* @ingroup cortex-m3_common
|
2014-06-11 14:23:27 +02:00
|
|
|
* @{
|
|
|
|
*
|
2015-02-17 13:16:26 +01:00
|
|
|
* @file
|
2014-06-11 14:23:27 +02:00
|
|
|
* @brief Crash handling functions implementation for ARM Cortex-based MCUs
|
|
|
|
*
|
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
2015-02-17 13:16:26 +01:00
|
|
|
* @author Joakim Gebart <joakim.gebart@eistec.se>
|
2014-06-11 14:23:27 +02:00
|
|
|
*/
|
|
|
|
|
2015-02-17 13:16:26 +01:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2014-06-11 14:23:27 +02:00
|
|
|
#include "cpu.h"
|
2015-02-17 13:16:26 +01:00
|
|
|
#include "irq.h"
|
2014-06-11 14:23:27 +02:00
|
|
|
#include "lpm.h"
|
|
|
|
#include "crash.h"
|
|
|
|
|
2015-02-17 13:16:26 +01:00
|
|
|
#define PANIC_STR_SIZE 80
|
2014-06-11 14:23:27 +02:00
|
|
|
|
|
|
|
/* "public" variables holding the crash data */
|
2015-02-17 13:16:26 +01:00
|
|
|
char panic_str[PANIC_STR_SIZE];
|
2014-06-11 14:23:27 +02:00
|
|
|
int panic_code;
|
|
|
|
|
|
|
|
/* flag preventing "recursive crash printing loop" */
|
|
|
|
static int crashed = 0;
|
|
|
|
|
|
|
|
/* WARNING: this function NEVER returns! */
|
|
|
|
NORETURN void core_panic(int crash_code, const char *message)
|
|
|
|
{
|
|
|
|
/* copy panic datas to "public" global variables */
|
|
|
|
panic_code = crash_code;
|
2015-02-17 13:16:26 +01:00
|
|
|
strncpy(panic_str, message, sizeof(panic_str));
|
|
|
|
/* strncpy does not add any null-termination. */
|
|
|
|
panic_str[sizeof(panic_str)-1] = '\0';
|
2014-06-11 14:23:27 +02:00
|
|
|
/* print panic message to console (if possible) */
|
|
|
|
if (crashed == 0) {
|
|
|
|
crashed = 1;
|
|
|
|
puts("******** SYSTEM FAILURE ********\n");
|
|
|
|
puts(message);
|
|
|
|
#if DEVELHELP
|
|
|
|
puts("******** RIOT HALTS HERE ********\n");
|
|
|
|
#else
|
|
|
|
puts("******** RIOT WILL REBOOT ********\n");
|
|
|
|
#endif
|
|
|
|
puts("\n\n");
|
|
|
|
}
|
|
|
|
/* disable watchdog and all possible sources of interrupts */
|
2015-02-17 13:16:26 +01:00
|
|
|
disableIRQ();
|
2014-06-11 14:23:27 +02:00
|
|
|
#if DEVELHELP
|
2015-02-17 13:16:26 +01:00
|
|
|
/* The bkpt instruction will signal to the debugger to break here. */
|
|
|
|
__ASM("bkpt #0");
|
2014-06-11 14:23:27 +02:00
|
|
|
/* enter infinite loop, into deepest possible sleep mode */
|
|
|
|
while (1) {
|
|
|
|
lpm_set(LPM_OFF);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
/* DEVELHELP not set => reboot system */
|
|
|
|
(void) reboot(RB_AUTOBOOT);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* tell the compiler that we won't return from this function
|
|
|
|
(even if we actually won't even get here...) */
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|