From a0d223eef7c33382231510db9c851f856c891323 Mon Sep 17 00:00:00 2001 From: Jeffrey Pfau Date: Tue, 25 Oct 2016 21:12:35 -0700 Subject: [PATCH] Debugger: Modularize CLI debugger --- CHANGES | 1 + CMakeLists.txt | 31 +-- src/arm/debugger/cli-debugger.c | 46 ++-- src/debugger/cli-debugger.c | 298 ++++++++++++-------------- src/debugger/cli-debugger.h | 27 ++- src/debugger/debugger.c | 6 - src/debugger/debugger.h | 2 - src/feature/commandline.c | 8 +- src/feature/editline/cli-el-backend.c | 122 +++++++++++ src/feature/editline/cli-el-backend.h | 23 ++ src/gb/cli.c | 13 +- src/gb/cli.h | 2 - src/gb/core.c | 7 - src/gba/core.c | 7 - src/gba/extra/cli.c | 13 +- src/gba/extra/cli.h | 2 - src/lr35902/debugger/cli-debugger.c | 28 ++- src/platform/sdl/main.c | 11 +- 18 files changed, 380 insertions(+), 267 deletions(-) create mode 100644 src/feature/editline/cli-el-backend.c create mode 100644 src/feature/editline/cli-el-backend.h diff --git a/CHANGES b/CHANGES index b0fd9901b..fc04094ab 100644 --- a/CHANGES +++ b/CHANGES @@ -29,6 +29,7 @@ Misc: - PSP2: Improved controller rumble - GB, GBA: Prevent loading null ROMs - VFS: Allow truncating memory chunk VFiles + - Debugger: Modularize CLI debugger 0.5.1: (2016-10-05) Bugfixes: diff --git a/CMakeLists.txt b/CMakeLists.txt index 3804de52e..e9f62f030 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -6,7 +6,7 @@ if(NOT MSVC) else() set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -D_CRT_SECURE_NO_WARNINGS /wd4003 /wd4244 /wd4146") endif() -set(USE_CLI_DEBUGGER ON CACHE BOOL "Whether or not to enable the CLI-mode ARM debugger") +set(USE_EDITLINE ON CACHE BOOL "Whether or not to enable the CLI-mode debugger") set(USE_GDB_STUB ON CACHE BOOL "Whether or not to enable the GDB stub ARM debugger") set(USE_FFMPEG ON CACHE BOOL "Whether or not to enable FFmpeg support") set(USE_ZLIB ON CACHE BOOL "Whether or not to enable zlib support") @@ -314,7 +314,7 @@ if(CMAKE_SYSTEM_NAME MATCHES .*BSD) list(APPEND LIBEDIT_LIBRARIES -ltermcap) endif() else() - find_feature(USE_CLI_DEBUGGER "libedit") + find_feature(USE_EDITLINE "libedit") endif() if(BUILD_GL) find_package(OpenGL QUIET) @@ -350,7 +350,11 @@ find_feature(USE_EPOXY "epoxy") find_feature(USE_CMOCKA "cmocka") # Features -set(DEBUGGER_SRC ${CMAKE_CURRENT_SOURCE_DIR}/src/debugger/debugger.c) +set(DEBUGGER_SRC + ${CMAKE_CURRENT_SOURCE_DIR}/src/debugger/debugger.c + ${CMAKE_CURRENT_SOURCE_DIR}/src/debugger/parser.c + ${CMAKE_CURRENT_SOURCE_DIR}/src/debugger/cli-debugger.c) + set(FEATURE_SRC) set(CPACK_DEBIAN_PACKAGE_DEPENDS "libc6") @@ -358,21 +362,13 @@ if(DISABLE_DEPS) set(USE_GDB_STUB OFF) endif() -if(USE_CLI_DEBUGGER) - list(APPEND FEATURES CLI_DEBUGGER) - list(APPEND FEATURE_SRC ${CMAKE_CURRENT_SOURCE_DIR}/src/debugger/cli-debugger.c) - list(APPEND FEATURE_SRC ${CMAKE_CURRENT_SOURCE_DIR}/src/debugger/parser.c) - if(M_CORE_GBA) - list(APPEND FEATURE_SRC ${CMAKE_CURRENT_SOURCE_DIR}/src/arm/debugger/cli-debugger.c) - list(APPEND FEATURE_SRC ${CMAKE_CURRENT_SOURCE_DIR}/src/gba/extra/cli.c) - endif() - if(M_CORE_GB) - list(APPEND FEATURE_SRC ${CMAKE_CURRENT_SOURCE_DIR}/src/lr35902/debugger/cli-debugger.c) - endif() +if(USE_EDITLINE) + list(APPEND FEATURES EDITLINE) include_directories(AFTER ${LIBEDIT_INCLUDE_DIRS}) link_directories(${LIBEDIT_LIBRARY_DIRS}) set(DEBUGGER_LIB ${LIBEDIT_LIBRARIES}) set(CPACK_DEBIAN_PACKAGE_DEPENDS "${CPACK_DEBIAN_PACKAGE_DEPENDS},libedit2") + list(APPEND FEATURE_SRC "${CMAKE_CURRENT_SOURCE_DIR}/src/feature/editline/cli-el-backend.c") else() set(DEBUGGER_LIB "") endif() @@ -548,6 +544,8 @@ if(M_CORE_GB) ${CMAKE_CURRENT_SOURCE_DIR}/src/lr35902/debugger/debugger.c ${GB_SRC} ${GB_RENDERER_SRC}) + list(APPEND CLI_SRC + ${CMAKE_CURRENT_SOURCE_DIR}/src/lr35902/debugger/cli-debugger.c) list(APPEND TEST_SRC ${LR35902_TEST_SRC} ${GB_TEST_SRC}) @@ -562,6 +560,9 @@ if(M_CORE_GBA) ${GBA_SRC} ${GBA_CHEATS_SRC} ${GBA_RENDERER_SRC}) + list(APPEND CLI_SRC + ${CMAKE_CURRENT_SOURCE_DIR}/src/arm/debugger/cli-debugger.c + ${CMAKE_CURRENT_SOURCE_DIR}/src/gba/extra/cli.c) list(APPEND TEST_SRC ${ARM_TEST_SRC} ${GBA_TEST_SRC}) @@ -801,7 +802,7 @@ if(NOT QUIET) message(STATUS " Game Boy Advance: ${M_CORE_GBA}") message(STATUS " Game Boy: ${M_CORE_GB}") message(STATUS "Features:") - message(STATUS " CLI debugger: ${USE_CLI_DEBUGGER}") + message(STATUS " CLI debugger: ${USE_EDITLINE}") message(STATUS " GDB stub: ${USE_GDB_STUB}") message(STATUS " Video recording: ${USE_FFMPEG}") message(STATUS " GIF recording: ${USE_MAGICK}") diff --git a/src/arm/debugger/cli-debugger.c b/src/arm/debugger/cli-debugger.c index ba078118d..dbee4c538 100644 --- a/src/arm/debugger/cli-debugger.c +++ b/src/arm/debugger/cli-debugger.c @@ -5,7 +5,6 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "cli-debugger.h" -#ifdef USE_CLI_DEBUGGER #include "arm/debugger/debugger.h" #include "arm/debugger/memory-debugger.h" #include "arm/decoder.h" @@ -38,15 +37,15 @@ static struct CLIDebuggerCommandSummary _armCommands[] = { { 0, 0, 0, 0 } }; -static inline void _printPSR(union PSR psr) { - printf("%08X [%c%c%c%c%c%c%c]\n", psr.packed, - psr.n ? 'N' : '-', - psr.z ? 'Z' : '-', - psr.c ? 'C' : '-', - psr.v ? 'V' : '-', - psr.i ? 'I' : '-', - psr.f ? 'F' : '-', - psr.t ? 'T' : '-'); +static inline void _printPSR(struct CLIDebuggerBackend* be, union PSR psr) { + be->printf(be, "%08X [%c%c%c%c%c%c%c]\n", psr.packed, + psr.n ? 'N' : '-', + psr.z ? 'Z' : '-', + psr.c ? 'C' : '-', + psr.v ? 'V' : '-', + psr.i ? 'I' : '-', + psr.f ? 'F' : '-', + psr.t ? 'T' : '-'); } static void _disassemble(struct CLIDebuggerSystem* debugger, struct CLIDebugVector* dv) { @@ -95,14 +94,15 @@ static void _disassembleMode(struct CLIDebugger* debugger, struct CLIDebugVector } static inline uint32_t _printLine(struct CLIDebugger* debugger, uint32_t address, enum ExecutionMode mode) { + struct CLIDebuggerBackend* be = debugger->backend; char disassembly[48]; struct ARMInstructionInfo info; - printf("%08X: ", address); + be->printf(be, "%08X: ", address); if (mode == MODE_ARM) { uint32_t instruction = debugger->d.core->busRead32(debugger->d.core, address); ARMDecodeARM(instruction, &info); ARMDisassemble(&info, address + WORD_SIZE_ARM * 2, disassembly, sizeof(disassembly)); - printf("%08X\t%s\n", instruction, disassembly); + be->printf(be, "%08X\t%s\n", instruction, disassembly); return WORD_SIZE_ARM; } else { struct ARMInstructionInfo info2; @@ -113,27 +113,28 @@ static inline uint32_t _printLine(struct CLIDebugger* debugger, uint32_t address ARMDecodeThumb(instruction2, &info2); if (ARMDecodeThumbCombine(&info, &info2, &combined)) { ARMDisassemble(&combined, address + WORD_SIZE_THUMB * 2, disassembly, sizeof(disassembly)); - printf("%04X %04X\t%s\n", instruction, instruction2, disassembly); + be->printf(be, "%04X %04X\t%s\n", instruction, instruction2, disassembly); return WORD_SIZE_THUMB * 2; } else { ARMDisassemble(&info, address + WORD_SIZE_THUMB * 2, disassembly, sizeof(disassembly)); - printf("%04X \t%s\n", instruction, disassembly); + be->printf(be, "%04X \t%s\n", instruction, disassembly); return WORD_SIZE_THUMB; } } } static void _printStatus(struct CLIDebuggerSystem* debugger) { + struct CLIDebuggerBackend* be = debugger->p->backend; struct ARMCore* cpu = debugger->p->d.core->cpu; int r; for (r = 0; r < 4; ++r) { - printf("%08X %08X %08X %08X\n", + be->printf(be, "%08X %08X %08X %08X\n", cpu->gprs[r << 2], cpu->gprs[(r << 2) + 1], cpu->gprs[(r << 2) + 2], cpu->gprs[(r << 2) + 3]); } - _printPSR(cpu->cpsr); + _printPSR(be, cpu->cpsr); int instructionLength; enum ExecutionMode mode = cpu->cpsr.t; if (mode == MODE_ARM) { @@ -145,13 +146,14 @@ static void _printStatus(struct CLIDebuggerSystem* debugger) { } static void _writeRegister(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; struct ARMCore* cpu = debugger->d.core->cpu; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } if (!dv->next || dv->next->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t regid = dv->intValue; @@ -163,8 +165,9 @@ static void _writeRegister(struct CLIDebugger* debugger, struct CLIDebugVector* } static void _setBreakpointARM(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -172,8 +175,9 @@ static void _setBreakpointARM(struct CLIDebugger* debugger, struct CLIDebugVecto } static void _setBreakpointThumb(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -215,5 +219,3 @@ void ARMCLIDebuggerCreate(struct CLIDebuggerSystem* debugger) { debugger->platformName = "ARM"; debugger->platformCommands = _armCommands; } - -#endif diff --git a/src/debugger/cli-debugger.c b/src/debugger/cli-debugger.c index 06d895c20..caf4e0250 100644 --- a/src/debugger/cli-debugger.c +++ b/src/debugger/cli-debugger.c @@ -5,11 +5,10 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "cli-debugger.h" -#ifdef USE_CLI_DEBUGGER - #include "core/core.h" #include "core/version.h" -#include "parser.h" +#include "debugger/parser.h" +#include "util/string.h" #include @@ -20,8 +19,6 @@ const char* ERROR_MISSING_ARGS = "Arguments missing"; // TODO: share const char* ERROR_OVERFLOW = "Arguments overflow"; -static struct CLIDebugger* _activeDebugger; - #ifndef NDEBUG static void _breakInto(struct CLIDebugger*, struct CLIDebugVector*); #endif @@ -48,8 +45,6 @@ static void _dumpByte(struct CLIDebugger*, struct CLIDebugVector*); static void _dumpHalfword(struct CLIDebugger*, struct CLIDebugVector*); static void _dumpWord(struct CLIDebugger*, struct CLIDebugVector*); -static void _breakIntoDefault(int signal); - static struct CLIDebuggerCommandSummary _debuggerCommands[] = { { "b", _setBreakpoint, CLIDVParse, "Set a breakpoint" }, { "break", _setBreakpoint, CLIDVParse, "Set a breakpoint" }, @@ -133,68 +128,64 @@ static void _disassemble(struct CLIDebugger* debugger, struct CLIDebugVector* dv } static void _print(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { - UNUSED(debugger); for (; dv; dv = dv->next) { - printf(" %u", dv->intValue); + debugger->backend->printf(debugger->backend, " %u", dv->intValue); } - printf("\n"); + debugger->backend->printf(debugger->backend, "\n"); } static void _printBin(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { - UNUSED(debugger); for (; dv; dv = dv->next) { - printf(" 0b"); + debugger->backend->printf(debugger->backend, " 0b"); int i = 32; while (i--) { - printf("%u", (dv->intValue >> i) & 1); + debugger->backend->printf(debugger->backend, "%u", (dv->intValue >> i) & 1); } } - printf("\n"); + debugger->backend->printf(debugger->backend, "\n"); } static void _printHex(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { - UNUSED(debugger); for (; dv; dv = dv->next) { - printf(" 0x%08X", dv->intValue); + debugger->backend->printf(debugger->backend, " 0x%08X", dv->intValue); } - printf("\n"); + debugger->backend->printf(debugger->backend, "\n"); } static void _printHelp(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { - UNUSED(debugger); UNUSED(dv); if (!dv) { - puts("Generic commands:"); + debugger->backend->printf(debugger->backend, "Generic commands:"); int i; for (i = 0; _debuggerCommands[i].name; ++i) { - printf("%-10s %s\n", _debuggerCommands[i].name, _debuggerCommands[i].summary); + debugger->backend->printf(debugger->backend, "%-10s %s\n", _debuggerCommands[i].name, _debuggerCommands[i].summary); } if (debugger->system) { - printf("%s commands:\n", debugger->system->platformName); + debugger->backend->printf(debugger->backend, "%s commands:\n", debugger->system->platformName); for (i = 0; debugger->system->platformCommands[i].name; ++i) { - printf("%-10s %s\n", debugger->system->platformCommands[i].name, debugger->system->platformCommands[i].summary); + debugger->backend->printf(debugger->backend, "%-10s %s\n", debugger->system->platformCommands[i].name, debugger->system->platformCommands[i].summary); } - printf("%s commands:\n", debugger->system->name); + debugger->backend->printf(debugger->backend, "%s commands:\n", debugger->system->name); for (i = 0; debugger->system->commands[i].name; ++i) { - printf("%-10s %s\n", debugger->system->commands[i].name, debugger->system->commands[i].summary); + debugger->backend->printf(debugger->backend, "%-10s %s\n", debugger->system->commands[i].name, debugger->system->commands[i].summary); } } } else { int i; for (i = 0; _debuggerCommands[i].name; ++i) { if (strcmp(_debuggerCommands[i].name, dv->charValue) == 0) { - printf(" %s\n", _debuggerCommands[i].summary); + debugger->backend->printf(debugger->backend, " %s\n", _debuggerCommands[i].summary); } } if (debugger->system) { for (i = 0; debugger->system->platformCommands[i].name; ++i) { if (strcmp(debugger->system->platformCommands[i].name, dv->charValue) == 0) { - printf(" %s\n", debugger->system->platformCommands[i].summary); + debugger->backend->printf(debugger->backend, " %s\n", debugger->system->platformCommands[i].summary); } } for (i = 0; debugger->system->commands[i].name; ++i) { if (strcmp(debugger->system->commands[i].name, dv->charValue) == 0) { - printf(" %s\n", debugger->system->commands[i].summary); + debugger->backend->printf(debugger->backend, " %s\n", debugger->system->commands[i].summary); } } } @@ -208,12 +199,12 @@ static void _quit(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { static void _readByte(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; uint8_t value = debugger->d.core->busRead8(debugger->d.core, address); - printf(" 0x%02X\n", value); + debugger->backend->printf(debugger->backend, " 0x%02X\n", value); } static void _reset(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { @@ -224,37 +215,37 @@ static void _reset(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { static void _readHalfword(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; uint16_t value = debugger->d.core->busRead16(debugger->d.core, address & ~1); - printf(" 0x%04X\n", value); + debugger->backend->printf(debugger->backend, " 0x%04X\n", value); } static void _readWord(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; uint32_t value = debugger->d.core->busRead32(debugger->d.core, address & ~3); - printf(" 0x%08X\n", value); + debugger->backend->printf(debugger->backend, " 0x%08X\n", value); } static void _writeByte(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } if (!dv->next || dv->next->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; uint32_t value = dv->next->intValue; if (value > 0xFF) { - printf("%s\n", ERROR_OVERFLOW); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_OVERFLOW); return; } debugger->d.core->busWrite8(debugger->d.core, address, value); @@ -262,17 +253,17 @@ static void _writeByte(struct CLIDebugger* debugger, struct CLIDebugVector* dv) static void _writeHalfword(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } if (!dv->next || dv->next->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; uint32_t value = dv->next->intValue; if (value > 0xFFFF) { - printf("%s\n", ERROR_OVERFLOW); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_OVERFLOW); return; } debugger->d.core->busWrite16(debugger->d.core, address, value); @@ -280,11 +271,11 @@ static void _writeHalfword(struct CLIDebugger* debugger, struct CLIDebugVector* static void _writeWord(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } if (!dv->next || dv->next->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -294,7 +285,7 @@ static void _writeWord(struct CLIDebugger* debugger, struct CLIDebugVector* dv) static void _dumpByte(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -307,18 +298,18 @@ static void _dumpByte(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (line > words) { line = words; } - printf("0x%08X:", address); + debugger->backend->printf(debugger->backend, "0x%08X:", address); for (; line > 0; --line, ++address, --words) { uint32_t value = debugger->d.core->busRead8(debugger->d.core, address); - printf(" %02X", value); + debugger->backend->printf(debugger->backend, " %02X", value); } - printf("\n"); + debugger->backend->printf(debugger->backend, "\n"); } } static void _dumpHalfword(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -331,18 +322,18 @@ static void _dumpHalfword(struct CLIDebugger* debugger, struct CLIDebugVector* d if (line > words) { line = words; } - printf("0x%08X:", address); + debugger->backend->printf(debugger->backend, "0x%08X:", address); for (; line > 0; --line, address += 2, --words) { uint32_t value = debugger->d.core->busRead16(debugger->d.core, address); - printf(" %04X", value); + debugger->backend->printf(debugger->backend, " %04X", value); } - printf("\n"); + debugger->backend->printf(debugger->backend, "\n"); } } static void _dumpWord(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -355,18 +346,18 @@ static void _dumpWord(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (line > words) { line = words; } - printf("0x%08X:", address); + debugger->backend->printf(debugger->backend, "0x%08X:", address); for (; line > 0; --line, address += 4, --words) { uint32_t value = debugger->d.core->busRead32(debugger->d.core, address); - printf(" %08X", value); + debugger->backend->printf(debugger->backend, " %08X", value); } - printf("\n"); + debugger->backend->printf(debugger->backend, "\n"); } } static void _setBreakpoint(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -375,11 +366,11 @@ static void _setBreakpoint(struct CLIDebugger* debugger, struct CLIDebugVector* static void _setWatchpoint(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } if (!debugger->d.platform->setWatchpoint) { - printf("Watchpoints are not supported by this platform.\n"); + debugger->backend->printf(debugger->backend, "Watchpoints are not supported by this platform.\n"); return; } uint32_t address = dv->intValue; @@ -388,7 +379,7 @@ static void _setWatchpoint(struct CLIDebugger* debugger, struct CLIDebugVector* static void _clearBreakpoint(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + debugger->backend->printf(debugger->backend, "%s\n", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; @@ -398,11 +389,6 @@ static void _clearBreakpoint(struct CLIDebugger* debugger, struct CLIDebugVector } } -static void _breakIntoDefault(int signal) { - UNUSED(signal); - mDebuggerEnter(&_activeDebugger->d, DEBUGGER_ENTER_MANUAL, 0); -} - static void _printStatus(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { UNUSED(dv); debugger->system->printStatus(debugger->system); @@ -537,8 +523,7 @@ struct CLIDebugVector* CLIDVStringParse(struct CLIDebugger* debugger, const char } else { adjusted = length; } - dvTemp.charValue = malloc(adjusted); - strncpy(dvTemp.charValue, string, adjusted); + dvTemp.charValue = strndup(string, adjusted); length -= adjusted; string += adjusted; @@ -579,13 +564,13 @@ static int _tryCommands(struct CLIDebugger* debugger, struct CLIDebuggerCommandS if (args) { dv = commands[i].parser(debugger, args, argsLen); if (dv && dv->type == CLIDV_ERROR_TYPE) { - printf("Parse error\n"); + debugger->backend->printf(debugger->backend, "Parse error\n"); _DVFree(dv); return false; } } } else if (args) { - printf("Wrong number of arguments\n"); + debugger->backend->printf(debugger->backend, "Wrong number of arguments\n"); return false; } commands[i].command(debugger, dv); @@ -617,155 +602,88 @@ static bool _parse(struct CLIDebugger* debugger, const char* line, size_t count) } } if (result < 0) { - printf("Command not found\n"); + debugger->backend->printf(debugger->backend, "Command not found\n"); } return false; } -static char* _prompt(EditLine* el) { - UNUSED(el); - return "> "; -} - static void _commandLine(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; const char* line; + size_t len; _printStatus(cliDebugger, 0); - int count = 0; - HistEvent ev; while (debugger->state == DEBUGGER_PAUSED) { - line = el_gets(cliDebugger->elstate, &count); - if (!line) { + line = cliDebugger->backend->readline(cliDebugger->backend, &len); + if (!line || len == 0) { debugger->state = DEBUGGER_SHUTDOWN; return; } if (line[0] == '\n') { - if (history(cliDebugger->histate, &ev, H_FIRST) >= 0) { - _parse(cliDebugger, ev.str, strlen(ev.str) - 1); + line = cliDebugger->backend->historyLast(cliDebugger->backend, &len); + if (line && len) { + _parse(cliDebugger, line, len); } } else { - _parse(cliDebugger, line, count - 1); - history(cliDebugger->histate, &ev, H_ENTER, line); + _parse(cliDebugger, line, len); + cliDebugger->backend->historyAppend(cliDebugger->backend, line); } } } static void _reportEntry(struct mDebugger* debugger, enum mDebuggerEntryReason reason, struct mDebuggerEntryInfo* info) { - UNUSED(debugger); + struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; switch (reason) { case DEBUGGER_ENTER_MANUAL: case DEBUGGER_ENTER_ATTACHED: break; case DEBUGGER_ENTER_BREAKPOINT: if (info) { - printf("Hit breakpoint at 0x%08X\n", info->address); + cliDebugger->backend->printf(cliDebugger->backend, "Hit breakpoint at 0x%08X\n", info->address); } else { - printf("Hit breakpoint\n"); + cliDebugger->backend->printf(cliDebugger->backend, "Hit breakpoint\n"); } break; case DEBUGGER_ENTER_WATCHPOINT: if (info) { if (info->accessType & WATCHPOINT_WRITE) { - printf("Hit watchpoint at 0x%08X: (new value = 0x%08x, old value = 0x%08X)\n", info->address, info->newValue, info->oldValue); + cliDebugger->backend->printf(cliDebugger->backend, "Hit watchpoint at 0x%08X: (new value = 0x%08x, old value = 0x%08X)\n", info->address, info->newValue, info->oldValue); } else { - printf("Hit watchpoint at 0x%08X: (value = 0x%08x)\n", info->address, info->oldValue); + cliDebugger->backend->printf(cliDebugger->backend, "Hit watchpoint at 0x%08X: (value = 0x%08x)\n", info->address, info->oldValue); } } else { - printf("Hit watchpoint\n"); + cliDebugger->backend->printf(cliDebugger->backend, "Hit watchpoint\n"); } break; case DEBUGGER_ENTER_ILLEGAL_OP: if (info) { - printf("Hit illegal opcode at 0x%08X: 0x%08X\n", info->address, info->opcode); + cliDebugger->backend->printf(cliDebugger->backend, "Hit illegal opcode at 0x%08X: 0x%08X\n", info->address, info->opcode); } else { - printf("Hit illegal opcode\n"); + cliDebugger->backend->printf(cliDebugger->backend, "Hit illegal opcode\n"); } break; } } -static unsigned char _tabComplete(EditLine* elstate, int ch) { - UNUSED(ch); - const LineInfo* li = el_line(elstate); - if (!li->buffer[0]) { - return CC_ERROR; - } - - const char* commandPtr; - size_t cmd = 0, len = 0; - const char* name = 0; - for (commandPtr = li->buffer; commandPtr <= li->cursor; ++commandPtr, ++len) { - for (; (name = _debuggerCommands[cmd].name); ++cmd) { - int cmp = strncasecmp(name, li->buffer, len); - if (cmp > 0) { - return CC_ERROR; - } - if (cmp == 0) { - break; - } - } - } - if (!name) { - return CC_ERROR; - } - if (_debuggerCommands[cmd + 1].name && strlen(_debuggerCommands[cmd + 1].name) >= len - 1 && name[len - 2] == _debuggerCommands[cmd + 1].name[len - 2]) { - --len; - const char* next = 0; - int i; - for (i = cmd + 1; _debuggerCommands[i].name; ++i) { - if (strncasecmp(name, _debuggerCommands[i].name, len)) { - break; - } - next = _debuggerCommands[i].name; - } - if (!next) { - return CC_ERROR; - } - - for (; name[len]; ++len) { - if (name[len] != next[len]) { - break; - } - char out[2] = { name[len], '\0' }; - el_insertstr(elstate, out); - } - return CC_REDISPLAY; - } - name += len - 1; - el_insertstr(elstate, name); - el_insertstr(elstate, " "); - return CC_REDISPLAY; -} - static void _cliDebuggerInit(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; - // TODO: get argv[0] - cliDebugger->elstate = el_init(binaryName, stdin, stdout, stderr); - el_set(cliDebugger->elstate, EL_PROMPT, _prompt); - el_set(cliDebugger->elstate, EL_EDITOR, "emacs"); - - el_set(cliDebugger->elstate, EL_CLIENTDATA, cliDebugger); - el_set(cliDebugger->elstate, EL_ADDFN, "tab-complete", "Tab completion", _tabComplete); - el_set(cliDebugger->elstate, EL_BIND, "\t", "tab-complete", 0); - cliDebugger->histate = history_init(); - HistEvent ev; - history(cliDebugger->histate, &ev, H_SETSIZE, 200); - el_set(cliDebugger->elstate, EL_HIST, history, cliDebugger->histate); - _activeDebugger = cliDebugger; - signal(SIGINT, _breakIntoDefault); + cliDebugger->backend->init(cliDebugger->backend); } static void _cliDebuggerDeinit(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; - history_end(cliDebugger->histate); - el_end(cliDebugger->elstate); - if (cliDebugger->system) { if (cliDebugger->system->deinit) { cliDebugger->system->deinit(cliDebugger->system); } free(cliDebugger->system); - cliDebugger->system = 0; + cliDebugger->system = NULL; + } + if (cliDebugger->backend) { + if (cliDebugger->backend->deinit) { + cliDebugger->backend->deinit(cliDebugger->backend); + } + free(cliDebugger->backend); + cliDebugger->backend = NULL; } } @@ -787,7 +705,8 @@ void CLIDebuggerCreate(struct CLIDebugger* debugger) { debugger->d.paused = _commandLine; debugger->d.entered = _reportEntry; - debugger->system = 0; + debugger->system = NULL; + debugger->backend = NULL; } void CLIDebuggerAttachSystem(struct CLIDebugger* debugger, struct CLIDebuggerSystem* system) { @@ -802,4 +721,61 @@ void CLIDebuggerAttachSystem(struct CLIDebugger* debugger, struct CLIDebuggerSys system->p = debugger; } -#endif +void CLIDebuggerAttachBackend(struct CLIDebugger* debugger, struct CLIDebuggerBackend* backend) { + if (debugger->backend) { + if (debugger->backend->deinit) { + debugger->backend->deinit(debugger->backend); + } + free(debugger->backend); + } + + debugger->backend = backend; + backend->p = debugger; +} + +bool CLIDebuggerTabComplete(struct CLIDebugger* debugger, const char* token, bool initial, size_t tokenLen) { + size_t cmd = 0; + size_t len; + const char* name = 0; + for (len = 1; len <= tokenLen; ++len) { + for (; (name = _debuggerCommands[cmd].name); ++cmd) { + int cmp = strncasecmp(name, token, len); + if (cmp > 0) { + return false; + } + if (cmp == 0) { + break; + } + } + } + if (!name) { + return false; + } + if (_debuggerCommands[cmd + 1].name && strlen(_debuggerCommands[cmd + 1].name) >= len && name[len - 1] == _debuggerCommands[cmd + 1].name[len - 1]) { + --len; + const char* next = 0; + int i; + for (i = cmd + 1; _debuggerCommands[i].name; ++i) { + if (strncasecmp(name, _debuggerCommands[i].name, len)) { + break; + } + next = _debuggerCommands[i].name; + } + if (!next) { + return false; + } + + for (; name[len]; ++len) { + if (name[len] != next[len]) { + break; + } + char out[2] = { name[len], '\0' }; + debugger->backend->lineAppend(debugger->backend, out); + } + return true; + } + name += len - 1; + debugger->backend->lineAppend(debugger->backend, name); + debugger->backend->lineAppend(debugger->backend, " "); + return true; +} diff --git a/src/debugger/cli-debugger.h b/src/debugger/cli-debugger.h index daad59d11..98e19b5ab 100644 --- a/src/debugger/cli-debugger.h +++ b/src/debugger/cli-debugger.h @@ -1,4 +1,4 @@ -/* Copyright (c) 2013-2014 Jeffrey Pfau +/* Copyright (c) 2013-2016 Jeffrey Pfau * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this @@ -10,9 +10,6 @@ #include "debugger.h" -#ifdef USE_CLI_DEBUGGER -#include - struct CLIDebugger; struct CLIDebugVector { @@ -59,13 +56,25 @@ struct CLIDebuggerSystem { const char* platformName; }; +struct CLIDebuggerBackend { + struct CLIDebugger* p; + + void (*init)(struct CLIDebuggerBackend*); + void (*deinit)(struct CLIDebuggerBackend*); + + ATTRIBUTE_FORMAT(printf, 2, 3) + void (*printf)(struct CLIDebuggerBackend*, const char* fmt, ...); + const char* (*readline)(struct CLIDebuggerBackend*, size_t* len); + void (*lineAppend)(struct CLIDebuggerBackend*, const char* line); + const char* (*historyLast)(struct CLIDebuggerBackend*, size_t* len); + void (*historyAppend)(struct CLIDebuggerBackend*, const char* line); +}; + struct CLIDebugger { struct mDebugger d; struct CLIDebuggerSystem* system; - - EditLine* elstate; - History* histate; + struct CLIDebuggerBackend* backend; }; struct CLIDebugVector* CLIDVParse(struct CLIDebugger* debugger, const char* string, size_t length); @@ -73,6 +82,8 @@ struct CLIDebugVector* CLIDVStringParse(struct CLIDebugger* debugger, const char void CLIDebuggerCreate(struct CLIDebugger*); void CLIDebuggerAttachSystem(struct CLIDebugger*, struct CLIDebuggerSystem*); -#endif +void CLIDebuggerAttachBackend(struct CLIDebugger*, struct CLIDebuggerBackend*); + +bool CLIDebuggerTabComplete(struct CLIDebugger*, const char* token, bool initial, size_t len); #endif diff --git a/src/debugger/debugger.c b/src/debugger/debugger.c index bad3b558c..6ddbc7dcb 100644 --- a/src/debugger/debugger.c +++ b/src/debugger/debugger.c @@ -7,9 +7,7 @@ #include "core/core.h" -#ifdef USE_CLI_DEBUGGER #include "debugger/cli-debugger.h" -#endif #ifdef USE_GDB_STUB #include "debugger/gdb-stub.h" @@ -29,9 +27,7 @@ struct mDebugger* mDebuggerCreate(enum mDebuggerType type, struct mCore* core) { union DebugUnion { struct mDebugger d; -#ifdef USE_CLI_DEBUGGER struct CLIDebugger cli; -#endif #ifdef USE_GDB_STUB struct GDBStub gdb; #endif @@ -40,13 +36,11 @@ struct mDebugger* mDebuggerCreate(enum mDebuggerType type, struct mCore* core) { union DebugUnion* debugger = malloc(sizeof(union DebugUnion)); switch (type) { -#ifdef USE_CLI_DEBUGGER case DEBUGGER_CLI: CLIDebuggerCreate(&debugger->cli); struct CLIDebuggerSystem* sys = core->cliDebuggerSystem(core); CLIDebuggerAttachSystem(&debugger->cli, sys); break; -#endif #ifdef USE_GDB_STUB case DEBUGGER_GDB: GDBStubCreate(&debugger->gdb); diff --git a/src/debugger/debugger.h b/src/debugger/debugger.h index c4aca9df3..a4ce367f6 100644 --- a/src/debugger/debugger.h +++ b/src/debugger/debugger.h @@ -18,9 +18,7 @@ extern const uint32_t DEBUGGER_ID; enum mDebuggerType { DEBUGGER_NONE = 0, -#ifdef USE_CLI_DEBUGGER DEBUGGER_CLI, -#endif #ifdef USE_GDB_STUB DEBUGGER_GDB, #endif diff --git a/src/feature/commandline.c b/src/feature/commandline.c index c0d5032d6..21c2a9af8 100644 --- a/src/feature/commandline.c +++ b/src/feature/commandline.c @@ -31,7 +31,7 @@ static const struct option _options[] = { { "bios", required_argument, 0, 'b' }, { "cheats", required_argument, 0, 'c' }, { "frameskip", required_argument, 0, 's' }, -#ifdef USE_CLI_DEBUGGER +#ifdef USE_EDITLINE { "debug", no_argument, 0, 'd' }, #endif #ifdef USE_GDB_STUB @@ -51,7 +51,7 @@ bool parseArguments(struct mArguments* args, int argc, char* const* argv, struct int ch; char options[64] = "b:c:hl:p:s:v:" -#ifdef USE_CLI_DEBUGGER +#ifdef USE_EDITLINE "d" #endif #ifdef USE_GDB_STUB @@ -82,7 +82,7 @@ bool parseArguments(struct mArguments* args, int argc, char* const* argv, struct case 'c': args->cheatsFile = strdup(optarg); break; -#ifdef USE_CLI_DEBUGGER +#ifdef USE_EDITLINE case 'd': if (args->debuggerType != DEBUGGER_NONE) { return false; @@ -209,7 +209,7 @@ void usage(const char* arg0, const char* extraOptions) { puts("\nGeneric options:"); puts(" -b, --bios FILE GBA BIOS file to use"); puts(" -c, --cheats FILE Apply cheat codes from a file"); -#ifdef USE_CLI_DEBUGGER +#ifdef USE_EDITLINE puts(" -d, --debug Use command-line debugger"); #endif #ifdef USE_GDB_STUB diff --git a/src/feature/editline/cli-el-backend.c b/src/feature/editline/cli-el-backend.c new file mode 100644 index 000000000..565837116 --- /dev/null +++ b/src/feature/editline/cli-el-backend.c @@ -0,0 +1,122 @@ +/* Copyright (c) 2013-2014 Jeffrey Pfau + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ +#include "cli-el-backend.h" + +#include "core/version.h" + +#include + +static struct CLIDebugger* _activeDebugger; + +static char* _prompt(EditLine* el) { + UNUSED(el); + return "> "; +} + +static void _breakIntoDefault(int signal) { + UNUSED(signal); + mDebuggerEnter(&_activeDebugger->d, DEBUGGER_ENTER_MANUAL, 0); +} + +static unsigned char _tabComplete(EditLine* elstate, int ch) { + UNUSED(ch); + const LineInfo* li = el_line(elstate); + if (!li->buffer[0]) { + return CC_ERROR; + } + + struct CLIDebuggerEditLineBackend* elbe; + el_get(elstate, EL_CLIENTDATA, &elbe); + // TODO: not always true + if (CLIDebuggerTabComplete(elbe->d.p, li->buffer, true, li->cursor - li->buffer)) { + return CC_REDISPLAY; + } + return CC_ERROR; +} + +ATTRIBUTE_FORMAT(printf, 2, 3) +void _CLIDebuggerEditLinePrintf(struct CLIDebuggerBackend* be, const char* fmt, ...) { + UNUSED(be); + va_list args; + va_start(args, fmt); + vprintf(fmt, args); + va_end(args); +} + +void _CLIDebuggerEditLineInit(struct CLIDebuggerBackend* be) { + struct CLIDebuggerEditLineBackend* elbe = (struct CLIDebuggerEditLineBackend*) be; + // TODO: get argv[0] + elbe->elstate = el_init(binaryName, stdin, stdout, stderr); + el_set(elbe->elstate, EL_PROMPT, _prompt); + el_set(elbe->elstate, EL_EDITOR, "emacs"); + + el_set(elbe->elstate, EL_CLIENTDATA, elbe); + el_set(elbe->elstate, EL_ADDFN, "tab-complete", "Tab completion", _tabComplete); + el_set(elbe->elstate, EL_BIND, "\t", "tab-complete", 0); + elbe->histate = history_init(); + HistEvent ev; + history(elbe->histate, &ev, H_SETSIZE, 200); + el_set(elbe->elstate, EL_HIST, history, elbe->histate); + _activeDebugger = be->p; + signal(SIGINT, _breakIntoDefault); +} + +void _CLIDebuggerEditLineDeinit(struct CLIDebuggerBackend* be) { + struct CLIDebuggerEditLineBackend* elbe = (struct CLIDebuggerEditLineBackend*) be; + history_end(elbe->histate); + el_end(elbe->elstate); +} + +const char* _CLIDebuggerEditLineReadLine(struct CLIDebuggerBackend* be, size_t* len) { + struct CLIDebuggerEditLineBackend* elbe = (struct CLIDebuggerEditLineBackend*) be; + int count; + *len = 0; + const char* line = el_gets(elbe->elstate, &count); + if (line && count >= 1) { + // Crop off newline + *len = (size_t) count - 1; + } + return line; +} +void _CLIDebuggerEditLineLineAppend(struct CLIDebuggerBackend* be, const char* line) { + struct CLIDebuggerEditLineBackend* elbe = (struct CLIDebuggerEditLineBackend*) be; + el_insertstr(elbe->elstate, line); +} + +const char* _CLIDebuggerEditLineHistoryLast(struct CLIDebuggerBackend* be, size_t* len) { + struct CLIDebuggerEditLineBackend* elbe = (struct CLIDebuggerEditLineBackend*) be; + HistEvent ev; + if (history(elbe->histate, &ev, H_FIRST) < 0) { + *len = 0; + return NULL; + } + const char* newline = strchr(ev.str, '\n'); + if (!newline) { + *len = strlen(ev.str); + } else { + *len = newline - ev.str; + } + + return ev.str; +} + +void _CLIDebuggerEditLineHistoryAppend(struct CLIDebuggerBackend* be, const char* line) { + struct CLIDebuggerEditLineBackend* elbe = (struct CLIDebuggerEditLineBackend*) be; + HistEvent ev; + history(elbe->histate, &ev, H_ENTER, line); +} + +struct CLIDebuggerBackend* CLIDebuggerEditLineBackendCreate(void) { + struct CLIDebuggerEditLineBackend* elbe = malloc(sizeof(*elbe)); + elbe->d.printf = _CLIDebuggerEditLinePrintf; + elbe->d.init = _CLIDebuggerEditLineInit; + elbe->d.deinit = _CLIDebuggerEditLineDeinit; + elbe->d.readline = _CLIDebuggerEditLineReadLine; + elbe->d.lineAppend = _CLIDebuggerEditLineLineAppend; + elbe->d.historyLast = _CLIDebuggerEditLineHistoryLast; + elbe->d.historyAppend = _CLIDebuggerEditLineHistoryAppend; + return &elbe->d; +} diff --git a/src/feature/editline/cli-el-backend.h b/src/feature/editline/cli-el-backend.h new file mode 100644 index 000000000..df3986e07 --- /dev/null +++ b/src/feature/editline/cli-el-backend.h @@ -0,0 +1,23 @@ +/* Copyright (c) 2013-2014 Jeffrey Pfau + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ +#ifndef CLI_EL_BACKEND_H +#define CLI_EL_BACKEND_H + +#include "debugger/cli-debugger.h" +#include "util/common.h" + +#include + +struct CLIDebuggerEditLineBackend { + struct CLIDebuggerBackend d; + + EditLine* elstate; + History* histate; +}; + +struct CLIDebuggerBackend* CLIDebuggerEditLineBackendCreate(void); + +#endif diff --git a/src/gb/cli.c b/src/gb/cli.c index 6aaae8ac1..0f8f7d945 100644 --- a/src/gb/cli.c +++ b/src/gb/cli.c @@ -12,8 +12,6 @@ #include "gb/video.h" #include "lr35902/debugger/cli-debugger.h" -#ifdef USE_CLI_DEBUGGER - static void _GBCLIDebuggerInit(struct CLIDebuggerSystem*); static bool _GBCLIDebuggerCustom(struct CLIDebuggerSystem*); static uint32_t _GBCLIDebuggerLookupIdentifier(struct CLIDebuggerSystem*, const char* name, struct CLIDebugVector* dv); @@ -90,14 +88,15 @@ static void _frame(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { } static void _load(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } int state = dv->intValue; if (state < 1 || state > 9) { - printf("State %u out of range", state); + be->printf(be, "State %u out of range", state); } struct GBCLIDebugger* gbDebugger = (struct GBCLIDebugger*) debugger->system; @@ -106,18 +105,18 @@ static void _load(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { } static void _save(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } int state = dv->intValue; if (state < 1 || state > 9) { - printf("State %u out of range", state); + be->printf(be, "State %u out of range", state); } struct GBCLIDebugger* gbDebugger = (struct GBCLIDebugger*) debugger->system; mCoreSaveState(gbDebugger->core, dv->intValue, SAVESTATE_SCREENSHOT); } -#endif diff --git a/src/gb/cli.h b/src/gb/cli.h index f9b2fca9a..013843ce5 100644 --- a/src/gb/cli.h +++ b/src/gb/cli.h @@ -6,7 +6,6 @@ #ifndef GB_CLI_H #define GB_CLI_H -#ifdef USE_CLI_DEBUGGER #include "debugger/cli-debugger.h" struct GBCLIDebugger { @@ -19,6 +18,5 @@ struct GBCLIDebugger { }; struct CLIDebuggerSystem* GBCLIDebuggerCreate(struct mCore*); -#endif #endif diff --git a/src/gb/core.c b/src/gb/core.c index 3caa1eed8..cfab4b3b3 100644 --- a/src/gb/core.c +++ b/src/gb/core.c @@ -430,10 +430,8 @@ static void _GBCoreRawWrite32(struct mCore* core, uint32_t address, int segment, static bool _GBCoreSupportsDebuggerType(struct mCore* core, enum mDebuggerType type) { UNUSED(core); switch (type) { -#ifdef USE_CLI_DEBUGGER case DEBUGGER_CLI: return true; -#endif default: return false; } @@ -448,12 +446,7 @@ static struct mDebuggerPlatform* _GBCoreDebuggerPlatform(struct mCore* core) { } static struct CLIDebuggerSystem* _GBCoreCliDebuggerSystem(struct mCore* core) { -#ifdef USE_CLI_DEBUGGER return GBCLIDebuggerCreate(core); -#else - UNUSED(core); - return NULL; -#endif } static void _GBCoreAttachDebugger(struct mCore* core, struct mDebugger* debugger) { diff --git a/src/gba/core.c b/src/gba/core.c index 1a37df6d8..c838fab69 100644 --- a/src/gba/core.c +++ b/src/gba/core.c @@ -442,10 +442,8 @@ static void _GBACoreRawWrite32(struct mCore* core, uint32_t address, int segment static bool _GBACoreSupportsDebuggerType(struct mCore* core, enum mDebuggerType type) { UNUSED(core); switch (type) { -#ifdef USE_CLI_DEBUGGER case DEBUGGER_CLI: return true; -#endif #ifdef USE_GDB_STUB case DEBUGGER_GDB: return true; @@ -464,12 +462,7 @@ static struct mDebuggerPlatform* _GBACoreDebuggerPlatform(struct mCore* core) { } static struct CLIDebuggerSystem* _GBACoreCliDebuggerSystem(struct mCore* core) { -#ifdef USE_CLI_DEBUGGER return &GBACLIDebuggerCreate(core)->d; -#else - UNUSED(core); - return NULL; -#endif } static void _GBACoreAttachDebugger(struct mCore* core, struct mDebugger* debugger) { diff --git a/src/gba/extra/cli.c b/src/gba/extra/cli.c index 7434645e9..ca16a9117 100644 --- a/src/gba/extra/cli.c +++ b/src/gba/extra/cli.c @@ -10,8 +10,6 @@ #include "gba/io.h" #include "gba/serialize.h" -#ifdef USE_CLI_DEBUGGER - static void _GBACLIDebuggerInit(struct CLIDebuggerSystem*); static bool _GBACLIDebuggerCustom(struct CLIDebuggerSystem*); static uint32_t _GBACLIDebuggerLookupIdentifier(struct CLIDebuggerSystem*, const char* name, struct CLIDebugVector* dv); @@ -87,14 +85,15 @@ static void _frame(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { } static void _load(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } int state = dv->intValue; if (state < 1 || state > 9) { - printf("State %u out of range", state); + be->printf(be, "State %u out of range", state); } struct GBACLIDebugger* gbaDebugger = (struct GBACLIDebugger*) debugger->system; @@ -105,18 +104,18 @@ static void _load(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { // TODO: Put back rewind static void _save(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { + struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - printf("%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s\n", ERROR_MISSING_ARGS); return; } int state = dv->intValue; if (state < 1 || state > 9) { - printf("State %u out of range", state); + be->printf(be, "State %u out of range", state); } struct GBACLIDebugger* gbaDebugger = (struct GBACLIDebugger*) debugger->system; mCoreSaveState(gbaDebugger->core, dv->intValue, SAVESTATE_SCREENSHOT); } -#endif diff --git a/src/gba/extra/cli.h b/src/gba/extra/cli.h index f06fd6439..347f45437 100644 --- a/src/gba/extra/cli.h +++ b/src/gba/extra/cli.h @@ -6,7 +6,6 @@ #ifndef GBA_CLI_H #define GBA_CLI_H -#ifdef USE_CLI_DEBUGGER #include "debugger/cli-debugger.h" struct mCore; @@ -21,6 +20,5 @@ struct GBACLIDebugger { }; struct GBACLIDebugger* GBACLIDebuggerCreate(struct mCore*); -#endif #endif diff --git a/src/lr35902/debugger/cli-debugger.c b/src/lr35902/debugger/cli-debugger.c index 0a3d9cc29..349578dbf 100644 --- a/src/lr35902/debugger/cli-debugger.c +++ b/src/lr35902/debugger/cli-debugger.c @@ -5,7 +5,6 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "cli-debugger.h" -#ifdef USE_CLI_DEBUGGER #include "core/core.h" #include "debugger/cli-debugger.h" #include "lr35902/lr35902.h" @@ -16,22 +15,23 @@ static struct CLIDebuggerCommandSummary _lr35902Commands[] = { { 0, 0, 0, 0 } }; -static inline void _printFlags(union FlagRegister f) { - printf("[%c%c%c%c]\n", - f.z ? 'Z' : '-', - f.n ? 'N' : '-', - f.h ? 'H' : '-', - f.c ? 'C' : '-'); +static inline void _printFlags(struct CLIDebuggerBackend* be, union FlagRegister f) { + be->printf(be, "[%c%c%c%c]\n", + f.z ? 'Z' : '-', + f.n ? 'N' : '-', + f.h ? 'H' : '-', + f.c ? 'C' : '-'); } static void _printStatus(struct CLIDebuggerSystem* debugger) { + struct CLIDebuggerBackend* be = debugger->p->backend; struct LR35902Core* cpu = debugger->p->d.core->cpu; - printf("A: %02X F: %02X (AF: %04X)\n", cpu->a, cpu->f.packed, cpu->af); - printf("B: %02X C: %02X (BC: %04X)\n", cpu->b, cpu->c, cpu->bc); - printf("D: %02X E: %02X (DE: %04X)\n", cpu->d, cpu->e, cpu->de); - printf("H: %02X L: %02X (HL: %04X)\n", cpu->h, cpu->l, cpu->hl); - printf("PC: %04X SP: %04X\n", cpu->pc, cpu->sp); - _printFlags(cpu->f); + be->printf(be, "A: %02X F: %02X (AF: %04X)\n", cpu->a, cpu->f.packed, cpu->af); + be->printf(be, "B: %02X C: %02X (BC: %04X)\n", cpu->b, cpu->c, cpu->bc); + be->printf(be, "D: %02X E: %02X (DE: %04X)\n", cpu->d, cpu->e, cpu->de); + be->printf(be, "H: %02X L: %02X (HL: %04X)\n", cpu->h, cpu->l, cpu->hl); + be->printf(be, "PC: %04X SP: %04X\n", cpu->pc, cpu->sp); + _printFlags(be, cpu->f); } static uint32_t _lookupPlatformIdentifier(struct CLIDebuggerSystem* debugger, const char* name, struct CLIDebugVector* dv) { @@ -89,5 +89,3 @@ void LR35902CLIDebuggerCreate(struct CLIDebuggerSystem* debugger) { debugger->platformName = "GB-Z80"; debugger->platformCommands = _lr35902Commands; } - -#endif diff --git a/src/platform/sdl/main.c b/src/platform/sdl/main.c index 616248924..20495b5b6 100644 --- a/src/platform/sdl/main.c +++ b/src/platform/sdl/main.c @@ -5,13 +5,14 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "main.h" -#ifdef USE_CLI_DEBUGGER #include "debugger/cli-debugger.h" -#endif #ifdef USE_GDB_STUB #include "debugger/gdb-stub.h" #endif +#ifdef USE_EDITLINE +#include "feature/editline/cli-el-backend.h" +#endif #include "core/core.h" #include "core/config.h" @@ -169,6 +170,12 @@ int mSDLRun(struct mSDLRenderer* renderer, struct mArguments* args) { mCoreAutoloadSave(renderer->core); struct mDebugger* debugger = mDebuggerCreate(args->debuggerType, renderer->core); if (debugger) { +#ifdef USE_EDITLINE + if (args->debuggerType == DEBUGGER_CLI) { + struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; + CLIDebuggerAttachBackend(cliDebugger, CLIDebuggerEditLineBackendCreate()); + } +#endif mDebuggerAttach(debugger, renderer->core); mDebuggerEnter(debugger, DEBUGGER_ENTER_MANUAL, NULL); }