pa1.1: add handler for x, info, si
This commit is contained in:
parent
4770b4ce97
commit
69264e69c6
3 changed files with 248 additions and 47 deletions
|
@ -37,6 +37,10 @@
|
||||||
|
|
||||||
typedef MUXDEF(CONFIG_ISA64, uint64_t, uint32_t) word_t;
|
typedef MUXDEF(CONFIG_ISA64, uint64_t, uint32_t) word_t;
|
||||||
typedef MUXDEF(CONFIG_ISA64, int64_t, int32_t) sword_t;
|
typedef MUXDEF(CONFIG_ISA64, int64_t, int32_t) sword_t;
|
||||||
|
static const word_t WORD_T_MAX = MUXDEF(CONFIG_ISA64, UINT64_MAX, UINT32_MAX);
|
||||||
|
static const sword_t SWORD_T_MAX = MUXDEF(CONFIG_ISA64, INT64_MAX, INT32_MAX);
|
||||||
|
static const sword_t SWORD_T_MIN = MUXDEF(CONFIG_ISA64, INT64_MIN, INT32_MIN);
|
||||||
|
#define WORD_BYTES MUXDEF(CONFIG_ISA64, 8, 4)
|
||||||
#define FMT_WORD MUXDEF(CONFIG_ISA64, "0x%016" PRIx64, "0x%08" PRIx32)
|
#define FMT_WORD MUXDEF(CONFIG_ISA64, "0x%016" PRIx64, "0x%08" PRIx32)
|
||||||
|
|
||||||
typedef word_t vaddr_t;
|
typedef word_t vaddr_t;
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
|
|
||||||
#include <isa.h>
|
#include <isa.h>
|
||||||
#include "local-include/reg.h"
|
#include "local-include/reg.h"
|
||||||
|
#include "macro.h"
|
||||||
|
|
||||||
const char *regs[] = {
|
const char *regs[] = {
|
||||||
"$0", "ra", "sp", "gp", "tp", "t0", "t1", "t2",
|
"$0", "ra", "sp", "gp", "tp", "t0", "t1", "t2",
|
||||||
|
@ -24,6 +25,14 @@ const char *regs[] = {
|
||||||
};
|
};
|
||||||
|
|
||||||
void isa_reg_display() {
|
void isa_reg_display() {
|
||||||
|
int colomn_per_row = 4;
|
||||||
|
for(int i = 0; i < ARRLEN(regs); i++) {
|
||||||
|
printf("\e[1;34m%3s\e[0m: " FMT_PADDR, reg_name(i), gpr(i));
|
||||||
|
if (i % colomn_per_row == 3)
|
||||||
|
putchar('\n');
|
||||||
|
else
|
||||||
|
putchar('|');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
word_t isa_reg_str2val(const char *s, bool *success) {
|
word_t isa_reg_str2val(const char *s, bool *success) {
|
||||||
|
|
|
@ -1,31 +1,71 @@
|
||||||
/***************************************************************************************
|
/***************************************************************************************
|
||||||
* Copyright (c) 2014-2022 Zihao Yu, Nanjing University
|
* Copyright (c) 2014-2022 Zihao Yu, Nanjing University
|
||||||
*
|
*
|
||||||
* NEMU is licensed under Mulan PSL v2.
|
* NEMU is licensed under Mulan PSL v2.
|
||||||
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
* You can use this software according to the terms and conditions of the Mulan
|
||||||
* You may obtain a copy of Mulan PSL v2 at:
|
*PSL v2. You may obtain a copy of Mulan PSL v2 at:
|
||||||
* http://license.coscl.org.cn/MulanPSL2
|
* http://license.coscl.org.cn/MulanPSL2
|
||||||
*
|
*
|
||||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
|
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY
|
||||||
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
|
*KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
|
||||||
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
*NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
||||||
*
|
*
|
||||||
* See the Mulan PSL v2 for more details.
|
* See the Mulan PSL v2 for more details.
|
||||||
***************************************************************************************/
|
***************************************************************************************/
|
||||||
|
|
||||||
#include <isa.h>
|
|
||||||
#include <cpu/cpu.h>
|
|
||||||
#include <readline/readline.h>
|
|
||||||
#include <readline/history.h>
|
|
||||||
#include "sdb.h"
|
#include "sdb.h"
|
||||||
|
#include "common.h"
|
||||||
|
#include "sys/types.h"
|
||||||
|
#include <cpu/cpu.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <isa.h>
|
||||||
|
#include <memory/paddr.h>
|
||||||
|
#include <readline/history.h>
|
||||||
|
#include <readline/readline.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
static int is_batch_mode = false;
|
static int is_batch_mode = false;
|
||||||
|
|
||||||
|
// command handlers
|
||||||
|
static int cmd_help(char *args);
|
||||||
|
static int cmd_c(char *args);
|
||||||
|
static int cmd_q(char *args);
|
||||||
|
static int cmd_x(char *args);
|
||||||
|
static int cmd_si(char *args);
|
||||||
|
static int cmd_info(char *args);
|
||||||
|
static int cmd_info_r(char *args);
|
||||||
|
static int cmd_info_w(char *args);
|
||||||
|
|
||||||
|
static struct CMDTable {
|
||||||
|
const char *name;
|
||||||
|
const char *description;
|
||||||
|
int (*handler)(char *);
|
||||||
|
struct CMDTable *subcommand;
|
||||||
|
int nr_subcommand;
|
||||||
|
} cmd_info_table[] =
|
||||||
|
{
|
||||||
|
{"r", "List all registers and their contents", cmd_info_r, NULL, 0},
|
||||||
|
{"w", "Status of specified watchpoints", cmd_info_w, NULL, 0},
|
||||||
|
},
|
||||||
|
cmd_table[] = {
|
||||||
|
{"help", "Display information about all supported commands", cmd_help,
|
||||||
|
NULL, 0},
|
||||||
|
{"c", "Continue the execution of the program", cmd_c, NULL, 0},
|
||||||
|
{"q", "Exit NEMU", cmd_q, NULL, 0},
|
||||||
|
{"x", "Examine content of physical memory address", cmd_x, NULL, 0},
|
||||||
|
{"si", "Execute next [n] program line", cmd_si, NULL, 0},
|
||||||
|
{"info", "Print information of registers or watchpoints", cmd_info,
|
||||||
|
cmd_info_table, ARRLEN(cmd_info_table)},
|
||||||
|
};
|
||||||
|
|
||||||
|
#define NR_CMD ARRLEN(cmd_table)
|
||||||
|
|
||||||
void init_regex();
|
void init_regex();
|
||||||
void init_wp_pool();
|
void init_wp_pool();
|
||||||
|
|
||||||
/* We use the `readline' library to provide more flexibility to read from stdin. */
|
/* We use the `readline' library to provide more flexibility to read from stdin.
|
||||||
static char* rl_gets() {
|
*/
|
||||||
|
static char *rl_gets() {
|
||||||
static char *line_read = NULL;
|
static char *line_read = NULL;
|
||||||
|
|
||||||
if (line_read) {
|
if (line_read) {
|
||||||
|
@ -33,7 +73,7 @@ static char* rl_gets() {
|
||||||
line_read = NULL;
|
line_read = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
line_read = readline("(nemu) ");
|
line_read = readline("\e[1;34m(nemu)\e[0m ");
|
||||||
|
|
||||||
if (line_read && *line_read) {
|
if (line_read && *line_read) {
|
||||||
add_history(line_read);
|
add_history(line_read);
|
||||||
|
@ -42,33 +82,168 @@ static char* rl_gets() {
|
||||||
return line_read;
|
return line_read;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Extract Integer from a string. Can handle hex, binary and decimal numbers.
|
||||||
|
* Print error if meet any error.
|
||||||
|
* Return `UINTMAX_MAX` if the string is invalid or number exceed the limit of
|
||||||
|
* uint.
|
||||||
|
*/
|
||||||
|
static word_t parse_uint(const char *arg, bool *success) {
|
||||||
|
if (arg == NULL) {
|
||||||
|
puts("Invalid uint argument.");
|
||||||
|
*success = false;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
int base = 10;
|
||||||
|
int token_length = strnlen(arg, 34);
|
||||||
|
if (token_length > 2) {
|
||||||
|
if (arg[0] == '0' && (arg[1] == 'b' || arg[1] == 'B')) {
|
||||||
|
base = 2;
|
||||||
|
arg = arg + 2;
|
||||||
|
} else if (arg[0] == '0' && (arg[1] == 'x' || arg[1] == 'X')) {
|
||||||
|
base = 16;
|
||||||
|
arg = arg + 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
char *endptr;
|
||||||
|
uintmax_t n = strtoumax(arg, &endptr, base);
|
||||||
|
if (errno == ERANGE || n > WORD_T_MAX) {
|
||||||
|
printf("%s is exceed the limit of uint\n", arg);
|
||||||
|
*success = false;
|
||||||
|
return 0;
|
||||||
|
} else if (arg == endptr) {
|
||||||
|
puts("Invalid uint argument.");
|
||||||
|
*success = false;
|
||||||
|
return 0;
|
||||||
|
} else if (n > WORD_T_MAX) {
|
||||||
|
*success = false;
|
||||||
|
return WORD_T_MAX;
|
||||||
|
} else {
|
||||||
|
*success = true;
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static paddr_t parse_expr(const char *arg, bool *success) {
|
||||||
|
if (arg == NULL) {
|
||||||
|
puts("Invalid expr argument.");
|
||||||
|
*success = false;
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
bool res = false;
|
||||||
|
// FIXME: We cannot use `parse_uint` here, it accept `-1234` as input
|
||||||
|
paddr_t addr = parse_uint(arg, &res);
|
||||||
|
*success = res;
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int cmd_c(char *args) {
|
static int cmd_c(char *args) {
|
||||||
cpu_exec(-1);
|
cpu_exec(-1);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int cmd_q(char *args) {
|
static int cmd_q(char *args) {
|
||||||
nemu_state.state = NEMU_QUIT;
|
nemu_state.state = NEMU_QUIT;
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int cmd_help(char *args);
|
/* Single stepping
|
||||||
|
* <step>: execute <step> step
|
||||||
|
*/
|
||||||
|
static int cmd_si(char *args) {
|
||||||
|
char *arg = strtok(NULL, " ");
|
||||||
|
if (arg == NULL) {
|
||||||
|
cpu_exec(1);
|
||||||
|
} else {
|
||||||
|
bool res = false;
|
||||||
|
word_t n = parse_uint(arg, &res);
|
||||||
|
if (!res)
|
||||||
|
goto wrong_usage;
|
||||||
|
cpu_exec(n);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
|
||||||
static struct {
|
wrong_usage:
|
||||||
const char *name;
|
printf("Invalid argument for command si: %s\n", args);
|
||||||
const char *description;
|
printf("Usage: si [N: uint]\n");
|
||||||
int (*handler) (char *);
|
return 0;
|
||||||
} cmd_table [] = {
|
}
|
||||||
{ "help", "Display information about all supported commands", cmd_help },
|
|
||||||
{ "c", "Continue the execution of the program", cmd_c },
|
|
||||||
{ "q", "Exit NEMU", cmd_q },
|
|
||||||
|
|
||||||
/* TODO: Add more commands */
|
static int cmd_info_r(char *args) {
|
||||||
|
isa_reg_display();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
};
|
static int cmd_info_w(char *args) {
|
||||||
|
printf("Not implemented");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
#define NR_CMD ARRLEN(cmd_table)
|
static int cmd_x(char *args) {
|
||||||
|
char *arg = strtok(NULL, " ");
|
||||||
|
bool res = false;
|
||||||
|
word_t n = parse_uint(arg, &res);
|
||||||
|
if (!res)
|
||||||
|
goto wrong_usage;
|
||||||
|
arg = strtok(NULL, " ");
|
||||||
|
word_t addr = parse_expr(arg, &res);
|
||||||
|
if (!res)
|
||||||
|
goto wrong_usage;
|
||||||
|
addr = addr & ~(WORD_BYTES - 1);
|
||||||
|
for (paddr_t paddr = addr; paddr < addr + n; paddr += WORD_BYTES) {
|
||||||
|
word_t value = paddr_read(addr, WORD_BYTES);
|
||||||
|
printf("\e[1;34m" FMT_PADDR "\e[0m"
|
||||||
|
" " FMT_WORD "\n",
|
||||||
|
paddr, value);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
wrong_usage:
|
||||||
|
printf("Invalid argument for command si: %s\n", args);
|
||||||
|
printf("Usage: x [N: uint] [EXPR: <expr>]\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cmd_info(char *args) {
|
||||||
|
char *arg = strtok(NULL, " ");
|
||||||
|
int i;
|
||||||
|
if (arg == NULL) {
|
||||||
|
goto wrong_usage;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
for (i = 0; i < ARRLEN(cmd_info_table); i++) {
|
||||||
|
if (strcmp(arg, cmd_info_table[i].name) == 0) {
|
||||||
|
cmd_info_table[i].handler(args);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wrong_usage:
|
||||||
|
printf("Invalid argument for command info: %s\n", args);
|
||||||
|
printf("Usage: info [r | w]\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int cmd_help_print(char *args, struct CMDTable *cur_cmd_table,
|
||||||
|
int cur_nr_cmd) {
|
||||||
|
int i;
|
||||||
|
char *arg = strtok(NULL, " ");
|
||||||
|
if (arg == NULL) {
|
||||||
|
return -1;
|
||||||
|
} else {
|
||||||
|
for (i = 0; i < cur_nr_cmd; i++) {
|
||||||
|
if (strcmp(arg, cur_cmd_table[i].name) == 0) {
|
||||||
|
printf("%s ", cur_cmd_table[i].name);
|
||||||
|
if (cmd_help_print(arg, cur_cmd_table[i].subcommand,
|
||||||
|
cur_cmd_table[i].nr_subcommand) == -1) {
|
||||||
|
printf("-- %s\n", cur_cmd_table[i].description);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static int cmd_help(char *args) {
|
static int cmd_help(char *args) {
|
||||||
/* extract the first argument */
|
/* extract the first argument */
|
||||||
|
@ -77,14 +252,23 @@ static int cmd_help(char *args) {
|
||||||
|
|
||||||
if (arg == NULL) {
|
if (arg == NULL) {
|
||||||
/* no argument given */
|
/* no argument given */
|
||||||
for (i = 0; i < NR_CMD; i ++) {
|
for (i = 0; i < NR_CMD; i++) {
|
||||||
printf("%s - %s\n", cmd_table[i].name, cmd_table[i].description);
|
printf("%s -- %s\n", cmd_table[i].name, cmd_table[i].description);
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
for (i = 0; i < NR_CMD; i++) {
|
||||||
for (i = 0; i < NR_CMD; i ++) {
|
|
||||||
if (strcmp(arg, cmd_table[i].name) == 0) {
|
if (strcmp(arg, cmd_table[i].name) == 0) {
|
||||||
printf("%s - %s\n", cmd_table[i].name, cmd_table[i].description);
|
printf("%s ", cmd_table[i].name);
|
||||||
|
if (cmd_help_print(args, cmd_table[i].subcommand,
|
||||||
|
cmd_table[i].nr_subcommand) == -1) {
|
||||||
|
printf("-- %s\n", cmd_table[i].description);
|
||||||
|
// Print available subcommands
|
||||||
|
for (int j = 0; j < cmd_table[i].nr_subcommand; j++) {
|
||||||
|
struct CMDTable *sub_cmd_table = cmd_table[i].subcommand;
|
||||||
|
printf(" > %s -- %s\n", sub_cmd_table[j].name,
|
||||||
|
sub_cmd_table[j].description);
|
||||||
|
}
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -93,9 +277,7 @@ static int cmd_help(char *args) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void sdb_set_batch_mode() {
|
void sdb_set_batch_mode() { is_batch_mode = true; }
|
||||||
is_batch_mode = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void sdb_mainloop() {
|
void sdb_mainloop() {
|
||||||
if (is_batch_mode) {
|
if (is_batch_mode) {
|
||||||
|
@ -103,12 +285,14 @@ void sdb_mainloop() {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (char *str; (str = rl_gets()) != NULL; ) {
|
for (char *str; (str = rl_gets()) != NULL;) {
|
||||||
char *str_end = str + strlen(str);
|
char *str_end = str + strlen(str);
|
||||||
|
|
||||||
/* extract the first token as the command */
|
/* extract the first token as the command */
|
||||||
char *cmd = strtok(str, " ");
|
char *cmd = strtok(str, " ");
|
||||||
if (cmd == NULL) { continue; }
|
if (cmd == NULL) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
/* treat the remaining string as the arguments,
|
/* treat the remaining string as the arguments,
|
||||||
* which may need further parsing
|
* which may need further parsing
|
||||||
|
@ -124,14 +308,18 @@ void sdb_mainloop() {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < NR_CMD; i ++) {
|
for (i = 0; i < NR_CMD; i++) {
|
||||||
if (strcmp(cmd, cmd_table[i].name) == 0) {
|
if (strcmp(cmd, cmd_table[i].name) == 0) {
|
||||||
if (cmd_table[i].handler(args) < 0) { return; }
|
if (cmd_table[i].handler(args) < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (i == NR_CMD) { printf("Unknown command '%s'\n", cmd); }
|
if (i == NR_CMD) {
|
||||||
|
printf("Unknown command '%s'\n", cmd);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue