2023-12-20 16:20:36 +00:00
|
|
|
/***************************************************************************************
|
2024-01-11 12:11:42 +00:00
|
|
|
* Copyright (c) 2014-2022 Zihao Yu, Nanjing University
|
|
|
|
*
|
|
|
|
* NEMU is licensed under Mulan PSL v2.
|
|
|
|
* You can use this software according to the terms and conditions of the Mulan
|
|
|
|
*PSL v2. You may obtain a copy of Mulan PSL v2 at:
|
|
|
|
* http://license.coscl.org.cn/MulanPSL2
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY
|
|
|
|
*KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
|
|
|
|
*NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
|
|
|
*
|
|
|
|
* See the Mulan PSL v2 for more details.
|
|
|
|
***************************************************************************************/
|
2023-12-20 16:20:36 +00:00
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
#include "sdb.h"
|
2023-12-20 16:20:36 +00:00
|
|
|
#include <cpu/cpu.h>
|
2024-01-11 12:11:42 +00:00
|
|
|
#include <isa.h>
|
2023-12-20 16:20:36 +00:00
|
|
|
#include <readline/history.h>
|
2024-01-11 12:11:42 +00:00
|
|
|
#include <readline/readline.h>
|
2023-12-20 16:20:36 +00:00
|
|
|
|
|
|
|
static int is_batch_mode = false;
|
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
// command handlers
|
|
|
|
static int cmd_help(char *args);
|
|
|
|
static int cmd_c(char *args);
|
|
|
|
static int cmd_q(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},
|
|
|
|
{"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)
|
|
|
|
|
2023-12-20 16:20:36 +00:00
|
|
|
void init_regex();
|
|
|
|
void init_wp_pool();
|
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
/* We use the `readline' library to provide more flexibility to read from stdin.
|
|
|
|
*/
|
|
|
|
static char *rl_gets() {
|
2023-12-20 16:20:36 +00:00
|
|
|
static char *line_read = NULL;
|
|
|
|
|
|
|
|
if (line_read) {
|
|
|
|
free(line_read);
|
|
|
|
line_read = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
line_read = readline("(nemu) ");
|
|
|
|
|
|
|
|
if (line_read && *line_read) {
|
|
|
|
add_history(line_read);
|
|
|
|
}
|
|
|
|
|
|
|
|
return line_read;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int cmd_c(char *args) {
|
|
|
|
cpu_exec(-1);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int cmd_q(char *args) {
|
2024-01-11 09:35:39 +00:00
|
|
|
nemu_state.state = NEMU_QUIT;
|
2024-01-11 09:32:50 +00:00
|
|
|
return -1;
|
2023-12-20 16:20:36 +00:00
|
|
|
}
|
|
|
|
|
2024-01-11 10:12:27 +00:00
|
|
|
/* Single stepping
|
|
|
|
* <step>: execute <step> step
|
|
|
|
*/
|
2024-01-11 09:50:33 +00:00
|
|
|
static int cmd_si(char *args) {
|
2024-01-11 12:18:53 +00:00
|
|
|
char *arg = strtok(NULL, " ");
|
2024-01-11 10:12:27 +00:00
|
|
|
if (arg == NULL) {
|
|
|
|
cpu_exec(1);
|
|
|
|
} else {
|
|
|
|
int base = 10;
|
2024-01-11 12:11:42 +00:00
|
|
|
int length = strnlen(arg, 34);
|
2024-01-11 10:12:27 +00:00
|
|
|
if (length > 2) {
|
|
|
|
if (arg[0] == '0' && (arg[1] == 'b' || arg[1] == 'B')) {
|
2024-01-11 12:11:42 +00:00
|
|
|
base = 2;
|
|
|
|
arg = arg + 2;
|
2024-01-11 10:12:27 +00:00
|
|
|
} else if (arg[0] == '0' && (arg[1] == 'x' || arg[1] == 'X')) {
|
2024-01-11 12:11:42 +00:00
|
|
|
base = 16;
|
|
|
|
arg = arg + 2;
|
2024-01-11 10:12:27 +00:00
|
|
|
}
|
|
|
|
}
|
2024-01-11 12:11:42 +00:00
|
|
|
if (arg[0] == '0')
|
2024-01-11 10:16:08 +00:00
|
|
|
return 0;
|
2024-01-11 10:12:27 +00:00
|
|
|
int n = strtoumax(arg, NULL, base);
|
|
|
|
if (n == UINTMAX_MAX) {
|
2024-01-11 10:17:20 +00:00
|
|
|
printf("%s is exceed the limit of uint\n", args);
|
2024-01-11 10:16:08 +00:00
|
|
|
return 0;
|
|
|
|
} else if (n == 0) {
|
2024-01-11 10:12:27 +00:00
|
|
|
printf("Invalid argument for command si: %s\n", args);
|
2024-01-11 12:11:42 +00:00
|
|
|
cmd_help(NULL);
|
2024-01-11 10:12:27 +00:00
|
|
|
return 0;
|
2024-01-11 12:11:42 +00:00
|
|
|
} else {
|
2024-01-11 10:12:27 +00:00
|
|
|
cpu_exec(n);
|
|
|
|
}
|
|
|
|
}
|
2024-01-11 09:50:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
static int cmd_info_r(char *args) { return 0; }
|
2023-12-20 16:20:36 +00:00
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
static int cmd_info_w(char *args) { return 0; }
|
2023-12-20 16:20:36 +00:00
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
static int cmd_info(char *args) {
|
|
|
|
char *arg = strtok(args, " ");
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < NR_CMD; i++) {
|
|
|
|
if (strcmp(arg, cmd_info_table[i].name) == 0) {
|
|
|
|
cmd_info_table[i].handler(args);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:25:29 +00:00
|
|
|
static int cmd_help_print(char *args, struct CMDTable* cur_cmd_table, int cur_nr_cmd) {
|
2024-01-11 12:11:42 +00:00
|
|
|
int i;
|
2024-01-11 12:22:35 +00:00
|
|
|
char *arg = strtok(NULL, " ");
|
2024-01-11 12:11:42 +00:00
|
|
|
if (arg == NULL) {
|
|
|
|
return -1;
|
|
|
|
} else {
|
2024-01-11 12:25:29 +00:00
|
|
|
for (i = 0; i < cur_nr_cmd; i++) {
|
2024-01-11 12:11:42 +00:00
|
|
|
if (strcmp(arg, cur_cmd_table[i].name) == 0) {
|
|
|
|
printf("%s ", cur_cmd_table[i].name);
|
2024-01-11 12:25:29 +00:00
|
|
|
if (cmd_help_print(arg, cur_cmd_table[i].subcommand, cur_cmd_table[i].nr_subcommand) == -1) {
|
2024-01-11 12:12:07 +00:00
|
|
|
printf("-- %s\n", cur_cmd_table[i].description);
|
2024-01-11 12:11:42 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2023-12-20 16:20:36 +00:00
|
|
|
|
|
|
|
static int cmd_help(char *args) {
|
|
|
|
/* extract the first argument */
|
|
|
|
char *arg = strtok(NULL, " ");
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (arg == NULL) {
|
|
|
|
/* no argument given */
|
2024-01-11 12:11:42 +00:00
|
|
|
for (i = 0; i < NR_CMD; i++) {
|
2024-01-11 12:18:53 +00:00
|
|
|
printf("%s -- %s\n", cmd_table[i].name, cmd_table[i].description);
|
2023-12-20 16:20:36 +00:00
|
|
|
}
|
2024-01-11 12:11:42 +00:00
|
|
|
} else {
|
|
|
|
for (i = 0; i < NR_CMD; i++) {
|
2023-12-20 16:20:36 +00:00
|
|
|
if (strcmp(arg, cmd_table[i].name) == 0) {
|
2024-01-11 12:11:42 +00:00
|
|
|
printf("%s ", cmd_table[i].name);
|
2024-01-11 12:25:29 +00:00
|
|
|
if (cmd_help_print(args, cmd_table[i].subcommand, cmd_table[i].nr_subcommand) == -1) {
|
2024-01-11 12:12:07 +00:00
|
|
|
printf("-- %s\n", cmd_table[i].description);
|
2024-01-11 12:28:18 +00:00
|
|
|
// Print available subcommands
|
|
|
|
for (int j = 0; j < cmd_table[i].nr_subcommand; j++) {
|
2024-01-11 12:30:13 +00:00
|
|
|
struct CMDTable *sub_cmd_table = cmd_table[i].subcommand;
|
2024-01-11 12:30:55 +00:00
|
|
|
printf(" > %s -- %s\n", sub_cmd_table[j].name, sub_cmd_table[j].description);
|
2024-01-11 12:28:18 +00:00
|
|
|
}
|
2024-01-11 12:11:42 +00:00
|
|
|
}
|
2023-12-20 16:20:36 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
printf("Unknown command '%s'\n", arg);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
void sdb_set_batch_mode() { is_batch_mode = true; }
|
2023-12-20 16:20:36 +00:00
|
|
|
|
|
|
|
void sdb_mainloop() {
|
|
|
|
if (is_batch_mode) {
|
|
|
|
cmd_c(NULL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
for (char *str; (str = rl_gets()) != NULL;) {
|
2023-12-20 16:20:36 +00:00
|
|
|
char *str_end = str + strlen(str);
|
|
|
|
|
|
|
|
/* extract the first token as the command */
|
|
|
|
char *cmd = strtok(str, " ");
|
2024-01-11 12:11:42 +00:00
|
|
|
if (cmd == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
2023-12-20 16:20:36 +00:00
|
|
|
|
|
|
|
/* treat the remaining string as the arguments,
|
|
|
|
* which may need further parsing
|
|
|
|
*/
|
|
|
|
char *args = cmd + strlen(cmd) + 1;
|
|
|
|
if (args >= str_end) {
|
|
|
|
args = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_DEVICE
|
|
|
|
extern void sdl_clear_event_queue();
|
|
|
|
sdl_clear_event_queue();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
int i;
|
2024-01-11 12:11:42 +00:00
|
|
|
for (i = 0; i < NR_CMD; i++) {
|
2023-12-20 16:20:36 +00:00
|
|
|
if (strcmp(cmd, cmd_table[i].name) == 0) {
|
2024-01-11 12:11:42 +00:00
|
|
|
if (cmd_table[i].handler(args) < 0) {
|
|
|
|
return;
|
|
|
|
}
|
2023-12-20 16:20:36 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-11 12:11:42 +00:00
|
|
|
if (i == NR_CMD) {
|
|
|
|
printf("Unknown command '%s'\n", cmd);
|
|
|
|
}
|
2023-12-20 16:20:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void init_sdb() {
|
|
|
|
/* Compile the regular expressions. */
|
|
|
|
init_regex();
|
|
|
|
|
|
|
|
/* Initialize the watchpoint pool. */
|
|
|
|
init_wp_pool();
|
|
|
|
}
|