mirror of
https://github.com/Telecominfraproject/OpenCellular.git
synced 2025-12-30 02:20:48 +00:00
Even when CONFIG_RESTRICTED_CONSOLE_COMMANDS is enabled, there are many commands that can't do anything dangerous. This marks some of those commands as safe to use, even when restrictions are enforced. I'm only marking commands that are used by the Cr50, since that's the only board that has restrictions. BUG=chrome-os-partner:55322 BRANCH=none TEST=make buildall, test on Cr50 hardware Change-Id: I6289d332830175b6adcb6b20cb4c21d01d27a25e Signed-off-by: Bill Richardson <wfrichar@chromium.org> Reviewed-on: https://chromium-review.googlesource.com/376188 Reviewed-by: Randall Spangler <rspangler@chromium.org>
244 lines
5.4 KiB
C
244 lines
5.4 KiB
C
/* Copyright (c) 2013 The Chromium OS Authors. All rights reserved.
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#include "common.h"
|
|
#include "console.h"
|
|
#include "cpu.h"
|
|
#include "hooks.h"
|
|
#include "host_command.h"
|
|
#include "panic.h"
|
|
#include "printf.h"
|
|
#include "system.h"
|
|
#include "task.h"
|
|
#include "timer.h"
|
|
#include "uart.h"
|
|
#include "util.h"
|
|
|
|
/* Panic data goes at the end of RAM. */
|
|
static struct panic_data * const pdata_ptr = PANIC_DATA_PTR;
|
|
|
|
/**
|
|
* Add a character directly to the UART buffer.
|
|
*
|
|
* @param context Context; ignored.
|
|
* @param c Character to write.
|
|
* @return 0 if the character was transmitted, 1 if it was dropped.
|
|
*/
|
|
#ifndef CONFIG_DEBUG_PRINTF
|
|
static int panic_txchar(void *context, int c)
|
|
{
|
|
if (c == '\n')
|
|
panic_txchar(context, '\r');
|
|
|
|
/* Wait for space in transmit FIFO */
|
|
while (!uart_tx_ready())
|
|
;
|
|
|
|
/* Write the character directly to the transmit FIFO */
|
|
uart_write_char(c);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void panic_puts(const char *outstr)
|
|
{
|
|
/* Flush the output buffer */
|
|
uart_flush_output();
|
|
|
|
/* Put all characters in the output buffer */
|
|
while (*outstr)
|
|
panic_txchar(NULL, *outstr++);
|
|
|
|
/* Flush the transmit FIFO */
|
|
uart_tx_flush();
|
|
}
|
|
|
|
void panic_printf(const char *format, ...)
|
|
{
|
|
va_list args;
|
|
|
|
/* Flush the output buffer */
|
|
uart_flush_output();
|
|
|
|
va_start(args, format);
|
|
vfnprintf(panic_txchar, NULL, format, args);
|
|
va_end(args);
|
|
|
|
/* Flush the transmit FIFO */
|
|
uart_tx_flush();
|
|
}
|
|
#endif
|
|
|
|
/**
|
|
* Display a message and reboot
|
|
*/
|
|
void panic_reboot(void)
|
|
{
|
|
panic_puts("\n\nRebooting...\n");
|
|
system_reset(0);
|
|
}
|
|
|
|
#ifdef CONFIG_DEBUG_ASSERT_REBOOTS
|
|
#ifdef CONFIG_DEBUG_ASSERT_BRIEF
|
|
void panic_assert_fail(const char *fname, int linenum)
|
|
{
|
|
panic_printf("\nASSERTION FAILURE at %s:%d\n", fname, linenum);
|
|
#ifdef CONFIG_SOFTWARE_PANIC
|
|
software_panic(PANIC_SW_ASSERT, linenum);
|
|
#else
|
|
panic_reboot();
|
|
#endif
|
|
}
|
|
#else
|
|
void panic_assert_fail(const char *msg, const char *func, const char *fname,
|
|
int linenum)
|
|
{
|
|
panic_printf("\nASSERTION FAILURE '%s' in %s() at %s:%d\n",
|
|
msg, func, fname, linenum);
|
|
#ifdef CONFIG_SOFTWARE_PANIC
|
|
software_panic(PANIC_SW_ASSERT, linenum);
|
|
#else
|
|
panic_reboot();
|
|
#endif
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
void panic(const char *msg)
|
|
{
|
|
panic_printf("\n** PANIC: %s\n", msg);
|
|
panic_reboot();
|
|
}
|
|
|
|
struct panic_data *panic_get_data(void)
|
|
{
|
|
return pdata_ptr->magic == PANIC_DATA_MAGIC ? pdata_ptr : NULL;
|
|
}
|
|
|
|
static void panic_init(void)
|
|
{
|
|
#ifdef CONFIG_HOSTCMD_EVENTS
|
|
struct panic_data *addr = panic_get_data();
|
|
|
|
/* Notify host of new panic event */
|
|
if (addr && !(addr->flags & PANIC_DATA_FLAG_OLD_HOSTEVENT)) {
|
|
host_set_single_event(EC_HOST_EVENT_PANIC);
|
|
addr->flags |= PANIC_DATA_FLAG_OLD_HOSTEVENT;
|
|
}
|
|
#endif
|
|
}
|
|
DECLARE_HOOK(HOOK_INIT, panic_init, HOOK_PRIO_DEFAULT);
|
|
|
|
#ifdef CONFIG_CMD_STACKOVERFLOW
|
|
static void stack_overflow_recurse(int n)
|
|
{
|
|
ccprintf("+%d", n);
|
|
|
|
/*
|
|
* Force task context switch, since that's where we do stack overflow
|
|
* checking.
|
|
*/
|
|
msleep(10);
|
|
|
|
stack_overflow_recurse(n+1);
|
|
|
|
/*
|
|
* Do work after the recursion, or else the compiler uses tail-chaining
|
|
* and we don't actually consume additional stack.
|
|
*/
|
|
ccprintf("-%d", n);
|
|
}
|
|
#endif /* CONFIG_CMD_STACKOVERFLOW */
|
|
|
|
/*****************************************************************************/
|
|
/* Console commands */
|
|
#ifdef CONFIG_CMD_CRASH
|
|
static int command_crash(int argc, char **argv)
|
|
{
|
|
if (argc < 2)
|
|
return EC_ERROR_PARAM1;
|
|
|
|
if (!strcasecmp(argv[1], "assert")) {
|
|
ASSERT(0);
|
|
} else if (!strcasecmp(argv[1], "divzero")) {
|
|
int zero = 0;
|
|
|
|
cflush();
|
|
ccprintf("%08x", (long)1 / zero);
|
|
} else if (!strcasecmp(argv[1], "udivzero")) {
|
|
int zero = 0;
|
|
|
|
cflush();
|
|
ccprintf("%08x", (unsigned long)1 / zero);
|
|
#ifdef CONFIG_CMD_STACKOVERFLOW
|
|
} else if (!strcasecmp(argv[1], "stack")) {
|
|
stack_overflow_recurse(1);
|
|
#endif
|
|
} else if (!strcasecmp(argv[1], "unaligned")) {
|
|
cflush();
|
|
ccprintf("%08x", *(volatile int *)0xcdef);
|
|
} else if (!strcasecmp(argv[1], "watchdog")) {
|
|
while (1)
|
|
;
|
|
} else if (!strcasecmp(argv[1], "hang")) {
|
|
interrupt_disable();
|
|
while (1)
|
|
;
|
|
} else {
|
|
return EC_ERROR_PARAM1;
|
|
}
|
|
|
|
/* Everything crashes, so shouldn't get back here */
|
|
return EC_ERROR_UNKNOWN;
|
|
}
|
|
DECLARE_CONSOLE_COMMAND(crash, command_crash,
|
|
"[assert | divzero | udivzero"
|
|
#ifdef CONFIG_CMD_STACKOVERFLOW
|
|
" | stack"
|
|
#endif
|
|
" | unaligned | watchdog | hang]",
|
|
"Crash the system (for testing)");
|
|
#endif
|
|
|
|
static int command_panicinfo(int argc, char **argv)
|
|
{
|
|
if (pdata_ptr->magic == PANIC_DATA_MAGIC) {
|
|
ccprintf("Saved panic data:%s\n",
|
|
(pdata_ptr->flags & PANIC_DATA_FLAG_OLD_CONSOLE ?
|
|
"" : " (NEW)"));
|
|
|
|
panic_data_print(pdata_ptr);
|
|
|
|
/* Data has now been printed */
|
|
pdata_ptr->flags |= PANIC_DATA_FLAG_OLD_CONSOLE;
|
|
} else {
|
|
ccprintf("No saved panic data available.\n");
|
|
}
|
|
return EC_SUCCESS;
|
|
}
|
|
DECLARE_SAFE_CONSOLE_COMMAND(panicinfo, command_panicinfo,
|
|
NULL,
|
|
"Print info from a previous panic");
|
|
|
|
/*****************************************************************************/
|
|
/* Host commands */
|
|
|
|
int host_command_panic_info(struct host_cmd_handler_args *args)
|
|
{
|
|
if (pdata_ptr->magic == PANIC_DATA_MAGIC) {
|
|
ASSERT(pdata_ptr->struct_size <= args->response_max);
|
|
memcpy(args->response, pdata_ptr, pdata_ptr->struct_size);
|
|
args->response_size = pdata_ptr->struct_size;
|
|
|
|
/* Data has now been returned */
|
|
pdata_ptr->flags |= PANIC_DATA_FLAG_OLD_HOSTCMD;
|
|
}
|
|
|
|
return EC_RES_SUCCESS;
|
|
}
|
|
DECLARE_HOST_COMMAND(EC_CMD_GET_PANIC_INFO,
|
|
host_command_panic_info,
|
|
EC_VER_MASK(0));
|