mirror of
https://github.com/ksherlock/mpw.git
synced 2025-01-25 09:30:40 +00:00
646 lines
11 KiB
C++
646 lines
11 KiB
C++
|
|
#include <cstdint>
|
|
#include <cctype>
|
|
#include <cstring>
|
|
#include <cstdlib>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
#include <array>
|
|
#include <unordered_set>
|
|
#include <unordered_map>
|
|
|
|
#include <bitset>
|
|
|
|
#include <readline/readline.h>
|
|
|
|
|
|
#include "loader.h"
|
|
#include "address_map.h"
|
|
|
|
#include <debugger/commands.h>
|
|
|
|
#include <cpu/defs.h>
|
|
#include <cpu/CpuModule.h>
|
|
|
|
#include <macos/traps.h>
|
|
#include <macos/sysequ.h>
|
|
|
|
bool ParseLine(const char *iter, Command *command);
|
|
|
|
extern "C" {
|
|
|
|
uint32_t debuggerReadLong(uint32_t address)
|
|
{
|
|
uint32_t tmp = 0;
|
|
for (unsigned i = 0; i < 4; ++i)
|
|
{
|
|
if (address < Flags.memorySize)
|
|
tmp = (tmp << 8) + Flags.memory[address++];
|
|
}
|
|
|
|
return tmp;
|
|
}
|
|
|
|
|
|
uint16_t debuggerReadWord(uint32_t address)
|
|
{
|
|
uint16_t tmp = 0;
|
|
for (unsigned i = 0; i < 2; ++i)
|
|
{
|
|
if (address < Flags.memorySize)
|
|
tmp = (tmp << 8) + Flags.memory[address++];
|
|
}
|
|
|
|
return tmp;
|
|
}
|
|
|
|
uint8_t debuggerReadByte(uint32_t address)
|
|
{
|
|
if (address < Flags.memorySize)
|
|
return Flags.memory[address];
|
|
|
|
return 0;
|
|
}
|
|
|
|
}
|
|
|
|
namespace {
|
|
|
|
bool sigInt = false;
|
|
void sigIntHandler(int)
|
|
{
|
|
sigInt = true;
|
|
}
|
|
|
|
AddressMap brkMap; // address breaks
|
|
AddressMap rbrkMap; // read breaks.
|
|
AddressMap wbrkMap; // write breaks.
|
|
ToolMap tbrkMap; // tool breaks.
|
|
|
|
|
|
|
|
void hexdump(const uint8_t *data, ssize_t size, uint32_t address = 0)
|
|
{
|
|
const char *HexMap = "0123456789abcdef";
|
|
|
|
char buffer1[16 * 3 + 1 + 1];
|
|
char buffer2[16 + 1];
|
|
ssize_t offset = 0;
|
|
unsigned i, j;
|
|
|
|
|
|
while(size > 0)
|
|
{
|
|
std::memset(buffer1, ' ', sizeof(buffer1));
|
|
std::memset(buffer2, ' ', sizeof(buffer2));
|
|
|
|
unsigned linelen = (unsigned)std::min(size, (ssize_t)16);
|
|
|
|
|
|
for (i = 0, j = 0; i < linelen; i++)
|
|
{
|
|
unsigned x = data[i];
|
|
buffer1[j++] = HexMap[x >> 4];
|
|
buffer1[j++] = HexMap[x & 0x0f];
|
|
j++;
|
|
if (i == 7) j++;
|
|
|
|
// isascii not part of std:: and may be a macro.
|
|
buffer2[i] = isascii(x) && std::isprint(x) ? x : '.';
|
|
|
|
}
|
|
|
|
buffer1[sizeof(buffer1)-1] = 0;
|
|
buffer2[sizeof(buffer2)-1] = 0;
|
|
|
|
|
|
std::printf("%08x:\t%s\t%s\n", address + (unsigned)offset, buffer1, buffer2);
|
|
offset += 16;
|
|
data += 16;
|
|
size -= 16;
|
|
}
|
|
std::printf("\n");
|
|
}
|
|
|
|
|
|
void printMacsbug(uint32_t pc, uint32_t opcode)
|
|
{
|
|
|
|
unsigned mboffset;
|
|
switch(opcode)
|
|
{
|
|
case 0x4E75: // rts
|
|
case 0x4ED0: // jmp (a0)
|
|
mboffset = 2;
|
|
break;
|
|
case 0x4E74: // rtd #
|
|
mboffset = 4;
|
|
break;
|
|
default:
|
|
return;
|
|
break;
|
|
}
|
|
|
|
|
|
pc += mboffset;
|
|
// check for MacsBug name after rts.
|
|
std::string s;
|
|
unsigned b = debuggerReadByte(pc);
|
|
if (b > 0x80 && b < 0xa0)
|
|
{
|
|
b -= 0x80;
|
|
pc++;
|
|
s.reserve(b);
|
|
for (unsigned i = 0; i < b; ++i)
|
|
{
|
|
s.push_back(debuggerReadByte(pc++));
|
|
}
|
|
printf("%s\n", s.c_str());
|
|
}
|
|
|
|
}
|
|
|
|
|
|
uint32_t disasm(uint32_t pc, uint16_t *op = nullptr)
|
|
{
|
|
|
|
static char strings[4][256];
|
|
|
|
if (pc >= Flags.memorySize)
|
|
{
|
|
if (op) *op = 0;
|
|
return pc;
|
|
}
|
|
|
|
uint16_t opcode = debuggerReadWord(pc);
|
|
if (op) *op = opcode;
|
|
|
|
if ((opcode & 0xf000) == 0xa000)
|
|
{
|
|
|
|
const char *name;
|
|
|
|
name = TrapName(opcode);
|
|
|
|
if (name)
|
|
{
|
|
printf("$%08X %-51s ; %04X\n", pc, name, opcode);
|
|
}
|
|
else
|
|
{
|
|
printf("$%08X Tool #$%04X ; %04X\n",
|
|
pc, opcode, opcode);
|
|
}
|
|
|
|
pc += 2;
|
|
return pc;
|
|
}
|
|
|
|
for (unsigned j = 0; j < 4; ++j) strings[j][0] = 0;
|
|
|
|
uint32_t newpc = cpuDisOpcode(pc, strings[0], strings[1], strings[2], strings[3]);
|
|
printf("%s %-10s %-40s ; %s\n", strings[0], strings[2], strings[3], strings[1]);
|
|
|
|
printMacsbug(pc, opcode);
|
|
|
|
return newpc;
|
|
}
|
|
|
|
|
|
|
|
bool step(bool trace)
|
|
{
|
|
// return false to break (toolbreak, address break, etc.)
|
|
|
|
uint16_t op;
|
|
cpuExecuteInstruction();
|
|
|
|
uint32_t pc = cpuGetPC();
|
|
if (trace) disasm(pc, &op);
|
|
else op = debuggerReadWord(pc);
|
|
|
|
if (Flags.traceMacsbug && !trace)
|
|
printMacsbug(pc, op);
|
|
|
|
// will this also be set by an interrupt?
|
|
if (cpuGetStop())
|
|
{
|
|
if (!trace) disasm(pc);
|
|
printf("CPU stopped\n");
|
|
return false;
|
|
}
|
|
|
|
if (sigInt)
|
|
{
|
|
if (!trace) disasm(pc);
|
|
printf("^C break\n");
|
|
sigInt = false;
|
|
return false;
|
|
}
|
|
|
|
|
|
// check for pc breaks
|
|
if (brkMap.lookup(pc))
|
|
{
|
|
if (!trace) disasm(pc);
|
|
printf("Address break: $%08x\n", pc);
|
|
return false;
|
|
}
|
|
|
|
// todo -- instruction break for rts /rtn
|
|
|
|
// check for toolbreaks.
|
|
if ((op & 0xf000) == 0xa000)
|
|
{
|
|
if (tbrkMap.lookup(op))
|
|
{
|
|
if (!trace) disasm(pc);
|
|
printf("Tool break: $%04x\n", op);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (pc > Flags.memorySize)
|
|
{
|
|
printf("PC out of range\n");
|
|
return false;
|
|
}
|
|
|
|
uint32_t sp = cpuGetAReg(7);
|
|
if (sp < Flags.stackRange.first)
|
|
{
|
|
printf("Stack overflow error\n");
|
|
return false;
|
|
}
|
|
|
|
if (sp > Flags.stackRange.second)
|
|
{
|
|
printf("Stack underflow error\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
void DebugHelp(const Command &cmd)
|
|
{
|
|
printf("help\n");
|
|
printf("break expression\n");
|
|
printf("step\n");
|
|
printf("continue\n");
|
|
printf("\n");
|
|
printf("print expression\n");
|
|
printf("list expression\n");
|
|
printf("dump expression\n");
|
|
printf("register=expression\n");
|
|
printf("\n");
|
|
printf("registers: a0-7, d0-7, pc, sp, fp, csr\n");
|
|
printf("\n");
|
|
}
|
|
|
|
|
|
void DebugPrint(const Command &cmd)
|
|
{
|
|
for (unsigned i = 0; i < cmd.argc; ++i)
|
|
{
|
|
uint32_t data = cmd.argv[i];
|
|
printf("$%08x %12u", data, data);
|
|
if (data & 0x80000000)
|
|
printf(" %12d", (int32_t)data);
|
|
if ((data & 0xffff8000) == 0x8000)
|
|
printf(" %6d", (int16_t)data);
|
|
|
|
printf("\n");
|
|
}
|
|
}
|
|
|
|
|
|
void DebugDump(const Command &cmd)
|
|
{
|
|
// TODO -- if no address, use previous address.
|
|
// TODO -- support range?
|
|
|
|
|
|
uint32_t start = cmd.argv[0];
|
|
uint32_t end = cmd.argc == 2 ? cmd.argv[1] : start + 256;
|
|
|
|
if (start >= Flags.memorySize) return;
|
|
|
|
end = std::min(end, Flags.memorySize);
|
|
ssize_t size = end - start;
|
|
|
|
hexdump(Flags.memory + start, size, start);
|
|
}
|
|
|
|
|
|
|
|
void DebugList(const Command &cmd)
|
|
{
|
|
// TODO -- if no address, use previous address.
|
|
// TODO -- support range?
|
|
if (cmd.argc == 1)
|
|
{
|
|
uint32_t pc = cmd.argv[0];
|
|
if (pc & 0x01)
|
|
{
|
|
printf("address is not aligned: $%08x\n", pc);
|
|
return;
|
|
}
|
|
|
|
for (unsigned i = 0; i < 32; ++i)
|
|
{
|
|
if (pc >= Flags.memorySize) break;
|
|
pc = disasm(pc);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void DebugPrintRegisters(const Command &cmd)
|
|
{
|
|
char srbits[20];
|
|
|
|
uint16_t sr = cpuGetSR();
|
|
|
|
srbits[0] = sr & (1 << 15) ? 'T' : ' ';
|
|
srbits[1] = sr & (1 << 14) ? 'T' : ' ';
|
|
srbits[2] = sr & (1 << 13) ? 'S' : ' ';
|
|
srbits[3] = sr & (1 << 12) ? 'M' : ' ';
|
|
srbits[4] = ' ';
|
|
srbits[5] = sr & (1 << 10) ? 'I' : ' ';
|
|
srbits[6] = sr & (1 << 9) ? 'I' : ' ';
|
|
srbits[7] = sr & (1 << 8) ? 'I' : ' ';
|
|
srbits[8] = ' ';
|
|
srbits[9] = ' ';
|
|
srbits[10] = ' ';
|
|
srbits[11] = sr & (1 << 4) ? 'X' : ' ';
|
|
srbits[12] = sr & (1 << 3) ? 'N' : ' ';
|
|
srbits[13] = sr & (1 << 2) ? 'Z' : ' ';
|
|
srbits[14] = sr & (1 << 1) ? 'V' : ' ';
|
|
srbits[15] = sr & (1 << 0) ? 'C' : ' ';
|
|
srbits[16] = 0;
|
|
|
|
|
|
printf(" 0 1 2 3 4 5 6 7\n");
|
|
printf("D: %08x %08x %08x %08x %08x %08x %08x %08x\n",
|
|
cpuGetDReg(0), cpuGetDReg(1), cpuGetDReg(2), cpuGetDReg(3),
|
|
cpuGetDReg(4), cpuGetDReg(5), cpuGetDReg(6), cpuGetDReg(7)
|
|
|
|
);
|
|
|
|
printf("A: %08x %08x %08x %08x %08x %08x %08x %08x\n",
|
|
cpuGetAReg(0), cpuGetAReg(1), cpuGetAReg(2), cpuGetAReg(3),
|
|
cpuGetAReg(4), cpuGetAReg(5), cpuGetAReg(6), cpuGetAReg(7)
|
|
);
|
|
|
|
printf("PC: %08X CSR: %04x %s\n", cpuGetPC(), sr, srbits);
|
|
|
|
}
|
|
|
|
|
|
void DebugToolBreak(Command &cmd)
|
|
{
|
|
for (unsigned i = 0; i < cmd.argc; ++i)
|
|
{
|
|
int32_t tool = (int32_t)cmd.argv[i];
|
|
bool remove = false;
|
|
|
|
if (tool < 0)
|
|
{
|
|
tool = -tool;
|
|
remove = true;
|
|
}
|
|
|
|
if (tool >= 0xa000 && tool <= 0xafff)
|
|
{
|
|
if (remove) tbrkMap.remove(tool);
|
|
else tbrkMap.add(tool);
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Invalid tool: $%04x\n", tool);
|
|
}
|
|
}
|
|
}
|
|
|
|
void DebugBreak(Command &cmd)
|
|
{
|
|
// 24-bit only, - address to remove.
|
|
for (unsigned i = 0; i < cmd.argc; ++i)
|
|
{
|
|
int32_t address = (int32_t)cmd.argv[i];
|
|
bool remove = false;
|
|
|
|
if (address < 0)
|
|
{
|
|
address = -address;
|
|
remove = true;
|
|
}
|
|
|
|
if ((address & 0xff000000) == 0)
|
|
{
|
|
if (remove) brkMap.remove(address);
|
|
else brkMap.add(address);
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Invalid address: $%08x\n", address);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void DebugStep(const Command &cmd)
|
|
{
|
|
// TODO - step n to step specified # of instructions.
|
|
// TODO -- step @address to step until address?
|
|
|
|
// disasm 1 line, execute it.
|
|
|
|
int count = 0;
|
|
if (cmd.argc == 1) count = (int)cmd.argv[0];
|
|
if (count < 1) count = 1;
|
|
|
|
// TODO -- move to common function...
|
|
for (int i = 0; i < count; ++i)
|
|
{
|
|
if (!step(true)) break;
|
|
}
|
|
}
|
|
|
|
|
|
void DebugContinue(const Command &cmd)
|
|
{
|
|
while (step(false)) ;
|
|
}
|
|
|
|
void DebugSetARegister(Command &cmd)
|
|
{
|
|
if (cmd.argc != 2) return;
|
|
unsigned reg = cmd.argv[0];
|
|
uint32_t value = cmd.argv[1];
|
|
|
|
if (reg > 7) return;
|
|
if (reg == 7)
|
|
{
|
|
// sp/7 must be aligned.
|
|
if (value & 0x01)
|
|
{
|
|
fprintf(stderr, "Address is not aligned: $%08x\n", value);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (value > Flags.memorySize)
|
|
{
|
|
fprintf(stderr, "Warning: address exceeeds memory size: $%08x\n", value);
|
|
}
|
|
|
|
cpuSetAReg(reg, value);
|
|
}
|
|
|
|
|
|
void DebugSetDRegister(Command &cmd)
|
|
{
|
|
if (cmd.argc != 2) return;
|
|
unsigned reg = cmd.argv[0];
|
|
uint32_t value = cmd.argv[1];
|
|
|
|
cpuSetDReg(reg, value);
|
|
}
|
|
|
|
void DebugSetXRegister(Command &cmd)
|
|
{
|
|
if (cmd.argc != 2) return;
|
|
unsigned reg = cmd.argv[0];
|
|
uint32_t value = cmd.argv[1];
|
|
|
|
if (reg == 0)
|
|
{
|
|
if (value & 0x01)
|
|
{
|
|
fprintf(stderr, "Address is not aligned: $%08x\n", value);
|
|
return;
|
|
}
|
|
if (value > Flags.memorySize)
|
|
{
|
|
fprintf(stderr, "Warning: address exceeeds memory size: $%08x\n", value);
|
|
}
|
|
cpuSetPC(value);
|
|
return;
|
|
}
|
|
|
|
if (reg == 1)
|
|
{
|
|
cpuSetSR(value);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO -- RUN command - reload, re-initialize, re-execute
|
|
// TODO -- parser calls commands directly (except trace/step/run/etc)
|
|
void DebugShell()
|
|
{
|
|
char *cp;
|
|
|
|
add_history("!Andy, it still has history!");
|
|
|
|
// start it up
|
|
printf("MPW Debugger shell\n\n");
|
|
disasm(cpuGetPC());
|
|
|
|
signal(SIGINT, sigIntHandler);
|
|
|
|
for(;;)
|
|
{
|
|
cp = readline("] ");
|
|
if (!cp)
|
|
{
|
|
printf("\n");
|
|
break; // prompt for exit?
|
|
}
|
|
// parse the command...
|
|
const char *iter = cp;
|
|
while (*iter && isspace(*iter)) ++iter;
|
|
|
|
if (*iter)
|
|
{
|
|
Command cmd;
|
|
std::memset(&cmd, 0, sizeof(cmd));
|
|
if (ParseLine(iter, &cmd))
|
|
{
|
|
switch(cmd.action)
|
|
{
|
|
case NullCommand:
|
|
break;
|
|
|
|
case Print:
|
|
DebugPrint(cmd);
|
|
break;
|
|
|
|
case Dump:
|
|
DebugDump(cmd);
|
|
break;
|
|
|
|
case List:
|
|
DebugList(cmd);
|
|
break;
|
|
|
|
case PrintRegisters:
|
|
DebugPrintRegisters(cmd);
|
|
break;
|
|
|
|
case Step:
|
|
DebugStep(cmd);
|
|
break;
|
|
|
|
case Continue:
|
|
DebugContinue(cmd);
|
|
break;
|
|
|
|
case TBreak:
|
|
DebugToolBreak(cmd);
|
|
break;
|
|
|
|
case Break:
|
|
DebugBreak(cmd);
|
|
break;
|
|
|
|
case SetARegister:
|
|
DebugSetARegister(cmd);
|
|
break;
|
|
|
|
case SetDRegister:
|
|
DebugSetDRegister(cmd);
|
|
break;
|
|
|
|
case SetXRegister:
|
|
DebugSetXRegister(cmd);
|
|
break;
|
|
|
|
default:
|
|
DebugHelp(cmd);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// todo -- don't add if same as previous command.
|
|
add_history(cp);
|
|
}
|
|
free(cp);
|
|
}
|
|
|
|
}
|
|
|