HatariWii/src/debug/debugcpu.c
2018-05-25 20:45:09 +02:00

789 lines
18 KiB
C

/*
Hatari - debugcpu.c
This file is distributed under the GNU General Public License, version 2
or at your option any later version. Read the file gpl.txt for details.
debugcpu.c - function needed for the CPU debugging tasks like memory
and register dumps.
*/
const char DebugCpu_fileid[] = "Hatari debugcpu.c : " __DATE__ " " __TIME__;
#include <stdio.h>
#include <ctype.h>
#include "config.h"
#include "main.h"
#include "breakcond.h"
#include "configuration.h"
#include "debugui.h"
#include "debug_priv.h"
#include "debugcpu.h"
#include "evaluate.h"
#include "hatari-glue.h"
#include "history.h"
#include "log.h"
#include "m68000.h"
#include "memorySnapShot.h"
#include "profile.h"
#include "stMemory.h"
#include "str.h"
#include "symbols.h"
#include "68kDisass.h"
#include "console.h"
#include "options.h"
#define MEMDUMP_COLS 16 /* memdump, number of bytes per row */
#define NON_PRINT_CHAR '.' /* character to display for non-printables */
static Uint32 disasm_addr; /* disasm address */
static Uint32 memdump_addr; /* memdump address */
static bool bCpuProfiling; /* Whether CPU profiling is activated */
static int nCpuActiveCBs = 0; /* Amount of active conditional breakpoints */
static int nCpuSteps = 0; /* Amount of steps for CPU single-stepping */
/**
* Load a binary file to a memory address.
*/
static int DebugCpu_LoadBin(int nArgc, char *psArgs[])
{
FILE *fp;
unsigned char c;
Uint32 address;
int i=0;
if (nArgc < 3)
{
return DebugUI_PrintCmdHelp(psArgs[0]);
}
if (!Eval_Number(psArgs[2], &address))
{
fprintf(stderr, "Invalid address!\n");
return DEBUGGER_CMDDONE;
}
if ((fp = fopen(psArgs[1], "rb")) == NULL)
{
fprintf(stderr, "Cannot open file '%s'!\n", psArgs[1]);
return DEBUGGER_CMDDONE;
}
/* TODO: more efficient would be to:
* - check file size
* - verify that it fits into valid memory area
* - flush emulated CPU data cache
* - read file contents directly into memory
*/
c = fgetc(fp);
while (!feof(fp))
{
i++;
STMemory_WriteByte(address++, c);
c = fgetc(fp);
}
fprintf(stderr," Read 0x%x bytes.\n", i);
fclose(fp);
return DEBUGGER_CMDDONE;
}
/**
* Dump memory from an address to a binary file.
*/
static int DebugCpu_SaveBin(int nArgc, char *psArgs[])
{
FILE *fp;
unsigned char c;
Uint32 address;
Uint32 bytes, i = 0;
if (nArgc < 4)
{
return DebugUI_PrintCmdHelp(psArgs[0]);
}
if (!Eval_Number(psArgs[2], &address))
{
fprintf(stderr, " Invalid address!\n");
return DEBUGGER_CMDDONE;
}
if (!Eval_Number(psArgs[3], &bytes))
{
fprintf(stderr, " Invalid length!\n");
return DEBUGGER_CMDDONE;
}
if ((fp = fopen(psArgs[1], "wb")) == NULL)
{
fprintf(stderr," Cannot open file '%s'!\n", psArgs[1]);
return DEBUGGER_CMDDONE;
}
while (i < bytes)
{
c = STMemory_ReadByte(address++);
fputc(c, fp);
i++;
}
fclose(fp);
fprintf(stderr, " Wrote 0x%x bytes.\n", bytes);
return DEBUGGER_CMDDONE;
}
/**
* Check whether given address matches any CPU symbol and whether
* there's profiling information available for it. If yes, show it.
*/
static void DebugCpu_ShowAddressInfo(Uint32 addr)
{
const char *symbol = Symbols_GetByCpuAddress(addr);
if (symbol)
fprintf(debugOutput, "%s:\n", symbol);
}
/**
* Dissassemble - arg = starting address, or PC.
*/
int DebugCpu_DisAsm(int nArgc, char *psArgs[])
{
Uint32 disasm_upper = 0;
int insts, max_insts;
uaecptr nextpc;
if (nArgc > 1)
{
switch (Eval_Range(psArgs[1], &disasm_addr, &disasm_upper, false))
{
case -1:
/* invalid value(s) */
return DEBUGGER_CMDDONE;
case 0:
/* single value */
break;
case 1:
/* range */
break;
}
}
else
{
/* continue */
if(!disasm_addr)
disasm_addr = M68000_GetPC();
}
/* limit is topmost address or instruction count */
if (disasm_upper)
{
max_insts = INT_MAX;
}
else
{
disasm_upper = 0xFFFFFFFF;
max_insts = ConfigureParams.Debugger.nDisasmLines;
}
/* output a range */
for (insts = 0; insts < max_insts && disasm_addr < disasm_upper; insts++)
{
DebugCpu_ShowAddressInfo(disasm_addr);
Disasm(debugOutput, (uaecptr)disasm_addr, &nextpc, 1);
disasm_addr = nextpc;
}
fflush(debugOutput);
return DEBUGGER_CMDCONT;
}
/**
* Readline match callback to list register names usable within debugger.
* STATE = 0 -> different text from previous one.
* Return next match or NULL if no matches.
*/
static char *DebugCpu_MatchRegister(const char *text, int state)
{
static const char* regs[] = {
"a0", "a1", "a2", "a3", "a4", "a5", "a6", "a7",
"d0", "d1", "d2", "d3", "d4", "d5", "d6", "d7",
"pc", "sr"
};
return DebugUI_MatchHelper(regs, ARRAYSIZE(regs), text, state);
}
/**
* Set address of the named 32-bit register to given argument.
* Return register size in bits or zero for uknown register name.
* Handles D0-7 data and A0-7 address registers, but not PC & SR
* registers as they need to be accessed using UAE accessors.
*/
int DebugCpu_GetRegisterAddress(const char *reg, Uint32 **addr)
{
char r0, r1;
if (!reg[0] || !reg[1] || reg[2])
return 0;
r0 = toupper((unsigned char)reg[0]);
r1 = toupper((unsigned char)reg[1]);
if (r0 == 'D') /* Data regs? */
{
if (r1 >= '0' && r1 <= '7')
{
*addr = &(Regs[REG_D0 + r1 - '0']);
return 32;
}
fprintf(stderr,"\tBad data register, valid values are 0-7\n");
return 0;
}
if(r0 == 'A') /* Address regs? */
{
if (r1 >= '0' && r1 <= '7')
{
*addr = &(Regs[REG_A0 + r1 - '0']);
return 32;
}
fprintf(stderr,"\tBad address register, valid values are 0-7\n");
return 0;
}
return 0;
}
/**
* Dump or set CPU registers
*/
int DebugCpu_Register(int nArgc, char *psArgs[])
{
char reg[3], *assign;
Uint32 value;
char *arg;
/* If no parameter has been given, simply dump all registers */
if (nArgc == 1)
{
uaecptr nextpc;
/* use the UAE function instead */
#ifdef WINUAE_FOR_HATARI
m68k_dumpstate_file(debugOutput, &nextpc);
#else
m68k_dumpstate(debugOutput, &nextpc);
#endif
fflush(debugOutput);
return DEBUGGER_CMDDONE;
}
arg = psArgs[1];
assign = strchr(arg, '=');
if (!assign)
{
goto error_msg;
}
*assign++ = '\0';
if (!Eval_Number(Str_Trim(assign), &value))
{
goto error_msg;
}
arg = Str_Trim(arg);
if (strlen(arg) != 2)
{
goto error_msg;
}
reg[0] = toupper((unsigned char)arg[0]);
reg[1] = toupper((unsigned char)arg[1]);
reg[2] = '\0';
/* set SR and update conditional flags for the UAE CPU core. */
if (reg[0] == 'S' && reg[1] == 'R')
{
M68000_SetSR(value);
}
else if (reg[0] == 'P' && reg[1] == 'C') /* set PC? */
{
M68000_SetPC(value);
}
else
{
Uint32 *regaddr;
/* check&set data and address registers */
if (DebugCpu_GetRegisterAddress(reg, &regaddr))
{
*regaddr = value;
}
else
{
goto error_msg;
}
}
return DEBUGGER_CMDDONE;
error_msg:
fprintf(stderr,"\tError, usage: r or r xx=yyyy\n\tWhere: xx=A0-A7, D0-D7, PC or SR.\n");
return DEBUGGER_CMDDONE;
}
/**
* CPU wrapper for BreakAddr_Command().
*/
static int DebugCpu_BreakAddr(int nArgc, char *psArgs[])
{
BreakAddr_Command(psArgs[1], false);
return DEBUGGER_CMDDONE;
}
/**
* CPU wrapper for BreakCond_Command().
*/
static int DebugCpu_BreakCond(int nArgc, char *psArgs[])
{
BreakCond_Command(psArgs[1], false);
return DEBUGGER_CMDDONE;
}
/**
* CPU wrapper for Profile_Command().
*/
static int DebugCpu_Profile(int nArgc, char *psArgs[])
{
return Profile_Command(nArgc, psArgs, false);
}
/**
* Do a memory dump, args = starting address.
*/
int DebugCpu_MemDump(int nArgc, char *psArgs[])
{
int i;
char c;
Uint32 memdump_upper = 0;
if (nArgc > 1)
{
switch (Eval_Range(psArgs[1], &memdump_addr, &memdump_upper, false))
{
case -1:
/* invalid value(s) */
return DEBUGGER_CMDDONE;
case 0:
/* single value */
break;
case 1:
/* range */
break;
}
} /* continue */
if (!memdump_upper)
{
memdump_upper = memdump_addr + MEMDUMP_COLS * ConfigureParams.Debugger.nMemdumpLines;
}
while (memdump_addr < memdump_upper)
{
fprintf(debugOutput, "%8.8X: ", memdump_addr); /* print address */
for (i = 0; i < MEMDUMP_COLS; i++) /* print hex data */
fprintf(debugOutput, "%2.2x ", STMemory_ReadByte(memdump_addr++));
fprintf(debugOutput, " "); /* print ASCII data */
for (i = 0; i < MEMDUMP_COLS; i++)
{
c = STMemory_ReadByte(memdump_addr-MEMDUMP_COLS+i);
if(!isprint((unsigned)c))
c = NON_PRINT_CHAR; /* non-printable as dots */
fprintf(debugOutput,"%c", c);
}
fprintf(debugOutput, "\n"); /* newline */
} /* while */
fflush(debugOutput);
return DEBUGGER_CMDCONT;
}
/**
* Command: Write to memory, arg = starting address, followed by bytes.
*/
static int DebugCpu_MemWrite(int nArgc, char *psArgs[])
{
int i, numBytes;
Uint32 write_addr, d;
unsigned char bytes[256]; /* store bytes */
if (nArgc < 3)
{
return DebugUI_PrintCmdHelp(psArgs[0]);
}
/* Read address */
if (!Eval_Number(psArgs[1], &write_addr))
{
fprintf(stderr, "Bad address!\n");
return DEBUGGER_CMDDONE;
}
numBytes = 0;
/* get bytes data */
for (i = 2; i < nArgc; i++)
{
if (!Eval_Number(psArgs[i], &d) || d > 255)
{
fprintf(stderr, "Bad byte argument: '%s'!\n", psArgs[i]);
return DEBUGGER_CMDDONE;
}
bytes[numBytes] = d & 0x0FF;
numBytes++;
}
/* write the data */
for (i = 0; i < numBytes; i++)
STMemory_WriteByte(write_addr + i, bytes[i]);
return DEBUGGER_CMDDONE;
}
/**
* Command: Continue CPU emulation / single-stepping
*/
static int DebugCpu_Continue(int nArgc, char *psArgv[])
{
int steps = 0;
if (nArgc > 1)
{
steps = atoi(psArgv[1]);
}
if (steps <= 0)
{
nCpuSteps = 0;
fprintf(stderr,"Returning to emulation...\n");
return DEBUGGER_END;
}
nCpuSteps = steps;
fprintf(stderr,"Returning to emulation for %i CPU instructions...\n", steps);
return DEBUGGER_END;
}
/**
* Command: Single-step CPU
*/
static int DebugCpu_Step(int nArgc, char *psArgv[])
{
nCpuSteps = 1;
return DEBUGGER_END;
}
/**
* Readline match callback to list next command opcode types.
* STATE = 0 -> different text from previous one.
* Return next match or NULL if no matches.
*/
static char *DebugCpu_MatchNext(const char *text, int state)
{
static const char* ntypes[] = {
"branch", "exception", "exreturn", "return", "subcall", "subreturn"
};
return DebugUI_MatchHelper(ntypes, ARRAYSIZE(ntypes), text, state);
}
/**
* Command: Step CPU, but proceed through subroutines
* Does this by temporary conditional breakpoint
*/
static int DebugCpu_Next(int nArgc, char *psArgv[])
{
char command[40];
if (nArgc > 1)
{
int optype;
if(strcmp(psArgv[1], "branch") == 0)
optype = CALL_BRANCH;
else if(strcmp(psArgv[1], "exception") == 0)
optype = CALL_EXCEPTION;
else if(strcmp(psArgv[1], "exreturn") == 0)
optype = CALL_EXCRETURN;
else if(strcmp(psArgv[1], "subcall") == 0)
optype = CALL_SUBROUTINE;
else if (strcmp(psArgv[1], "subreturn") == 0)
optype = CALL_SUBRETURN;
else if (strcmp(psArgv[1], "return") == 0)
optype = CALL_SUBRETURN | CALL_EXCRETURN;
else
{
fprintf(stderr, "Unrecognized opcode type given!\n");
return DEBUGGER_CMDDONE;
}
sprintf(command, "CpuOpcodeType & $%x > 0 :once :quiet\n", optype);
}
else
{
Uint32 optype, nextpc;
optype = DebugCpu_OpcodeType();
/* can this instruction be stepped normally? */
if (optype != CALL_SUBROUTINE && optype != CALL_EXCEPTION)
{
nCpuSteps = 1;
return DEBUGGER_END;
}
nextpc = Disasm_GetNextPC(M68000_GetPC());
sprintf(command, "pc=$%x :once :quiet\n", nextpc);
}
/* use breakpoint, not steps */
if (BreakCond_Command(command, false))
{
nCpuSteps = 0;
return DEBUGGER_END;
}
return DEBUGGER_CMDDONE;
}
/* helper to get instruction type */
Uint32 DebugCpu_OpcodeType(void)
{
/* cannot use OpcodeFamily like profiler does,
* as that's for previous instructions
*/
Uint16 opcode = STMemory_ReadWord(M68000_GetPC());
if (opcode == 0x4e74 || /* RTD */
opcode == 0x4e75 || /* RTS */
opcode == 0x4e77) /* RTR */
return CALL_SUBRETURN;
if (opcode == 0x4e73) /* RTE */
return CALL_EXCRETURN;
/* NOTE: BSR needs to be matched before BRA/BCC! */
if ((opcode & 0xff00) == 0x6100 || /* BSR */
(opcode & 0xffc0) == 0x4e80) /* JSR */
return CALL_SUBROUTINE;
/* TODO: ftrapcc, chk2? */
if (opcode == 0x4e72 || /* STOP */
opcode == 0x4afc || /* ILLEGAL */
opcode == 0x4e76 || /* TRAPV */
(opcode & 0xfff0) == 0x4e40 || /* TRAP */
(opcode & 0xf1c0) == 0x4180 || /* CHK */
(opcode & 0xfff8) == 0x4848) /* BKPT */
return CALL_EXCEPTION;
/* TODO: fbcc, fdbcc */
if ((opcode & 0xf000) == 0x6000 || /* BRA / BCC */
(opcode & 0xffc0) == 0x4ec0 || /* JMP */
(opcode & 0xf0f8) == 0x50c8) /* DBCC */
return CALL_BRANCH;
return CALL_UNKNOWN;
}
/**
* CPU instructions since continuing emulation
*/
static Uint32 nCpuInstructions;
Uint32 DebugCpu_InstrCount(void)
{
return nCpuInstructions;
}
/**
* This function is called after each CPU instruction when debugging is enabled.
*/
void DebugCpu_Check(void)
{
nCpuInstructions++;
if (bCpuProfiling)
{
Profile_CpuUpdate();
}
if (LOG_TRACE_LEVEL((TRACE_CPU_DISASM|TRACE_CPU_SYMBOLS)))
{
DebugCpu_ShowAddressInfo(M68000_GetPC());
}
if (nCpuActiveCBs)
{
if (BreakCond_MatchCpu())
{
DebugUI(REASON_CPU_BREAKPOINT);
/* make sure we don't decrease step count
* below, before even even getting out of here
*/
if (nCpuSteps)
nCpuSteps++;
}
}
if (nCpuSteps)
{
nCpuSteps--;
if (nCpuSteps == 0)
DebugUI(REASON_CPU_STEPS);
}
if (History_TrackCpu())
{
History_AddCpu();
}
if (ConOutDevice != CONOUT_DEVICE_NONE)
{
Console_Check();
}
}
/**
* Should be called before returning back emulation to tell the CPU core
* to call us after each instruction if "real-time" debugging like
* breakpoints has been set.
*/
void DebugCpu_SetDebugging(void)
{
bCpuProfiling = Profile_CpuStart();
nCpuActiveCBs = BreakCond_CpuBreakPointCount();
if (nCpuActiveCBs || nCpuSteps || bCpuProfiling || History_TrackCpu()
|| LOG_TRACE_LEVEL((TRACE_CPU_DISASM|TRACE_CPU_SYMBOLS))
|| ConOutDevice != CONOUT_DEVICE_NONE)
{
M68000_SetSpecial(SPCFLAG_DEBUGGER);
nCpuInstructions = 0;
}
else
M68000_UnsetSpecial(SPCFLAG_DEBUGGER);
}
static const dbgcommand_t cpucommands[] =
{
{ NULL, NULL, "CPU commands", NULL, NULL, NULL, false },
/* NULL as match function will complete file names */
{ DebugCpu_BreakAddr, Symbols_MatchCpuCodeAddress,
"address", "a",
"set CPU PC address breakpoints",
BreakAddr_Description,
true },
{ DebugCpu_BreakCond, BreakCond_MatchCpuVariable,
"breakpoint", "b",
"set/remove/list conditional CPU breakpoints",
BreakCond_Description,
true },
{ DebugCpu_DisAsm, Symbols_MatchCpuCodeAddress,
"disasm", "d",
"disassemble from PC, or given address",
"[<start address>[-<end address>]]\n"
"\tIf no address is given, this command disassembles from the last\n"
"\tposition or from current PC if no last position is available.",
false },
{ DebugCpu_Profile, Profile_Match,
"profile", "",
"profile CPU code",
Profile_Description,
false },
{ DebugCpu_Register, DebugCpu_MatchRegister,
"cpureg", "r",
"dump register values or set register to value",
"[REG=value]\n"
"\tSet CPU register to value or dumps all register if no parameter\n"
"\thas been specified.",
true },
{ DebugCpu_MemDump, Symbols_MatchCpuDataAddress,
"memdump", "m",
"dump memory",
"[<start address>[-<end address>]]\n"
"\tdump memory at address or continue dump from previous address.",
false },
{ DebugCpu_MemWrite, Symbols_MatchCpuAddress,
"memwrite", "w",
"write bytes to memory",
"address byte1 [byte2 ...]\n"
"\tWrite bytes to a memory address, bytes are space separated\n"
"\tvalues in current number base.",
false },
{ DebugCpu_LoadBin, NULL,
"loadbin", "l",
"load a file into memory",
"filename address\n"
"\tLoad the file <filename> into memory starting at <address>.",
false },
{ DebugCpu_SaveBin, NULL,
"savebin", "",
"save memory to a file",
"filename address length\n"
"\tSave the memory block at <address> with given <length> to\n"
"\tthe file <filename>.",
false },
{ Symbols_Command, Symbols_MatchCommand,
"symbols", "",
"load CPU symbols & their addresses",
Symbols_Description,
false },
{ DebugCpu_Step, NULL,
"step", "s",
"single-step CPU",
"\n"
"\tExecute next CPU instruction (equals 'c 1')",
false },
{ DebugCpu_Next, DebugCpu_MatchNext,
"next", "n",
"step CPU through subroutine calls / to given instruction type",
"[instruction type]\n"
"\tSame as 'step' command if there are no subroutine calls.\n"
"\tWhen there are, those calls are treated as one instruction.\n"
"\tIf argument is given, continues until instruction of given\n"
"\ttype is encountered.",
false },
{ DebugCpu_Continue, NULL,
"cont", "c",
"continue emulation / CPU single-stepping",
"[steps]\n"
"\tLeave debugger and continue emulation for <steps> CPU instructions\n"
"\tor forever if no steps have been specified.",
false }
};
/**
* Should be called when debugger is first entered to initialize
* CPU debugging variables.
*
* if you want disassembly or memdumping to start/continue from
* specific address, you can set them here. If disassembly
* address is zero, disassembling starts from PC.
*
* returns number of CPU commands and pointer to array of them.
*/
int DebugCpu_Init(const dbgcommand_t **table)
{
memdump_addr = 0;
disasm_addr = 0;
*table = cpucommands;
return ARRAYSIZE(cpucommands);
}
/**
* Should be called when debugger is re-entered to reset
* relevant CPU debugging variables.
*/
void DebugCpu_InitSession(void)
{
disasm_addr = M68000_GetPC();
Profile_CpuStop();
}