diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/cpu.c | 86 | ||||
-rw-r--r-- | src/gui.c | 112 | ||||
-rw-r--r-- | src/log.c | 6 | ||||
-rw-r--r-- | src/main.c | 25 | ||||
-rw-r--r-- | src/mem.c | 2 |
5 files changed, 206 insertions, 25 deletions
@@ -1,3 +1,5 @@ +#include <stdlib.h> + #include <cpu.h> #include <load.h> #include <log.h> @@ -89,10 +91,18 @@ #define GET_RM8() (MOD() == 3 ? GET_R8(RM()) : *(uint8_t *)mem_phys(ADDR())) #define GET_RM32() (MOD() == 3 ? GET_R32(RM()) : *(uint32_t *)mem_phys(ADDR())) +static struct cpu_interface *interface; + +struct rip_history { + uint64_t rip; + struct rip_history *prev; +}; + static uint64_t rip = 0; -static int (*pos_instructions[256])(void); -static int (*neg_instructions[256])(void); +static err (*pos_instructions[256])(void); +static err (*neg_instructions[256])(void); static uint64_t regs[REGISTERS_COUNT] = { 0 }; +static struct rip_history *rip_history; static void print_state(void) { @@ -102,19 +112,19 @@ static void print_state(void) regs[R10], regs[R11], regs[R12], regs[R13], regs[R14], regs[R15]); } -static int pos_unknown_instruction(void) +static err pos_unknown_instruction(void) { errln("unknown positive instruction at rip=%d", rip); return ERR; } -static int neg_unknown_instruction(void) +static err neg_unknown_instruction(void) { errln("unknown negative instruction at rip=%d", rip); return ERR; } -static int pos_opcode(void) +static err pos_opcode(void) { uint8_t op = U8(); // TODO: is_*, j* @@ -124,14 +134,14 @@ static int pos_opcode(void) return ERR; } -static int pos_mov_r32_rm32(void) +static err pos_mov_r32_rm32(void) { uint8_t modrm = U8(); SET_R32(REG(), GET_RM32()); return OK; } -static int pos_mov_r32_imm32(void) +static err pos_mov_r32_imm32(void) { rip--; uint8_t reg = U8() - 0xb8; @@ -139,12 +149,12 @@ static int pos_mov_r32_imm32(void) return OK; } -static int pos_nop(void) +static err pos_nop(void) { return OK; } -static int pos_int(void) +static err pos_int(void) { uint8_t op = U8(); if (op == 0x80) { @@ -182,6 +192,11 @@ void cpu_set_reg(uint8_t reg, uint64_t val) regs[reg] = val; } +void cpu_register_interface(struct cpu_interface *cpu) +{ + interface = cpu; +} + void cpu_exec(const char *path) { initialize(); @@ -193,14 +208,49 @@ void cpu_exec(const char *path) } rip = addr; - int ret = OK; - while (ret) { - uint8_t instr = U8(); - if (!instr) - return; - logln("%x", instr); - ret = pos_instructions[instr](); - print_state(); + rip_history = malloc(sizeof(*rip_history)); + rip_history->rip = rip; + rip_history->prev = 0; +} + +err cpu_next(void) +{ + uint8_t instr = U8(); + if (!instr) + return END; + logln("%x", instr); + err ret = pos_instructions[instr](); + print_state(); + + struct rip_history *next = malloc(sizeof(*next)); + next->rip = rip; + next->prev = rip_history; + rip_history = next; + return ret; +} + +err cpu_prev(void) +{ + struct rip_history *prev = rip_history->prev; + if (!prev) + return END; + rip = prev->rip; + uint8_t instr = U8(); + if (!instr) + return ERR; + err ret = neg_instructions[instr](); + print_state(); + + free(rip_history); + rip_history = prev; + return ret; +} + +void cpu_destroy(void) +{ + while (rip_history) { + struct rip_history *temp = rip_history->prev; + free(rip_history); + rip_history = temp; } - errln("error while executing"); } diff --git a/src/gui.c b/src/gui.c new file mode 100644 index 0000000..297ff6f --- /dev/null +++ b/src/gui.c @@ -0,0 +1,112 @@ +#include <gtk/gtk.h> + +#include <gui.h> +#include <log.h> + +enum stepper_response_type { + STEP_PREV, + STEP_NEXT, +}; + +static struct gui_interface *interface; + +static void error_dialog(GtkWindow *parent, const char *msg) +{ + GtkWidget *dialog = gtk_dialog_new_with_buttons( + msg, parent, GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, + "OK", GTK_RESPONSE_CLOSE, NULL); + GtkWidget *content = gtk_dialog_get_content_area(GTK_DIALOG(dialog)); + GtkWidget *label = gtk_label_new(msg); + g_signal_connect_swapped(dialog, "response", + G_CALLBACK(gtk_window_destroy), dialog); + + gtk_box_append(GTK_BOX(content), label); + gtk_widget_show(dialog); +} + +static void stepper_response(GtkWidget *widget, enum stepper_response_type type) +{ + err ret = OK; + + switch (type) { + case STEP_PREV: + ret = interface->step_prev(); + break; + case STEP_NEXT: + ret = interface->step_next(); + break; + default: + break; + } + + if (ret == END || ret == ERR) { + error_dialog(GTK_WINDOW(widget), + ret == END ? "Reached end" : "An error occured"); + } +} + +static void init_stepper_dialog(GtkWindow *parent) +{ + GtkWidget *stepper = gtk_dialog_new_with_buttons( + "Stepper", parent, + GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, "Previous", + 1, "Next", 2, NULL); + + GtkWidget *content = gtk_dialog_get_content_area(GTK_DIALOG(stepper)); + GtkWidget *label = gtk_label_new("Stepper"); + + g_signal_connect_swapped(stepper, "response", + G_CALLBACK(stepper_response), stepper); + + gtk_box_append(GTK_BOX(content), label); + gtk_widget_show(stepper); +} + +static void init_execution_window(GtkApplication *app) +{ + GtkWidget *execution = gtk_application_window_new(app); + gtk_window_set_title(GTK_WINDOW(execution), "Execution"); + gtk_window_set_default_size(GTK_WINDOW(execution), 200, 200); + gtk_widget_show(execution); + + init_stepper_dialog(GTK_WINDOW(execution)); +} + +static void activate(GtkApplication *app, gpointer user_data) +{ + (void)user_data; + init_execution_window(app); +} + +void gui_reg_names(const char *names, int n) +{ + (void)names; + (void)n; +} + +void gui_reg_update(int reg, uint64_t value) +{ + (void)reg; + (void)value; +} + +void gui_instr_done(char *instr) +{ + (void)instr; +} + +void gui_register_interface(struct gui_interface *gui) +{ + interface = gui; +} + +void gui_init(void) +{ + g_object_set(gtk_settings_get_default(), + "gtk-application-prefer-dark-theme", TRUE, NULL); + GtkApplication *app = gtk_application_new("de.melvars.simsalasim", + G_APPLICATION_DEFAULT_FLAGS); + g_signal_connect(app, "activate", G_CALLBACK(activate), NULL); + g_application_run(G_APPLICATION(app), 0, 0); + g_object_unref(app); +} @@ -9,7 +9,8 @@ static pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER; -void __logln(const char *func, const char *format, ...) +__attribute__((__format__(__printf__, 2, 0))) void +__logln(const char *func, const char *format, ...) { pthread_mutex_lock(&mutex); @@ -25,7 +26,8 @@ void __logln(const char *func, const char *format, ...) pthread_mutex_unlock(&mutex); } -void __errln(const char *func, const char *format, ...) +__attribute__((__format__(__printf__, 2, 0))) void +__errln(const char *func, const char *format, ...) { pthread_mutex_lock(&mutex); @@ -1,19 +1,36 @@ #include <cpu.h> #include <mem.h> #include <log.h> +#include <gui.h> + +static struct cpu_interface cpu = { + .reg_names = gui_reg_names, + .reg_update = gui_reg_update, + .instr_done = gui_instr_done, +}; + +static struct gui_interface gui = { + .step_next = cpu_next, + .step_prev = cpu_prev, +}; int main(int argc, char *argv[]) { - (void)argc; - (void)argv; - if (argc != 2) { logln("invalid arguments"); return 1; } + cpu_register_interface(&cpu); cpu_exec(argv[1]); - mem_free_all(); + /* while (cpu_next() == OK) */ + /* ; */ + + gui_register_interface(&gui); + gui_init(); + + cpu_destroy(); + mem_destroy(); return 0; } @@ -46,7 +46,7 @@ void *mem_phys(vaddr virt) return &zero; } -void mem_free_all(void) +void mem_destroy(void) { struct memory *iterator = head; while (iterator) { |