#include #include #include #include #include #include #include #include #include #include #include #include #include "loader.h" #include "address_map.h" #include #include #include #include #include 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); } }