--- hatari/src/debug/breakcond.c 2019/04/09 08:48:37 1.1 +++ hatari/src/debug/breakcond.c 2019/04/09 08:49:25 1.1.1.2 @@ -17,18 +17,23 @@ const char BreakCond_fileid[] = "Hatari #include #include "config.h" #include "main.h" +#include "file.h" #include "m68000.h" #include "memorySnapShot.h" #include "dsp.h" #include "stMemory.h" #include "str.h" +#include "screen.h" /* for defines needed by video.h */ #include "video.h" /* for Hatari video variable addresses */ #include "debug_priv.h" #include "breakcond.h" #include "debugcpu.h" +#include "debugInfo.h" +#include "debugui.h" #include "evaluate.h" #include "symbols.h" +#include "68kDisass.h" /* set to 1 to enable parsing function tracing / debug output */ @@ -42,7 +47,7 @@ const char BreakCond_fileid[] = "Hatari #define BC_DEFAULT_DSP_SPACE 'P' -enum { +typedef enum { /* plain number */ VALUE_TYPE_NUMBER = 0, @@ -55,7 +60,7 @@ enum { /* size must match register size used in BreakCond_ParseRegister() */ VALUE_TYPE_REG16 = 16, VALUE_TYPE_REG32 = 32 -} typedef value_t; +} value_t; static inline bool is_register_type(value_t vtype) { /* type used for CPU/DSP registers */ @@ -85,13 +90,19 @@ typedef struct { } bc_condition_t; typedef struct { + char *filename; /* file where to read commands to do on hit */ + int skip; /* how many times to hit before breaking */ + bool once; /* remove after hit&break */ + bool trace; /* trace mode, don't break */ + bool lock; /* tracing + show locked info */ +} bc_options_t; + +typedef struct { char *expression; + bc_options_t options; bc_condition_t conditions[BC_MAX_CONDITIONS_PER_BREAKPOINT]; int ccount; /* condition count */ int hits; /* how many times breakpoint hit */ - int skip; /* how many times to hit before breaking */ - bool once; /* remove after hit&break */ - bool trace; /* trace mode, don't break */ } bc_breakpoint_t; static bc_breakpoint_t BreakPointsCpu[BC_MAX_CONDITION_BREAKPOINTS]; @@ -115,9 +126,11 @@ bool BreakCond_Save(const char *filename int i; if (!(BreakPointCpuCount || BreakPointDspCount)) { - if (remove(filename)) { - perror("ERROR"); - return false; + if (File_Exists(filename)) { + if (remove(filename)) { + perror("ERROR"); + return false; + } } return true; } @@ -316,23 +329,37 @@ static int BreakCond_MatchBreakPoints(bc int i; for (i = 0; i < count; bp++, i++) { + if (BreakCond_MatchConditions(bp->conditions, bp->ccount)) { + BreakCond_ShowTracked(bp->conditions, bp->ccount); + bp->hits++; - if (bp->skip && (bp->hits % bp->skip) == 0) { + if (bp->options.skip && + (bp->hits % bp->options.skip) == 0) { return 0; } fprintf(stderr, "%d. %s breakpoint condition(s) matched %d times.\n", i+1, name, bp->hits); - if (bp->trace) { - return 0; - } + BreakCond_Print(bp); - if (bp->once) { + if (bp->options.once) { BreakCond_Remove(i+1, (bp-i == BreakPointsDsp)); } - /* indexes for BreakCond_Remove() start from 1 */ - return i + 1; + if (bp->options.lock) { + DebugCpu_InitSession(); + DebugDsp_InitSession(); + DebugInfo_ShowSessionInfo(); + } + if (bp->options.filename) { + DebugUI_ParseFile(bp->options.filename); + } + if (bp->options.trace) { + return 0; + } else { + /* indexes for BreakCond_Remove() start from 1 */ + return i + 1; + } } } return 0; @@ -403,12 +430,112 @@ static Uint32 GetFrameCycles(void) return fcycles; } +/* helpers for TOS OS call opcode accessor functions */ +#define INVALID_OPCODE 0xFFFFu + +static inline Uint16 getLineOpcode(Uint8 line) +{ + Uint32 pc; + Uint16 instr; + pc = M68000_GetPC(); + instr = STMemory_ReadWord(pc); + /* for opcode X, Line-A = 0xA00X, Line-F = 0xF00X */ + if ((instr >> 12) == line) { + return instr & 0xFF; + } + return INVALID_OPCODE; +} +static inline bool isTrap(Uint8 trap) +{ + Uint32 pc; + Uint16 instr; + pc = M68000_GetPC(); + instr = STMemory_ReadWord(pc); + return (instr == (Uint16)0x4e40u + trap); +} +static inline Uint16 getControlOpcode(void) +{ + /* Control[] address from D1, opcode in Control[0] */ + return STMemory_ReadWord(STMemory_ReadLong(Regs[REG_D1])); +} +static inline Uint16 getStackOpcode(void) +{ + return STMemory_ReadWord(Regs[REG_A7]); +} + +/* Actual TOS OS call opcode accessor functions */ +static Uint32 GetLineAOpcode(void) +{ + return getLineOpcode(0xA); +} +static Uint32 GetLineFOpcode(void) +{ + return getLineOpcode(0xF); +} +static Uint32 GetGemdosOpcode(void) +{ + if (isTrap(1)) { + return getStackOpcode(); + } + return INVALID_OPCODE; +} +static Uint32 GetBiosOpcode(void) +{ + if (isTrap(13)) { + return getStackOpcode(); + } + return INVALID_OPCODE; +} +static Uint32 GetXbiosOpcode(void) +{ + if (isTrap(14)) { + return getStackOpcode(); + } + return INVALID_OPCODE; +} +static Uint32 GetAesOpcode(void) +{ + if (isTrap(2)) { + Uint16 d0 = Regs[REG_D0]; + if (d0 == 0xC8) { + return getControlOpcode(); + } else if (d0 == 0xC9) { + /* same as appl_yield() */ + return 0x11; + } + } + return INVALID_OPCODE; +} +static Uint32 GetVdiOpcode(void) +{ + if (isTrap(2)) { + Uint16 d0 = Regs[REG_D0]; + if (d0 == 0x73) { + return getControlOpcode(); + } else if (d0 == 0xFFFE) { + /* -2 = vq_[v]gdos() */ + return 0xFFFE; + } + } + return INVALID_OPCODE; +} + /* sorted by variable name so that this can be bisected */ static const var_addr_t hatari_vars[] = { + { "AesOpcode", (Uint32*)GetAesOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" }, + { "BiosOpcode", (Uint32*)GetBiosOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" }, + { "BSS", (Uint32*)DebugInfo_GetBSS, VALUE_TYPE_FUNCTION32, 0, "invalid before Desktop is up" }, + { "DATA", (Uint32*)DebugInfo_GetDATA, VALUE_TYPE_FUNCTION32, 0, "invalid before Desktop is up" }, { "FrameCycles", (Uint32*)GetFrameCycles, VALUE_TYPE_FUNCTION32, 0, NULL }, + { "GemdosOpcode", (Uint32*)GetGemdosOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" }, { "HBL", (Uint32*)&nHBL, VALUE_TYPE_VAR32, sizeof(nHBL)*8, NULL }, + { "LineAOpcode", (Uint32*)GetLineAOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" }, { "LineCycles", (Uint32*)GetLineCycles, VALUE_TYPE_FUNCTION32, 0, "is always divisable by 4" }, - { "VBL", (Uint32*)&nVBLs, VALUE_TYPE_VAR32, sizeof(nVBLs)*8, NULL } + { "LineFOpcode", (Uint32*)GetLineFOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" }, + { "TEXT", (Uint32*)DebugInfo_GetTEXT, VALUE_TYPE_FUNCTION32, 0, "invalid before Desktop is up" }, + { "VBL", (Uint32*)&nVBLs, VALUE_TYPE_VAR32, sizeof(nVBLs)*8, NULL }, + { "VdiOpcode", (Uint32*)GetVdiOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" }, + { "XbiosOpcode", (Uint32*)GetXbiosOpcode, VALUE_TYPE_FUNCTION32, 16, "by default FFFF" } }; @@ -483,6 +610,22 @@ static bool BreakCond_ParseVariable(cons return false; } +/** + * If given string is a Hatari variable name, set value to given + * variable value and return true, otherwise return false. + */ +bool BreakCond_GetHatariVariable(const char *name, Uint32 *value) +{ + bc_value_t bc_value; + if (!BreakCond_ParseVariable(name, &bc_value)) { + return false; + } + bc_value.mask = 0xffffffff; + bc_value.is_indirect = false; + *value = BreakCond_GetValue(&bc_value); + return true; +} + /** * If given string matches a suitable symbol, set bc_value @@ -1202,7 +1345,7 @@ static void BreakCond_CheckTracking(bc_b * Parse given breakpoint expression and store it. * Return true for success and false for failure. */ -static bool BreakCond_Parse(const char *expression, bool bForDsp, bool trace, bool once, int skip) +static bool BreakCond_Parse(const char *expression, bc_options_t *options, bool bForDsp) { parser_state_t pstate; bc_breakpoint_t *bp; @@ -1236,17 +1379,25 @@ static bool BreakCond_Parse(const char * fprintf(stderr, "%s condition breakpoint %d with %d condition(s) added:\n\t%s\n", name, *bcount, ccount, bp->expression); BreakCond_CheckTracking(bp); - if (skip) { - fprintf(stderr, "-> Break only on every %d hit.\n", skip); - bp->skip = skip; + if (options->skip) { + fprintf(stderr, "-> Break only on every %d hit.\n", options->skip); + bp->options.skip = options->skip; } - if (once) { + if (options->once) { fprintf(stderr, "-> Once, delete after breaking.\n"); - bp->once = once; + bp->options.once = true; } - if (trace) { + if (options->trace) { fprintf(stderr, "-> Trace instead of breaking, but show still hits.\n"); - bp->trace = trace; + if (options->lock) { + fprintf(stderr, "-> Show also info selected with lock command.\n"); + bp->options.lock = true; + } + bp->options.trace = true; + } + if (options->filename) { + fprintf(stderr, "-> Execute debugger commands from '%s' file on hit.\n", options->filename); + bp->options.filename = strdup(options->filename); } } else { if (normalized) { @@ -1283,14 +1434,21 @@ static bool BreakCond_Parse(const char * static void BreakCond_Print(bc_breakpoint_t *bp) { fprintf(stderr, "\t%s", bp->expression); - if (bp->skip) { - fprintf(stderr, " :%d", bp->skip); + if (bp->options.skip) { + fprintf(stderr, " :%d", bp->options.skip); } - if (bp->once) { + if (bp->options.once) { fprintf(stderr, " :once"); } - if (bp->trace) { - fprintf(stderr, " :trace"); + if (bp->options.trace) { + if (bp->options.lock) { + fprintf(stderr, " :lock"); + } else { + fprintf(stderr, " :trace"); + } + } + if (bp->options.filename) { + fprintf(stderr, " :file %s", bp->options.filename); } fprintf(stderr, "\n"); } @@ -1326,10 +1484,10 @@ static bool BreakCond_Remove(int positio const char *name; bc_breakpoint_t *bp; int *bcount, offset; - + bcount = BreakCond_GetListInfo(&bp, &name, bForDsp); if (!*bcount) { - fprintf(stderr, "No (more) breakpoints to remove.\n"); + fprintf(stderr, "No (more) %s breakpoints to remove.\n", name); return false; } if (position < 1 || position > *bcount) { @@ -1340,6 +1498,9 @@ static bool BreakCond_Remove(int positio fprintf(stderr, "Removed %s breakpoint %d:\n", name, position); BreakCond_Print(&(bp[offset])); free(bp[offset].expression); + if (bp[offset].options.filename) { + free(bp[offset].options.filename); + } bp[offset].expression = NULL; if (position < *bcount) { @@ -1385,26 +1546,23 @@ static void BreakCond_Help(void) Uint32 value; int i; fputs( -" breakpoint = [ && ... ] [option]\n" -" condition = [.mode] [& ] [.mode]\n" +" condition = [.mode] [& ] [.mode]\n" "\n" " where:\n" " value = [(] [)]\n" -" number = [#|$|%]\n" +" number/mask = [#|$|%]\n" " comparison = '<' | '>' | '=' | '!'\n" " addressing mode (width) = 'b' | 'w' | 'l'\n" " addressing mode (space) = 'p' | 'x' | 'y'\n" -" option = : | 'once' | 'trace'\n" "\n" " If the value is in parenthesis like in '($ff820)' or '(a0)', then\n" " the used value will be read from the memory address pointed by it.\n" "\n" -" If the value expressions on both sides of the comparison are exactly\n" -" the same, right side is replaced with its current value and for\n" -" inequality ('!') comparison, the breakpoint tracks all further changes\n" -" for the given address/register expression. 'trace' option for continuing\n" -" without breaking can be useful with this. 'once' option removes breakpoint\n" -" after hit and giving count as option will break only on every hit.\n" +" If the parsed value expressions on both sides of it are exactly\n" +" the same, right side is replaced with its current value. For\n" +" inequality ('!') comparison, the breakpoint will additionally track\n" +" all further changes for the given address/register expression value.\n" +" (This is useful for tracking register and memory value changes.)\n" "\n" " M68k addresses can have byte (b), word (w) or long (l, default) width.\n" " DSP addresses belong to different address spaces: P, X or Y. Note that\n" @@ -1424,7 +1582,7 @@ static void BreakCond_Help(void) hatari_vars[i].name, hatari_vars[i].vtype); continue; } - fprintf(stderr, " - %s (%d)", hatari_vars[i].name, value); + fprintf(stderr, " - %s ($%x)", hatari_vars[i].name, value); if (hatari_vars[i].constraints) { fprintf(stderr, ", %s\n", hatari_vars[i].constraints); } else { @@ -1443,14 +1601,77 @@ static void BreakCond_Help(void) /* ------------- breakpoint condition parsing, public API ------------ */ const char BreakCond_Description[] = - "[ [:|once|trace] | | help | all ]\n" - "\tSet breakpoint with given , remove breakpoint with\n" - "\tgiven or list all breakpoints when no args are given.\n" - "\tAdding ':trace' to end of condition causes breakpoint match\n" - "\tjust to be printed, not break. Adding ':once' will delete\n" - "\tthe breakpoint after it's hit. Adding ':' will break\n" - "\tonly on every hit. 'help' outputs breakpoint condition\n" - "\tsyntax help, 'all' removes all breakpoints."; + " [&& ...] [: