b2704e693c
git-svn-id: file:///srv/svn/repos/haiku/trunk/current@3162 a95241bf-73f2-0310-859d-f6bbb57e9c96
494 lines
9.0 KiB
C
494 lines
9.0 KiB
C
/* This file contains the debugger */
|
|
|
|
/*
|
|
** Copyright 2001, Travis Geiselbrecht. All rights reserved.
|
|
** Distributed under the terms of the NewOS License.
|
|
*/
|
|
|
|
#include <kernel.h>
|
|
#include <debug.h>
|
|
#include <arch/int.h>
|
|
#include <smp.h>
|
|
#include <console.h>
|
|
#include <memheap.h>
|
|
#include <gdb.h>
|
|
#include <Errors.h>
|
|
#include <int.h>
|
|
|
|
#include <arch/dbg_console.h>
|
|
#include <arch/debug.h>
|
|
#include <arch/cpu.h>
|
|
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <malloc.h>
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
|
|
|
|
int dbg_register_file[2][14]; /* XXXmpetit -- must be made generic */
|
|
|
|
|
|
static bool serial_debug_on = false;
|
|
static spinlock dbg_spinlock = 0;
|
|
static int debugger_on_cpu = -1;
|
|
|
|
struct debugger_command
|
|
{
|
|
struct debugger_command *next;
|
|
int (*func)(int, char **);
|
|
const char *name;
|
|
const char *description;
|
|
};
|
|
|
|
static struct debugger_command *commands;
|
|
|
|
#define LINE_BUF_SIZE 1024
|
|
#define MAX_ARGS 16
|
|
#define HISTORY_SIZE 16
|
|
|
|
static char line_buf[HISTORY_SIZE][LINE_BUF_SIZE] = { "", };
|
|
static char parse_line[LINE_BUF_SIZE] = "";
|
|
static int cur_line = 0;
|
|
static char *args[MAX_ARGS] = { NULL, };
|
|
|
|
#define distance(a, b) ((a) < (b) ? (b) - (a) : (a) - (b))
|
|
|
|
|
|
static int
|
|
debug_read_line(char *buf, int max_len)
|
|
{
|
|
char c;
|
|
int ptr = 0;
|
|
bool done = false;
|
|
int cur_history_spot = cur_line;
|
|
|
|
while (!done) {
|
|
c = arch_dbg_con_read();
|
|
switch (c) {
|
|
case '\n':
|
|
case '\r':
|
|
buf[ptr++] = '\0';
|
|
dbg_puts("\n");
|
|
done = true;
|
|
break;
|
|
case 8: // backspace
|
|
if (ptr > 0) {
|
|
dbg_puts("\x1b[1D"); // move to the left one
|
|
dbg_putch(' ');
|
|
dbg_puts("\x1b[1D"); // move to the left one
|
|
ptr--;
|
|
}
|
|
break;
|
|
case 27: // escape sequence
|
|
c = arch_dbg_con_read(); // should be '['
|
|
c = arch_dbg_con_read();
|
|
switch (c) {
|
|
case 67: // right arrow acts like space
|
|
buf[ptr++] = ' ';
|
|
dbg_putch(' ');
|
|
break;
|
|
case 68: // left arrow acts like backspace
|
|
if (ptr > 0) {
|
|
dbg_puts("\x1b[1D"); // move to the left one
|
|
dbg_putch(' ');
|
|
dbg_puts("\x1b[1D"); // move to the left one
|
|
ptr--;
|
|
}
|
|
break;
|
|
case 65: // up arrow
|
|
case 66: // down arrow
|
|
{
|
|
int history_line = 0;
|
|
|
|
// dprintf("1c %d h %d ch %d\n", cur_line, history_line, cur_history_spot);
|
|
|
|
if (c == 65) {
|
|
// up arrow
|
|
history_line = cur_history_spot - 1;
|
|
if (history_line < 0)
|
|
history_line = HISTORY_SIZE - 1;
|
|
} else {
|
|
// down arrow
|
|
if (cur_history_spot != cur_line) {
|
|
history_line = cur_history_spot + 1;
|
|
if(history_line >= HISTORY_SIZE)
|
|
history_line = 0;
|
|
} else
|
|
break; // nothing to do here
|
|
}
|
|
|
|
// dprintf("2c %d h %d ch %d\n", cur_line, history_line, cur_history_spot);
|
|
|
|
// swap the current line with something from the history
|
|
if(ptr > 0)
|
|
dprintf("\x1b[%dD", ptr); // move to beginning of line
|
|
|
|
strcpy(buf, line_buf[history_line]);
|
|
ptr = strlen(buf);
|
|
dprintf("%s\x1b[K", buf); // print the line and clear the rest
|
|
cur_history_spot = history_line;
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
case '$':
|
|
case '+':
|
|
/* HACK ALERT!!!
|
|
*
|
|
* If we get a $ at the beginning of the line
|
|
* we assume we are talking with GDB
|
|
*/
|
|
if (ptr == 0) {
|
|
strcpy(buf, "gdb");
|
|
ptr= 4;
|
|
done= true;
|
|
break;
|
|
} else {
|
|
/* fall thru */
|
|
}
|
|
default:
|
|
buf[ptr++] = c;
|
|
dbg_putch(c);
|
|
}
|
|
if (ptr >= max_len - 2) {
|
|
buf[ptr++] = '\0';
|
|
dbg_puts("\n");
|
|
done = true;
|
|
break;
|
|
}
|
|
}
|
|
return ptr;
|
|
}
|
|
|
|
|
|
static int
|
|
debug_parse_line(char *buf, char **argv, int *argc, int max_args)
|
|
{
|
|
int pos = 0;
|
|
|
|
strcpy(parse_line, buf);
|
|
|
|
if (!isspace(parse_line[0])) {
|
|
argv[0] = parse_line;
|
|
*argc = 1;
|
|
} else
|
|
*argc = 0;
|
|
|
|
while (parse_line[pos] != '\0') {
|
|
if (isspace(parse_line[pos])) {
|
|
parse_line[pos] = '\0';
|
|
// scan all of the whitespace out of this
|
|
while (isspace(parse_line[++pos]))
|
|
;
|
|
if (parse_line[pos] == '\0')
|
|
break;
|
|
argv[*argc] = &parse_line[pos];
|
|
(*argc)++;
|
|
|
|
if (*argc >= max_args - 1)
|
|
break;
|
|
}
|
|
pos++;
|
|
}
|
|
|
|
return *argc;
|
|
}
|
|
|
|
|
|
static void
|
|
kernel_debugger_loop()
|
|
{
|
|
int argc;
|
|
struct debugger_command *cmd;
|
|
cpu_status state;
|
|
|
|
state = disable_interrupts();
|
|
|
|
dprintf("kernel debugger on cpu %d\n", smp_get_current_cpu());
|
|
debugger_on_cpu = smp_get_current_cpu();
|
|
|
|
cmd = NULL;
|
|
|
|
for(;;) {
|
|
dprintf("kdebug> ");
|
|
debug_read_line(line_buf[cur_line], LINE_BUF_SIZE);
|
|
debug_parse_line(line_buf[cur_line], args, &argc, MAX_ARGS);
|
|
|
|
// We support calling last executed command again if
|
|
// B_KDEDUG_CONT was returned last time, so cmd != NULL
|
|
if (argc <= 0 && cmd == NULL)
|
|
continue;
|
|
|
|
debugger_on_cpu = smp_get_current_cpu();
|
|
|
|
if (argc > 0) {
|
|
// search command by name
|
|
cmd = commands;
|
|
while (cmd != NULL) {
|
|
if (strcmp(args[0], cmd->name) == 0)
|
|
break;
|
|
cmd = cmd->next;
|
|
};
|
|
};
|
|
|
|
if (cmd == NULL)
|
|
dprintf("unknown command, enter \"help\" to get a list of all supported commands\n");
|
|
else {
|
|
int rc;
|
|
|
|
rc = cmd->func(argc, args);
|
|
|
|
if (rc == B_KDEBUG_QUIT)
|
|
break; // okay, exit now.
|
|
|
|
if (rc != B_KDEBUG_CONT)
|
|
cmd = NULL; // forget last command executed...
|
|
};
|
|
|
|
cur_line++;
|
|
if (cur_line >= HISTORY_SIZE)
|
|
cur_line = 0;
|
|
}
|
|
|
|
restore_interrupts(state);
|
|
}
|
|
|
|
|
|
void
|
|
kernel_debugger(const char * message)
|
|
{
|
|
arch_dbg_save_registers(&(dbg_register_file[smp_get_current_cpu()][0]));
|
|
|
|
if (message) {
|
|
dprintf(message);
|
|
dprintf("\n");
|
|
};
|
|
|
|
dprintf("Welcome to Kernel Debugging Land...\n");
|
|
kernel_debugger_loop();
|
|
}
|
|
|
|
|
|
void
|
|
panic(const char *fmt, ...)
|
|
{
|
|
int ret = 0;
|
|
va_list args;
|
|
char temp[128];
|
|
cpu_status state;
|
|
|
|
// XXX by setting kernel_startup = true, we disable
|
|
// XXX the interrupt check in semaphore code etc.
|
|
// XXX should be renamed?
|
|
kernel_startup = true;
|
|
|
|
dbg_set_serial_debug(true);
|
|
|
|
state = disable_interrupts();
|
|
|
|
va_start(args, fmt);
|
|
ret = vsprintf(temp, fmt, args);
|
|
va_end(args);
|
|
|
|
dprintf("PANIC%d: %s", smp_get_current_cpu(), temp);
|
|
|
|
if (debugger_on_cpu != smp_get_current_cpu()) {
|
|
// halt all of the other cpus
|
|
|
|
// XXX need to flush current smp mailbox to make sure this goes
|
|
// through. Otherwise it'll hang
|
|
smp_send_broadcast_ici(SMP_MSG_CPU_HALT, 0, 0, 0, NULL, SMP_MSG_FLAG_SYNC);
|
|
}
|
|
|
|
kernel_debugger(NULL);
|
|
|
|
restore_interrupts(state);
|
|
}
|
|
|
|
|
|
void
|
|
dprintf(const char *fmt, ...)
|
|
{
|
|
va_list args;
|
|
char temp[512];
|
|
|
|
if (serial_debug_on) {
|
|
va_start(args, fmt);
|
|
vsprintf(temp, fmt, args);
|
|
va_end(args);
|
|
|
|
dbg_puts(temp);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
char
|
|
dbg_putch(char c)
|
|
{
|
|
char ret;
|
|
cpu_status state = disable_interrupts();
|
|
acquire_spinlock(&dbg_spinlock);
|
|
|
|
if (serial_debug_on)
|
|
ret = arch_dbg_con_putch(c);
|
|
else
|
|
ret = c;
|
|
|
|
release_spinlock(&dbg_spinlock);
|
|
restore_interrupts(state);
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
void
|
|
dbg_puts(const char *s)
|
|
{
|
|
cpu_status state = disable_interrupts();
|
|
acquire_spinlock(&dbg_spinlock);
|
|
|
|
if (serial_debug_on)
|
|
arch_dbg_con_puts(s);
|
|
|
|
release_spinlock(&dbg_spinlock);
|
|
restore_interrupts(state);
|
|
}
|
|
|
|
|
|
int
|
|
add_debugger_command(char *name, int (*func)(int, char **), char *desc)
|
|
{
|
|
cpu_status state;
|
|
struct debugger_command *cmd;
|
|
|
|
cmd = (struct debugger_command *)malloc(sizeof(struct debugger_command));
|
|
if (cmd == NULL)
|
|
return ENOMEM;
|
|
|
|
cmd->func = func;
|
|
cmd->name = name;
|
|
cmd->description = desc;
|
|
|
|
state = disable_interrupts();
|
|
acquire_spinlock(&dbg_spinlock);
|
|
|
|
cmd->next = commands;
|
|
commands = cmd;
|
|
|
|
release_spinlock(&dbg_spinlock);
|
|
restore_interrupts(state);
|
|
|
|
return B_NO_ERROR;
|
|
}
|
|
|
|
|
|
int
|
|
remove_debugger_command(char * name, int (*func)(int, char **))
|
|
{
|
|
struct debugger_command *cmd = commands;
|
|
struct debugger_command *prev = NULL;
|
|
cpu_status state;
|
|
|
|
state = disable_interrupts();
|
|
acquire_spinlock(&dbg_spinlock);
|
|
|
|
while (cmd) {
|
|
if (!strcmp(cmd->name, name) && cmd->func == func)
|
|
break;
|
|
|
|
prev = cmd;
|
|
cmd = cmd->next;
|
|
}
|
|
|
|
if (cmd) {
|
|
if (cmd == commands)
|
|
commands = cmd->next;
|
|
else
|
|
prev->next = cmd->next;
|
|
}
|
|
|
|
release_spinlock(&dbg_spinlock);
|
|
restore_interrupts(state);
|
|
|
|
if (cmd) {
|
|
free(cmd);
|
|
return B_NO_ERROR;
|
|
}
|
|
|
|
return B_NAME_NOT_FOUND;
|
|
}
|
|
|
|
|
|
static int
|
|
cmd_reboot(int argc, char **argv)
|
|
{
|
|
reboot();
|
|
return 0; // I'll be really suprised if this line ever run! ;-)
|
|
}
|
|
|
|
|
|
static int
|
|
cmd_help(int argc, char **argv)
|
|
{
|
|
struct debugger_command *cmd;
|
|
|
|
dprintf("debugger commands:\n");
|
|
cmd = commands;
|
|
while (cmd != NULL) {
|
|
dprintf(" %-32s\t\t%s\n", cmd->name, cmd->description);
|
|
cmd = cmd->next;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int
|
|
dbg_init(kernel_args *ka)
|
|
{
|
|
commands = NULL;
|
|
|
|
return arch_dbg_con_init(ka);
|
|
}
|
|
|
|
|
|
int
|
|
dbg_init2(kernel_args *ka)
|
|
{
|
|
add_debugger_command("help", &cmd_help, "List all debugger commands");
|
|
add_debugger_command("reboot", &cmd_reboot, "Reboot");
|
|
add_debugger_command("gdb", &cmd_gdb, "Connect to remote gdb");
|
|
|
|
return arch_dbg_init(ka);
|
|
}
|
|
|
|
|
|
bool
|
|
dbg_set_serial_debug(bool new_val)
|
|
{
|
|
int temp = serial_debug_on;
|
|
serial_debug_on = new_val;
|
|
return temp;
|
|
}
|
|
|
|
|
|
bool
|
|
dbg_get_serial_debug()
|
|
{
|
|
return serial_debug_on;
|
|
}
|
|
|
|
// Wrapper(s) for BeOS R5 compatibility:
|
|
|
|
bool
|
|
set_dprintf_enabled(bool new_state)
|
|
{
|
|
return dbg_set_serial_debug(new_state);
|
|
}
|
|
|
|
|