TeleStern/ao-tools/ao-dbg/ao-dbg-parse.c

210 lines
5.4 KiB
C

/*
* Copyright © 2008 Keith Packard <keithp@keithp.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
*/
#include "ao-dbg.h"
static struct command_function functions[] = {
{ "help", "?", command_help, "help", "Print this list\n" },
{ "quit", "q", command_quit, "[q]uit", "Quit\n" },
{ "di", "di", command_di, "di <start> <end>",
"Dump imem\n" },
{ "ds", "ds", command_ds, "ds <start> <end>",
"Dump sprs\n" },
{ "dx", "dx", command_dx, "dx <start> <end>",
"Dump xaddr\n" },
{ "set", "t", command_set, "se[t] mem <prefix> <address> <data> ...",
"Set mem {xram|rom|iram|sfr}\n"
"set bit <addr>\n" },
{ "dump", "d", command_dump, "[d]ump <prefix> <start> <end>",
"Dump {xram|rom|iram|sfr} <start> <end>\n" },
{ "file", "file", command_file, "file <filename>",
"Pretend to load executable from <filename>\n" },
{ "pc", "p", command_pc, "[p]c [addr]",
"Get or set pc value\n" },
{ "break", "b", command_break,"[b]reak <addr>",
"Set break point\n" },
{ "clear", "c", command_clear,"[c]lear <addr>",
"Clear break point\n" },
{ "run", "r", command_run, "[r]un [start] [stop]",
"Run with optional start and temp breakpoint addresses\n" },
{ "go", "g", command_run, "[g]o [start] [stop]",
"Run with optional start and temp breakpoint addresses\n" },
{ "next", "n", command_next, "[n]ext",
"Step over one instruction, past any call\n" },
{ "step", "s", command_step, "[s]tep",
"Single step\n" },
{ "load", "l", command_load, "[l]oad <file>",
"Load a hex file into memory or flash" },
{ "halt", "h", command_halt, "[h]alt",
"Halt the processor\n" },
{ "reset","res",command_reset, "[res]et",
"Reset the CPU\n" },
{ "status","status",command_status, "status",
"Display CC1111 debug status\n" },
{ "info", "i", command_info, "[i]info",
"Get information\n" },
{ "stop", "stop", command_stop, "stop",
"Ignored\n" },
{ NULL, NULL, NULL, NULL, NULL },
};
#ifndef FALSE
#define FALSE 0
#define TRUE 1
#endif
struct command_function *
command_string_to_function(struct command_function *functions, char *name)
{
int i;
for (i = 0; functions[i].name; i++)
if (!strcmp(name, functions[i].name) ||
!strcmp(name, functions[i].alias))
return &functions[i];
return NULL;
}
enum command_result
command_function_help(struct command_function *functions, int argc, char **argv)
{
int i;
struct command_function *func;
if (argc == 1) {
for (i = 0; functions[i].name; i++)
s51_printf("%-10s%s\n", functions[i].name,
functions[i].usage);
} else {
for (i = 1; i < argc; i++) {
func = command_string_to_function(functions, argv[i]);
if (!func) {
s51_printf("%-10s unknown command\n", argv[i]);
return command_syntax;
}
s51_printf("%-10s %s\n%s", func->name,
func->usage, func->help);
}
}
return command_debug;
}
static int
command_split_into_words(char *line, char **argv)
{
char quotechar;
int argc;
argc = 0;
while (*line) {
while (isspace(*line))
line++;
if (!*line)
break;
if (*line == '"') {
quotechar = *line++;
*argv++ = line;
argc++;
while (*line && *line != quotechar)
line++;
if (*line)
*line++ = '\0';
} else {
*argv++ = line;
argc++;
while (*line && !isspace(*line))
line++;
if (*line)
*line++ = '\0';
}
}
*argv = 0;
return argc;
}
enum command_result
command_help(int argc, char **argv)
{
return command_function_help(functions, argc, argv);
}
void
command_syntax_error(int argc, char **argv)
{
s51_printf("Syntax error in:");
while (*argv)
s51_printf(" %s", *argv++);
s51_printf("\n");
}
void
command_read (void)
{
int argc;
char line[1024];
char *argv[20];
enum command_result result;
struct command_function *func;
if (!s51_tty) {
if (!s51_device)
s51_device = getenv("AO_DBG_DEVICE");
s51_tty = cc_usbdevs_find_by_arg(s51_device, "TeleDongle");
}
s51_dbg = ccdbg_open (s51_tty);
if (!s51_dbg)
exit(1);
ccdbg_debug_mode(s51_dbg);
ccdbg_halt(s51_dbg);
s51_printf("Welcome to the non-simulated processor\n");
for (;;) {
if (s51_read_line (line, sizeof line) == 0)
break;
s51_interrupted = 0;
argc = command_split_into_words(line, argv);
if (argc > 0) {
func = command_string_to_function(functions, argv[0]);
if (!func)
command_syntax_error(argc, argv);
else
{
result = (*func->func)(argc, argv);
if (s51_interrupted)
result = command_interrupt;
switch (result) {
case command_syntax:
command_syntax_error(argc, argv);
break;
case command_error:
s51_printf("Error\n");
break;
case command_success:
break;
case command_interrupt:
ccdbg_halt(s51_dbg);
s51_printf("Interrupted\n");
break;
default:
break;
}
}
}
}
ccdbg_close(s51_dbg);
s51_printf("...\n");
}