2015-05-09 18:12:18 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 INRIA
|
|
|
|
* Copyright (C) 2015 Eistec AB
|
2015-05-29 17:17:11 +02:00
|
|
|
* Copyright (C) 2015 Kaspar Schleiser <kaspar@schleiser.de>
|
2015-05-09 18:12:18 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup core_util
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Crash handling functions
|
|
|
|
*
|
|
|
|
* @author Kévin Roussel <Kevin.Roussel@inria.fr>
|
|
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2015-05-29 17:17:11 +02:00
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
2015-05-09 18:12:18 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
2015-09-04 16:49:21 +02:00
|
|
|
#include "assert.h"
|
2016-03-09 01:27:23 +01:00
|
|
|
#include "kernel_defines.h"
|
2015-05-09 18:12:18 +02:00
|
|
|
#include "cpu.h"
|
|
|
|
#include "irq.h"
|
|
|
|
#include "panic.h"
|
|
|
|
#include "arch/panic_arch.h"
|
2017-01-09 18:06:26 +01:00
|
|
|
#include "periph/pm.h"
|
2015-09-17 16:29:02 +02:00
|
|
|
#include "log.h"
|
2015-05-09 18:12:18 +02:00
|
|
|
|
2015-09-12 12:43:15 +02:00
|
|
|
#if defined(DEVELHELP) && defined(MODULE_PS)
|
2015-05-09 18:28:57 +02:00
|
|
|
#include "ps.h"
|
|
|
|
#endif
|
|
|
|
|
2015-11-27 18:08:59 +01:00
|
|
|
const char assert_crash_message[] = "FAILED ASSERTION.";
|
2015-09-04 16:49:21 +02:00
|
|
|
|
2015-05-09 18:12:18 +02:00
|
|
|
/* flag preventing "recursive crash printing loop" */
|
|
|
|
static int crashed = 0;
|
|
|
|
|
2017-01-09 18:06:26 +01:00
|
|
|
void __attribute__((weak)) panic_arch(void) {}
|
|
|
|
|
2015-05-09 18:12:18 +02:00
|
|
|
/* WARNING: this function NEVER returns! */
|
2015-08-26 16:01:04 +02:00
|
|
|
NORETURN void core_panic(core_panic_t crash_code, const char *message)
|
2015-05-09 18:12:18 +02:00
|
|
|
{
|
2016-07-26 12:43:25 +02:00
|
|
|
#ifdef NDEBUG
|
2015-06-01 21:40:38 +02:00
|
|
|
(void) crash_code;
|
2016-07-26 12:43:25 +02:00
|
|
|
#endif
|
2015-06-01 21:40:38 +02:00
|
|
|
|
2015-05-09 18:12:18 +02:00
|
|
|
if (crashed == 0) {
|
2015-05-29 17:17:11 +02:00
|
|
|
/* print panic message to console (if possible) */
|
2015-05-09 18:12:18 +02:00
|
|
|
crashed = 1;
|
2015-09-04 16:50:25 +02:00
|
|
|
#ifndef NDEBUG
|
|
|
|
if (crash_code == PANIC_ASSERT_FAIL) {
|
|
|
|
cpu_print_last_instruction();
|
|
|
|
}
|
|
|
|
#endif
|
2015-09-17 16:29:02 +02:00
|
|
|
LOG_ERROR("*** RIOT kernel panic:\n%s\n\n", message);
|
2015-09-12 12:43:15 +02:00
|
|
|
#ifdef DEVELHELP
|
2015-05-09 18:28:57 +02:00
|
|
|
#ifdef MODULE_PS
|
|
|
|
ps();
|
2015-09-17 16:29:02 +02:00
|
|
|
LOG_ERROR("\n");
|
2015-05-09 18:28:57 +02:00
|
|
|
#endif
|
|
|
|
|
2015-09-17 16:29:02 +02:00
|
|
|
LOG_ERROR("*** halted.\n\n");
|
2015-05-09 18:12:18 +02:00
|
|
|
#else
|
2015-09-17 16:29:02 +02:00
|
|
|
LOG_ERROR("*** rebooting...\n\n");
|
2015-05-09 18:12:18 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/* disable watchdog and all possible sources of interrupts */
|
2016-03-19 09:25:47 +01:00
|
|
|
irq_disable();
|
2015-05-09 18:12:18 +02:00
|
|
|
panic_arch();
|
|
|
|
#ifndef DEVELHELP
|
|
|
|
/* DEVELHELP not set => reboot system */
|
2017-01-09 18:06:26 +01:00
|
|
|
pm_reboot();
|
|
|
|
#else
|
|
|
|
/* DEVELHELP set => power off system */
|
|
|
|
pm_off();
|
2015-05-09 18:12:18 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* tell the compiler that we won't return from this function
|
|
|
|
(even if we actually won't even get here...) */
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|