Mercurial > vba-linux
diff src/sdl/debugger.cpp @ 1:f9f4f1b99eed
importing src directory
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Sat, 03 Mar 2012 10:31:27 -0600 (2012-03-03) |
parents | |
children |
line wrap: on
line diff
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/src/sdl/debugger.cpp Sat Mar 03 10:31:27 2012 -0600 1.3 @@ -0,0 +1,1460 @@ 1.4 +// VisualBoyAdvance - Nintendo Gameboy/GameboyAdvance (TM) emulator. 1.5 +// Copyright (C) 1999-2003 Forgotten 1.6 +// Copyright (C) 2004 Forgotten and the VBA development team 1.7 + 1.8 +// This program is free software; you can redistribute it and/or modify 1.9 +// it under the terms of the GNU General Public License as published by 1.10 +// the Free Software Foundation; either version 2, or(at your option) 1.11 +// any later version. 1.12 +// 1.13 +// This program is distributed in the hope that it will be useful, 1.14 +// but WITHOUT ANY WARRANTY; without even the implied warranty of 1.15 +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 1.16 +// GNU General Public License for more details. 1.17 +// 1.18 +// You should have received a copy of the GNU General Public License 1.19 +// along with this program; if not, write to the Free Software Foundation, 1.20 +// Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. 1.21 + 1.22 +extern "C" { 1.23 +#include <stdio.h> 1.24 +#include <stdlib.h> 1.25 +} // FIXME: should use c++ headers instead 1.26 + 1.27 +#include <string.h> 1.28 + 1.29 +#include "Port.h" 1.30 +#include "gba/GBA.h" 1.31 +#include "gba/GBAGlobals.h" 1.32 +#include "gba/GBACheats.h" 1.33 +#include "gba/armdis.h" 1.34 +#include "gba/elf.h" 1.35 +#include "common/System.h" 1.36 +#include "exprNode.h" 1.37 + 1.38 +extern bool debugger; 1.39 +extern int emulating; 1.40 + 1.41 +extern struct EmulatedSystem theEmulator; 1.42 + 1.43 +#define debuggerReadMemory(addr) \ 1.44 + READ32LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) 1.45 + 1.46 +#define debuggerReadHalfWord(addr) \ 1.47 + READ16LE((&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask])) 1.48 + 1.49 +#define debuggerReadByte(addr) \ 1.50 + map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] 1.51 + 1.52 +#define debuggerWriteMemory(addr, value) \ 1.53 + WRITE32LE(&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask], value) 1.54 + 1.55 +#define debuggerWriteHalfWord(addr, value) \ 1.56 + WRITE16LE(&map[(addr)>>24].address[(addr) & map[(addr)>>24].mask], value) 1.57 + 1.58 +#define debuggerWriteByte(addr, value) \ 1.59 + map[(addr)>>24].address[(addr) & map[(addr)>>24].mask] = (value) 1.60 + 1.61 +struct breakpointInfo { 1.62 + u32 address; 1.63 + u32 value; 1.64 + int size; 1.65 +}; 1.66 + 1.67 +struct DebuggerCommand { 1.68 + const char *name; 1.69 + void (*function)(int,char **); 1.70 + const char *help; 1.71 + const char *syntax; 1.72 +}; 1.73 + 1.74 +void debuggerContinueAfterBreakpoint(); 1.75 + 1.76 +void debuggerHelp(int,char **); 1.77 +void debuggerNext(int,char **); 1.78 +void debuggerContinue(int, char **); 1.79 +void debuggerRegisters(int, char **); 1.80 +void debuggerBreak(int, char **); 1.81 +void debuggerBreakDelete(int, char **); 1.82 +void debuggerBreakList(int, char **); 1.83 +void debuggerBreakArm(int, char **); 1.84 +void debuggerBreakWriteClear(int, char **); 1.85 +void debuggerBreakThumb(int, char **); 1.86 +void debuggerBreakWrite(int, char **); 1.87 +void debuggerDebug(int, char **); 1.88 +void debuggerDisassemble(int, char **); 1.89 +void debuggerDisassembleArm(int, char **); 1.90 +void debuggerDisassembleThumb(int, char **); 1.91 +void debuggerEditByte(int, char **); 1.92 +void debuggerEditHalfWord(int, char **); 1.93 +void debuggerEdit(int, char **); 1.94 +void debuggerIo(int, char **); 1.95 +void debuggerLocals(int, char **); 1.96 +void debuggerMemoryByte(int, char **); 1.97 +void debuggerMemoryHalfWord(int, char **); 1.98 +void debuggerMemory(int, char **); 1.99 +void debuggerPrint(int, char **); 1.100 +void debuggerQuit(int, char **); 1.101 +void debuggerSetRadix(int, char **); 1.102 +void debuggerSymbols(int, char **); 1.103 +void debuggerVerbose(int, char **); 1.104 +void debuggerWhere(int, char **); 1.105 + 1.106 +DebuggerCommand debuggerCommands[] = { 1.107 + { "?", debuggerHelp, "Shows this help information. Type ? <command> for command help", "[<command>]" }, 1.108 + { "ba", debuggerBreakArm, "Adds an ARM breakpoint", "<address>" }, 1.109 + { "bd", debuggerBreakDelete,"Deletes a breakpoint", "<number>" }, 1.110 + { "bl", debuggerBreakList, "Lists breakpoints" }, 1.111 + { "bpw", debuggerBreakWrite, "Break on write", "<address> <size>" }, 1.112 + { "bpwc", debuggerBreakWriteClear, "Clear break on write", NULL }, 1.113 + { "break", debuggerBreak, "Adds a breakpoint on the given function", "<function>|<line>|<file:line>" }, 1.114 + { "bt", debuggerBreakThumb, "Adds a THUMB breakpoint", "<address>" }, 1.115 + { "c", debuggerContinue, "Continues execution" , NULL }, 1.116 + { "d", debuggerDisassemble, "Disassembles instructions", "[<address> [<number>]]" }, 1.117 + { "da", debuggerDisassembleArm, "Disassembles ARM instructions", "[<address> [<number>]]" }, 1.118 + { "dt", debuggerDisassembleThumb, "Disassembles THUMB instructions", "[<address> [<number>]]" }, 1.119 + { "eb", debuggerEditByte, "Modify memory location (byte)", "<address> <hex value>" }, 1.120 + { "eh", debuggerEditHalfWord,"Modify memory location (half-word)","<address> <hex value>" }, 1.121 + { "ew", debuggerEdit, "Modify memory location (word)", "<address> <hex value" }, 1.122 + { "h", debuggerHelp, "Shows this help information. Type h <command> for command help", "[<command>]" }, 1.123 + { "io", debuggerIo, "Show I/O registers status", "[video|video2|dma|timer|misc]" }, 1.124 + { "locals", debuggerLocals, "Shows local variables", NULL }, 1.125 + { "mb", debuggerMemoryByte, "Shows memory contents (bytes)", "<address>" }, 1.126 + { "mh", debuggerMemoryHalfWord, "Shows memory contents (half-words)", "<address>"}, 1.127 + { "mw", debuggerMemory, "Shows memory contents (words)", "<address>" }, 1.128 + { "n", debuggerNext, "Executes the next instruction", "[<count>]" }, 1.129 + { "print", debuggerPrint, "Print the value of a expression (if known)", "[/x|/o|/d] <expression>" }, 1.130 + { "q", debuggerQuit, "Quits the emulator", NULL }, 1.131 + { "r", debuggerRegisters, "Shows ARM registers", NULL }, 1.132 + { "radix", debuggerSetRadix, "Sets the print radix", "<radix>" }, 1.133 + { "symbols", debuggerSymbols, "List symbols", "[<symbol>]" }, 1.134 +#ifndef FINAL_VERSION 1.135 + { "trace", debuggerDebug, "Sets the trace level", "<value>" }, 1.136 +#endif 1.137 +#ifdef DEV_VERSION 1.138 + { "verbose", debuggerVerbose, "Change verbose setting", "<value>" }, 1.139 +#endif 1.140 + { "where", debuggerWhere, "Shows call chain", NULL }, 1.141 + { NULL, NULL, NULL, NULL} // end marker 1.142 +}; 1.143 + 1.144 +breakpointInfo debuggerBreakpointList[100]; 1.145 + 1.146 +int debuggerNumOfBreakpoints = 0; 1.147 +bool debuggerAtBreakpoint = false; 1.148 +int debuggerBreakpointNumber = 0; 1.149 +int debuggerRadix = 0; 1.150 + 1.151 +void debuggerApplyBreakpoint(u32 address, int num, int size) 1.152 +{ 1.153 + if(size) 1.154 + debuggerWriteMemory(address, (u32)(0xe1200070 | 1.155 + (num & 0xf) | 1.156 + ((num<<4)&0xf0))); 1.157 + else 1.158 + debuggerWriteHalfWord(address, 1.159 + (u16)(0xbe00 | num)); 1.160 +} 1.161 + 1.162 +void debuggerDisableBreakpoints() 1.163 +{ 1.164 + for(int i = 0; i < debuggerNumOfBreakpoints; i++) { 1.165 + if(debuggerBreakpointList[i].size) 1.166 + debuggerWriteMemory(debuggerBreakpointList[i].address, 1.167 + debuggerBreakpointList[i].value); 1.168 + else 1.169 + debuggerWriteHalfWord(debuggerBreakpointList[i].address, 1.170 + debuggerBreakpointList[i].value); 1.171 + } 1.172 +} 1.173 + 1.174 +void debuggerEnableBreakpoints(bool skipPC) 1.175 +{ 1.176 + for(int i = 0; i < debuggerNumOfBreakpoints; i++) { 1.177 + if(debuggerBreakpointList[i].address == armNextPC && skipPC) 1.178 + continue; 1.179 + 1.180 + debuggerApplyBreakpoint(debuggerBreakpointList[i].address, 1.181 + i, 1.182 + debuggerBreakpointList[i].size); 1.183 + } 1.184 +} 1.185 + 1.186 +void debuggerUsage(const char *cmd) 1.187 +{ 1.188 + for(int i = 0; ; i++) { 1.189 + if(debuggerCommands[i].name) { 1.190 + if(!strcmp(debuggerCommands[i].name, cmd)) { 1.191 + printf("%s %s\t%s\n", 1.192 + debuggerCommands[i].name, 1.193 + debuggerCommands[i].syntax ? debuggerCommands[i].syntax : "", 1.194 + debuggerCommands[i].help); 1.195 + break; 1.196 + } 1.197 + } else { 1.198 + printf("Unrecognized command '%s'.", cmd); 1.199 + break; 1.200 + } 1.201 + } 1.202 +} 1.203 + 1.204 +void debuggerPrintBaseType(Type *t, u32 value, u32 location, 1.205 + LocationType type, 1.206 + int bitSize, int bitOffset) 1.207 +{ 1.208 + if(bitSize) { 1.209 + if(bitOffset) 1.210 + value >>= ((t->size*8)-bitOffset-bitSize); 1.211 + value &= (1 << bitSize)-1; 1.212 + } else { 1.213 + if(t->size == 2) 1.214 + value &= 0xFFFF; 1.215 + else if(t->size == 1) 1.216 + value &= 0xFF; 1.217 + } 1.218 + 1.219 + if(t->size == 8) { 1.220 + u64 value = 0; 1.221 + if(type == LOCATION_memory) { 1.222 + value = debuggerReadMemory(location) | 1.223 + ((u64)debuggerReadMemory(location+4)<<32); 1.224 + } else if(type == LOCATION_register) { 1.225 + value = reg[location].I | ((u64)reg[location+1].I << 32); 1.226 + } 1.227 + switch(t->encoding) { 1.228 + case DW_ATE_signed: 1.229 + switch(debuggerRadix) { 1.230 + case 0: 1.231 + printf("%lld", value); 1.232 + break; 1.233 + case 1: 1.234 + printf("0x%llx", value); 1.235 + break; 1.236 + case 2: 1.237 + printf("0%llo", value); 1.238 + break; 1.239 + } 1.240 + break; 1.241 + case DW_ATE_unsigned: 1.242 + switch(debuggerRadix) { 1.243 + case 0: 1.244 + printf("%llu", value); 1.245 + break; 1.246 + case 1: 1.247 + printf("0x%llx", value); 1.248 + break; 1.249 + case 2: 1.250 + printf("0%llo", value); 1.251 + break; 1.252 + } 1.253 + break; 1.254 + default: 1.255 + printf("Unknowing 64-bit encoding\n"); 1.256 + } 1.257 + return; 1.258 + } 1.259 + 1.260 + switch(t->encoding) { 1.261 + case DW_ATE_boolean: 1.262 + if(value) 1.263 + printf("true"); 1.264 + else 1.265 + printf("false"); 1.266 + break; 1.267 + case DW_ATE_signed: 1.268 + switch(debuggerRadix) { 1.269 + case 0: 1.270 + printf("%d", value); 1.271 + break; 1.272 + case 1: 1.273 + printf("0x%x", value); 1.274 + break; 1.275 + case 2: 1.276 + printf("0%o", value); 1.277 + break; 1.278 + } 1.279 + break; 1.280 + case DW_ATE_unsigned: 1.281 + case DW_ATE_unsigned_char: 1.282 + switch(debuggerRadix) { 1.283 + case 0: 1.284 + printf("%u", value); 1.285 + break; 1.286 + case 1: 1.287 + printf("0x%x", value); 1.288 + break; 1.289 + case 2: 1.290 + printf("0%o", value); 1.291 + break; 1.292 + } 1.293 + break; 1.294 + default: 1.295 + printf("UNKNOWN BASE %d %08x", t->encoding, value); 1.296 + } 1.297 +} 1.298 + 1.299 +char *debuggerPrintType(Type *t) 1.300 +{ 1.301 + char buffer[1024]; 1.302 + static char buffer2[1024]; 1.303 + 1.304 + if(t->type == TYPE_pointer) { 1.305 + if(t->pointer) 1.306 + strcpy(buffer, debuggerPrintType(t->pointer)); 1.307 + else 1.308 + strcpy(buffer, "void"); 1.309 + sprintf(buffer2, "%s *", buffer); 1.310 + return buffer2; 1.311 + } else if(t->type == TYPE_reference) { 1.312 + strcpy(buffer, debuggerPrintType(t->pointer)); 1.313 + sprintf(buffer2, "%s &", buffer); 1.314 + return buffer2; 1.315 + } 1.316 + return t->name; 1.317 +} 1.318 + 1.319 +void debuggerPrintValueInternal(Function *, Type *, ELFBlock *, int, int, u32); 1.320 +void debuggerPrintValueInternal(Function *f, Type *t, 1.321 + int bitSize, int bitOffset, 1.322 + u32 objLocation, LocationType type); 1.323 + 1.324 +u32 debuggerGetValue(u32 location, LocationType type) 1.325 +{ 1.326 + switch(type) { 1.327 + case LOCATION_memory: 1.328 + return debuggerReadMemory(location); 1.329 + case LOCATION_register: 1.330 + return reg[location].I; 1.331 + case LOCATION_value: 1.332 + return location; 1.333 + } 1.334 + return 0; 1.335 +} 1.336 + 1.337 +void debuggerPrintPointer(Type *t, u32 value) 1.338 +{ 1.339 + printf("(%s)0x%08x", debuggerPrintType(t), value); 1.340 +} 1.341 + 1.342 +void debuggerPrintReference(Type *t, u32 value) 1.343 +{ 1.344 + printf("(%s)0x%08x", debuggerPrintType(t), value); 1.345 +} 1.346 + 1.347 +void debuggerPrintFunction(Type *t, u32 value) 1.348 +{ 1.349 + printf("(%s)0x%08x", debuggerPrintType(t), value); 1.350 +} 1.351 + 1.352 +void debuggerPrintArray(Type *t, u32 value) 1.353 +{ 1.354 + // todo 1.355 + printf("(%s[])0x%08x", debuggerPrintType(t->array->type), value); 1.356 +} 1.357 + 1.358 +void debuggerPrintMember(Function *f, 1.359 + Member *m, 1.360 + u32 objLocation, 1.361 + u32 location) 1.362 +{ 1.363 + int bitSize = m->bitSize; 1.364 + if(bitSize) { 1.365 + u32 value = 0; 1.366 + int off = m->bitOffset; 1.367 + int size = m->byteSize; 1.368 + u32 v = 0; 1.369 + if(size == 1) 1.370 + v = debuggerReadByte(location); 1.371 + else if(size == 2) 1.372 + v = debuggerReadHalfWord(location); 1.373 + else if(size == 4) 1.374 + v = debuggerReadMemory(location); 1.375 + 1.376 + while(bitSize) { 1.377 + int top = size*8 - off; 1.378 + int bot = top - bitSize; 1.379 + top--; 1.380 + if(bot >= 0) { 1.381 + value = (v >> (size*8 - bitSize - off)) & ((1 << bitSize)-1); 1.382 + bitSize = 0; 1.383 + } else { 1.384 + value |= (v & ((1 << top)-1)) << (bitSize - top); 1.385 + bitSize -= (top+1); 1.386 + location -= size; 1.387 + off = 0; 1.388 + if(size == 1) 1.389 + v = debuggerReadByte(location); 1.390 + else if(size == 2) 1.391 + v = debuggerReadHalfWord(location); 1.392 + else 1.393 + v = debuggerReadMemory(location); 1.394 + } 1.395 + } 1.396 + debuggerPrintBaseType(m->type, value, location, LOCATION_memory, 1.397 + bitSize, 0); 1.398 + } else { 1.399 + debuggerPrintValueInternal(f, m->type, m->location, m->bitSize, 1.400 + m->bitOffset, objLocation); 1.401 + } 1.402 +} 1.403 + 1.404 +void debuggerPrintStructure(Function *f, Type *t, u32 objLocation) 1.405 +{ 1.406 + printf("{"); 1.407 + int count = t->structure->memberCount; 1.408 + int i = 0; 1.409 + while(i < count) { 1.410 + Member *m = &t->structure->members[i]; 1.411 + printf("%s=", m->name); 1.412 + LocationType type; 1.413 + u32 location = elfDecodeLocation(f, m->location, &type, objLocation); 1.414 + debuggerPrintMember(f, m, objLocation, location); 1.415 + i++; 1.416 + if(i < count) 1.417 + printf(","); 1.418 + } 1.419 + printf("}"); 1.420 +} 1.421 + 1.422 +void debuggerPrintUnion(Function *f, Type *t, u32 objLocation) 1.423 +{ 1.424 + // todo 1.425 + printf("{"); 1.426 + int count = t->structure->memberCount; 1.427 + int i = 0; 1.428 + while(i < count) { 1.429 + Member *m = &t->structure->members[i]; 1.430 + printf("%s=", m->name); 1.431 + debuggerPrintMember(f, m, objLocation, 0); 1.432 + i++; 1.433 + if(i < count) 1.434 + printf(","); 1.435 + } 1.436 + printf("}"); 1.437 +} 1.438 + 1.439 +void debuggerPrintEnum(Type *t, u32 value) 1.440 +{ 1.441 + int i; 1.442 + for(i = 0; i < t->enumeration->count; i++) { 1.443 + EnumMember *m = (EnumMember *)&t->enumeration->members[i]; 1.444 + if(value == m->value) { 1.445 + puts(m->name); 1.446 + return; 1.447 + } 1.448 + } 1.449 + printf("(UNKNOWN VALUE) %d", value); 1.450 +} 1.451 + 1.452 +void debuggerPrintValueInternal(Function *f, Type *t, 1.453 + int bitSize, int bitOffset, 1.454 + u32 objLocation, LocationType type) 1.455 +{ 1.456 + u32 value = debuggerGetValue(objLocation, type); 1.457 + if(!t) { 1.458 + printf("void"); 1.459 + return; 1.460 + } 1.461 + switch(t->type) { 1.462 + case TYPE_base: 1.463 + debuggerPrintBaseType(t, value, objLocation, type, bitSize, bitOffset); 1.464 + break; 1.465 + case TYPE_pointer: 1.466 + debuggerPrintPointer(t, value); 1.467 + break; 1.468 + case TYPE_reference: 1.469 + debuggerPrintReference(t, value); 1.470 + break; 1.471 + case TYPE_function: 1.472 + debuggerPrintFunction(t, value); 1.473 + break; 1.474 + case TYPE_array: 1.475 + debuggerPrintArray(t, objLocation); 1.476 + break; 1.477 + case TYPE_struct: 1.478 + debuggerPrintStructure(f, t, objLocation); 1.479 + break; 1.480 + case TYPE_union: 1.481 + debuggerPrintUnion(f, t, objLocation); 1.482 + break; 1.483 + case TYPE_enum: 1.484 + debuggerPrintEnum(t, value); 1.485 + break; 1.486 + default: 1.487 + printf("%08x", value); 1.488 + break; 1.489 + } 1.490 +} 1.491 + 1.492 +void debuggerPrintValueInternal(Function *f, Type *t, ELFBlock *loc, 1.493 + int bitSize, int bitOffset, u32 objLocation) 1.494 +{ 1.495 + LocationType type; 1.496 + u32 location; 1.497 + if(loc) { 1.498 + if(objLocation) 1.499 + location = elfDecodeLocation(f, loc, &type, objLocation); 1.500 + else 1.501 + location = elfDecodeLocation(f, loc,&type); 1.502 + } else { 1.503 + location = objLocation; 1.504 + type = LOCATION_memory; 1.505 + } 1.506 + 1.507 + debuggerPrintValueInternal(f, t, bitSize, bitOffset, location, type); 1.508 +} 1.509 + 1.510 +void debuggerPrintValue(Function *f, Object *o) 1.511 +{ 1.512 + debuggerPrintValueInternal(f, o->type, o->location, 0, 0, 0); 1.513 + 1.514 + printf("\n"); 1.515 +} 1.516 + 1.517 +void debuggerSymbols(int argc, char **argv) 1.518 +{ 1.519 + int i = 0; 1.520 + u32 value; 1.521 + u32 size; 1.522 + int type; 1.523 + bool match = false; 1.524 + int matchSize = 0; 1.525 + char *matchStr = NULL; 1.526 + 1.527 + if(argc == 2) { 1.528 + match = true; 1.529 + matchSize = strlen(argv[1]); 1.530 + matchStr = argv[1]; 1.531 + } 1.532 + printf("Symbol Value Size Type \n"); 1.533 + printf("-------------------- ------- -------- -------\n"); 1.534 + char *s = NULL; 1.535 + while((s = elfGetSymbol(i, &value, &size, &type))) { 1.536 + if(*s) { 1.537 + if(match) { 1.538 + if(strncmp(s, matchStr, matchSize) != 0) { 1.539 + i++; 1.540 + continue; 1.541 + } 1.542 + } 1.543 + const char *ts = "?"; 1.544 + switch(type) { 1.545 + case 2: 1.546 + ts = "ARM"; 1.547 + break; 1.548 + case 0x0d: 1.549 + ts = "THUMB"; 1.550 + break; 1.551 + case 1: 1.552 + ts = "DATA"; 1.553 + break; 1.554 + } 1.555 + printf("%-20s %08x %08x %-7s\n", 1.556 + s, value, size, ts); 1.557 + } 1.558 + i++; 1.559 + } 1.560 +} 1.561 + 1.562 +void debuggerSetRadix(int argc, char **argv) 1.563 +{ 1.564 + if(argc != 2) 1.565 + debuggerUsage(argv[0]); 1.566 + else { 1.567 + int r = atoi(argv[1]); 1.568 + 1.569 + bool error = false; 1.570 + switch(r) { 1.571 + case 10: 1.572 + debuggerRadix = 0; 1.573 + break; 1.574 + case 8: 1.575 + debuggerRadix = 2; 1.576 + break; 1.577 + case 16: 1.578 + debuggerRadix = 1; 1.579 + break; 1.580 + default: 1.581 + error = true; 1.582 + printf("Unknown radix %d. Valid values are 8, 10 and 16.\n", r); 1.583 + break; 1.584 + } 1.585 + if(!error) 1.586 + printf("Radix set to %d\n", r); 1.587 + } 1.588 +} 1.589 + 1.590 +void debuggerPrint(int argc, char **argv) 1.591 +{ 1.592 + if(argc != 2 && argc != 3) { 1.593 + debuggerUsage(argv[0]); 1.594 + } else { 1.595 + u32 pc = armNextPC; 1.596 + Function *f = NULL; 1.597 + CompileUnit *u = NULL; 1.598 + 1.599 + elfGetCurrentFunction(pc, 1.600 + &f, &u); 1.601 + 1.602 + int oldRadix = debuggerRadix; 1.603 + if(argc == 3) { 1.604 + if(argv[1][0] == '/') { 1.605 + if(argv[1][1] == 'x') 1.606 + debuggerRadix = 1; 1.607 + else if(argv[1][1] == 'o') 1.608 + debuggerRadix = 2; 1.609 + else if(argv[1][1] == 'd') 1.610 + debuggerRadix = 0; 1.611 + else { 1.612 + printf("Unknown format %c\n", argv[1][1]); 1.613 + return; 1.614 + } 1.615 + } else { 1.616 + printf("Unknown option %s\n", argv[1]); 1.617 + return; 1.618 + } 1.619 + } 1.620 + 1.621 + char *s = argc == 2 ? argv[1] : argv[2]; 1.622 + 1.623 + extern char *exprString; 1.624 + extern int exprCol; 1.625 + extern int yyparse(); 1.626 + exprString = s; 1.627 + exprCol = 0; 1.628 + if(!yyparse()) { 1.629 + extern Node *result; 1.630 + if(result->resolve(result, f, u)) { 1.631 + if(result->member) 1.632 + debuggerPrintMember(f, 1.633 + result->member, 1.634 + result->objLocation, 1.635 + result->location); 1.636 + else 1.637 + debuggerPrintValueInternal(f, result->type, 0, 0, 1.638 + result->location, 1.639 + result->locType); 1.640 + printf("\n"); 1.641 + } else { 1.642 + printf("Error resolving expression\n"); 1.643 + } 1.644 + } else { 1.645 + printf("Error parsing expression:\n"); 1.646 + printf("%s\n", s); 1.647 + exprCol--; 1.648 + for(int i = 0; i < exprCol; i++) 1.649 + printf(" "); 1.650 + printf("^\n"); 1.651 + } 1.652 + extern void exprCleanBuffer(); 1.653 + exprCleanBuffer(); 1.654 + exprNodeCleanUp(); 1.655 + debuggerRadix = oldRadix; 1.656 + } 1.657 +} 1.658 + 1.659 +void debuggerHelp(int n, char **args) 1.660 +{ 1.661 + if(n == 2) { 1.662 + debuggerUsage(args[1]); 1.663 + } else { 1.664 + for(int i = 0; ; i++) { 1.665 + if(debuggerCommands[i].name) { 1.666 + printf("%s\t%s\n", debuggerCommands[i].name, debuggerCommands[i].help); 1.667 + } else 1.668 + break; 1.669 + } 1.670 + } 1.671 +} 1.672 + 1.673 +void debuggerDebug(int n, char **args) 1.674 +{ 1.675 + if(n == 2) { 1.676 + int v = 0; 1.677 + sscanf(args[1], "%d", &v); 1.678 + systemDebug = v; 1.679 + printf("Debug level set to %d\n", systemDebug); 1.680 + } else 1.681 + debuggerUsage("trace"); 1.682 +} 1.683 + 1.684 +void debuggerVerbose(int n, char **args) 1.685 +{ 1.686 + if(n == 2) { 1.687 + int v = 0; 1.688 + sscanf(args[1], "%d", &v); 1.689 + systemVerbose = v; 1.690 + printf("Verbose level set to %d\n", systemVerbose); 1.691 + } else 1.692 + debuggerUsage("verbose"); 1.693 +} 1.694 + 1.695 +void debuggerWhere(int n, char **args) 1.696 +{ 1.697 + void elfPrintCallChain(u32); 1.698 + elfPrintCallChain(armNextPC); 1.699 +} 1.700 + 1.701 +void debuggerLocals(int n, char **args) 1.702 +{ 1.703 + Function *f = NULL; 1.704 + CompileUnit *u = NULL; 1.705 + u32 pc = armNextPC; 1.706 + if(elfGetCurrentFunction(pc, 1.707 + &f, &u)) { 1.708 + Object *o = f->parameters; 1.709 + while(o) { 1.710 + printf("%s=", o->name); 1.711 + debuggerPrintValue(f, o); 1.712 + o = o->next; 1.713 + } 1.714 + 1.715 + o = f->variables; 1.716 + while(o) { 1.717 + bool visible = o->startScope ? pc>=o->startScope : true; 1.718 + if(visible) 1.719 + visible = o->endScope ? pc < o->endScope : true; 1.720 + if(visible) { 1.721 + printf("%s=", o->name); 1.722 + debuggerPrintValue(f, o); 1.723 + } 1.724 + o = o->next; 1.725 + } 1.726 + } else { 1.727 + printf("No information for current address\n"); 1.728 + } 1.729 +} 1.730 + 1.731 +void debuggerNext(int n, char **args) 1.732 +{ 1.733 + int count = 1; 1.734 + if(n == 2) { 1.735 + sscanf(args[1], "%d", &count); 1.736 + } 1.737 + for(int i = 0; i < count; i++) { 1.738 + if(debuggerAtBreakpoint) { 1.739 + debuggerContinueAfterBreakpoint(); 1.740 + debuggerEnableBreakpoints(false); 1.741 + } else 1.742 + theEmulator.emuMain(1); 1.743 + } 1.744 + debuggerDisableBreakpoints(); 1.745 + Function *f = NULL; 1.746 + CompileUnit *u = NULL; 1.747 + u32 a = armNextPC; 1.748 + if(elfGetCurrentFunction(a, &f, &u)) { 1.749 + char *file; 1.750 + int line = elfFindLine(u, f, a, &file); 1.751 + 1.752 + printf("File %s, function %s, line %d\n", file, f->name, 1.753 + line); 1.754 + } 1.755 + debuggerRegisters(0, NULL); 1.756 +} 1.757 + 1.758 +void debuggerContinue(int n, char **args) 1.759 +{ 1.760 + if(debuggerAtBreakpoint) 1.761 + debuggerContinueAfterBreakpoint(); 1.762 + debuggerEnableBreakpoints(false); 1.763 + debugger = false; 1.764 +} 1.765 + 1.766 +void debuggerSignal(int sig,int number) 1.767 +{ 1.768 + switch(sig) { 1.769 + case 4: 1.770 + { 1.771 + printf("Illegal instruction at %08x\n", armNextPC); 1.772 + debugger = true; 1.773 + } 1.774 + break; 1.775 + case 5: 1.776 + { 1.777 + printf("Breakpoint %d reached\n", number); 1.778 + debugger = true; 1.779 + debuggerAtBreakpoint = true; 1.780 + debuggerBreakpointNumber = number; 1.781 + debuggerDisableBreakpoints(); 1.782 + 1.783 + Function *f = NULL; 1.784 + CompileUnit *u = NULL; 1.785 + 1.786 + if(elfGetCurrentFunction(armNextPC, &f, &u)) { 1.787 + char *file; 1.788 + int line = elfFindLine(u,f,armNextPC,&file); 1.789 + printf("File %s, function %s, line %d\n", file, f->name, 1.790 + line); 1.791 + } 1.792 + } 1.793 + break; 1.794 + default: 1.795 + printf("Unknown signal %d\n", sig); 1.796 + break; 1.797 + } 1.798 +} 1.799 + 1.800 +void debuggerBreakList(int, char **) 1.801 +{ 1.802 + printf("Num Address Type Symbol\n"); 1.803 + printf("--- -------- ----- ------\n"); 1.804 + for(int i = 0; i < debuggerNumOfBreakpoints; i++) { 1.805 + printf("%3d %08x %s %s\n",i, debuggerBreakpointList[i].address, 1.806 + debuggerBreakpointList[i].size ? "ARM" : "THUMB", 1.807 + elfGetAddressSymbol(debuggerBreakpointList[i].address)); 1.808 + } 1.809 +} 1.810 + 1.811 +void debuggerBreakDelete(int n, char **args) 1.812 +{ 1.813 + if(n == 2) { 1.814 + int n = 0; 1.815 + sscanf(args[1], "%d", &n); 1.816 + printf("Deleting breakpoint %d (%d)\n", n, debuggerNumOfBreakpoints); 1.817 + if(n >= 0 && n < debuggerNumOfBreakpoints) { 1.818 + n++; 1.819 + if(n < debuggerNumOfBreakpoints) { 1.820 + for(int i = n; i < debuggerNumOfBreakpoints; i++) { 1.821 + debuggerBreakpointList[i-1].address = 1.822 + debuggerBreakpointList[i].address; 1.823 + debuggerBreakpointList[i-1].value = 1.824 + debuggerBreakpointList[i].value; 1.825 + debuggerBreakpointList[i-1].size = 1.826 + debuggerBreakpointList[i].size; 1.827 + } 1.828 + } 1.829 + debuggerNumOfBreakpoints--; 1.830 + } 1.831 + } else 1.832 + debuggerUsage("bd"); 1.833 +} 1.834 + 1.835 +void debuggerBreak(int n, char **args) 1.836 +{ 1.837 + if(n == 2) { 1.838 + u32 address = 0; 1.839 + u32 value = 0; 1.840 + int type = 0; 1.841 + char *s = args[1]; 1.842 + char c = *s; 1.843 + if(strchr(s, ':')) { 1.844 + char *name = s; 1.845 + char *l = strchr(s, ':'); 1.846 + *l++ = 0; 1.847 + int line = atoi(l); 1.848 + 1.849 + u32 addr; 1.850 + Function *f; 1.851 + CompileUnit *u; 1.852 + 1.853 + if(elfFindLineInModule(&addr, name, line)) { 1.854 + if(elfGetCurrentFunction(addr, &f, &u)) { 1.855 + u32 addr2; 1.856 + if(elfGetSymbolAddress(f->name, &addr2, &value, &type)) { 1.857 + address = addr; 1.858 + } else { 1.859 + printf("Unable to get function symbol data\n"); 1.860 + return; 1.861 + } 1.862 + } else { 1.863 + printf("Unable to find function for address\n"); 1.864 + return; 1.865 + } 1.866 + } else { 1.867 + printf("Unable to find module or line\n"); 1.868 + return; 1.869 + } 1.870 + } else if(c >= '0' && c <= '9') { 1.871 + int line = atoi(s); 1.872 + Function *f; 1.873 + CompileUnit *u; 1.874 + u32 addr; 1.875 + 1.876 + if(elfGetCurrentFunction(armNextPC, &f, &u)) { 1.877 + if(elfFindLineInUnit(&addr, u, line)) { 1.878 + if(elfGetCurrentFunction(addr, &f, &u)) { 1.879 + u32 addr2; 1.880 + if(elfGetSymbolAddress(f->name, &addr2, &value, &type)) { 1.881 + address = addr; 1.882 + } else { 1.883 + printf("Unable to get function symbol data\n"); 1.884 + return; 1.885 + } 1.886 + } else { 1.887 + printf("Unable to find function for address\n"); 1.888 + return; 1.889 + } 1.890 + } else { 1.891 + printf("Unable to find line\n"); 1.892 + return; 1.893 + } 1.894 + } else { 1.895 + printf("Cannot find current function\n"); 1.896 + return; 1.897 + } 1.898 + } else { 1.899 + if(!elfGetSymbolAddress(s, &address, &value, &type)) { 1.900 + printf("Function %s not found\n", args[1]); 1.901 + return; 1.902 + } 1.903 + } 1.904 + if(type == 0x02 || type == 0x0d) { 1.905 + int i = debuggerNumOfBreakpoints; 1.906 + int size = 0; 1.907 + if(type == 2) 1.908 + size = 1; 1.909 + debuggerBreakpointList[i].address = address; 1.910 + debuggerBreakpointList[i].value = type == 0x02 ? 1.911 + debuggerReadMemory(address) : debuggerReadHalfWord(address); 1.912 + debuggerBreakpointList[i].size = size; 1.913 + // debuggerApplyBreakpoint(address, i, size); 1.914 + debuggerNumOfBreakpoints++; 1.915 + if(size) 1.916 + printf("Added ARM breakpoint at %08x\n", address); 1.917 + else 1.918 + printf("Added THUMB breakpoint at %08x\n", address); 1.919 + } else { 1.920 + printf("%s is not a function symbol\n", args[1]); 1.921 + } 1.922 + } else 1.923 + debuggerUsage("break"); 1.924 +} 1.925 + 1.926 +void debuggerBreakThumb(int n, char **args) 1.927 +{ 1.928 + if(n == 2) { 1.929 + u32 address = 0; 1.930 + sscanf(args[1],"%x", &address); 1.931 + int i = debuggerNumOfBreakpoints; 1.932 + debuggerBreakpointList[i].address = address; 1.933 + debuggerBreakpointList[i].value = debuggerReadHalfWord(address); 1.934 + debuggerBreakpointList[i].size = 0; 1.935 + // debuggerApplyBreakpoint(address, i, 0); 1.936 + debuggerNumOfBreakpoints++; 1.937 + printf("Added THUMB breakpoint at %08x\n", address); 1.938 + } else 1.939 + debuggerUsage("bt"); 1.940 +} 1.941 + 1.942 +void debuggerBreakArm(int n, char **args) 1.943 +{ 1.944 + if(n == 2) { 1.945 + u32 address = 0; 1.946 + sscanf(args[1],"%x", &address); 1.947 + int i = debuggerNumOfBreakpoints; 1.948 + debuggerBreakpointList[i].address = address; 1.949 + debuggerBreakpointList[i].value = debuggerReadMemory(address); 1.950 + debuggerBreakpointList[i].size = 1; 1.951 + // debuggerApplyBreakpoint(address, i, 1); 1.952 + debuggerNumOfBreakpoints++; 1.953 + printf("Added ARM breakpoint at %08x\n", address); 1.954 + } else 1.955 + debuggerUsage("ba"); 1.956 +} 1.957 + 1.958 +void debuggerBreakOnWrite(u32 *mem, u32 oldvalue, u32 value, int size) 1.959 +{ 1.960 + u32 address = 0; 1.961 + if(mem >= (u32*)&workRAM[0] && mem <= (u32*)&workRAM[0x3ffff]) 1.962 + address = 0x2000000 + ((u64)mem - (u64)&workRAM[0]); 1.963 + else 1.964 + address = 0x3000000 + ((u64)mem - (u64)&internalRAM[0]); 1.965 + 1.966 + if(size == 2) 1.967 + printf("Breakpoint (on write) address %08x old:%08x new:%08x\n", 1.968 + address, oldvalue, value); 1.969 + else if(size == 1) 1.970 + printf("Breakpoint (on write) address %08x old:%04x new:%04x\n", 1.971 + address, (u16)oldvalue,(u16)value); 1.972 + else 1.973 + printf("Breakpoint (on write) address %08x old:%02x new:%02x\n", 1.974 + address, (u8)oldvalue, (u8)value); 1.975 + debugger = true; 1.976 +} 1.977 + 1.978 +void debuggerBreakWriteClear(int n, char **args) 1.979 +{ 1.980 + memset(freezeWorkRAM, false, 0x40000); 1.981 + memset(freezeInternalRAM, false, 0x8000); 1.982 + printf("Cleared all break on write\n"); 1.983 +} 1.984 + 1.985 +void debuggerBreakWrite(int n, char **args) 1.986 +{ 1.987 + if(n == 3) { 1.988 + if(cheatsNumber != 0) { 1.989 + printf("Cheats are enabled. Cannot continue.\n"); 1.990 + return; 1.991 + } 1.992 + u32 address = 0; 1.993 + sscanf(args[1], "%x", &address); 1.994 + int n = 0; 1.995 + sscanf(args[2], "%d", &n); 1.996 + 1.997 + if(address < 0x2000000 || address > 0x3007fff) { 1.998 + printf("Invalid address: %08x\n", address); 1.999 + return; 1.1000 + } 1.1001 + 1.1002 + if(address > 0x203ffff && address < 0x3000000) { 1.1003 + printf("Invalid address: %08x\n", address); 1.1004 + return; 1.1005 + } 1.1006 + 1.1007 + u32 final = address + n; 1.1008 + 1.1009 + if(address < 0x2040000 && final > 0x2040000) { 1.1010 + printf("Invalid byte count: %d\n", n); 1.1011 + return; 1.1012 + } else if(address < 0x3008000 && final > 0x3008000) { 1.1013 + printf("Invalid byte count: %d\n", n); 1.1014 + return; 1.1015 + } 1.1016 + printf("Added break on write at %08x for %d bytes\n", address, n); 1.1017 + for(int i = 0; i < n; i++) { 1.1018 + if((address >> 24) == 2) 1.1019 + freezeWorkRAM[address & 0x3ffff] = true; 1.1020 + else 1.1021 + freezeInternalRAM[address & 0x7fff] = true; 1.1022 + address++; 1.1023 + } 1.1024 + } else 1.1025 + debuggerUsage("bpw"); 1.1026 +} 1.1027 + 1.1028 +void debuggerDisassembleArm(int n, char **args) 1.1029 +{ 1.1030 + char buffer[80]; 1.1031 + u32 pc = reg[15].I; 1.1032 + pc -= 4; 1.1033 + int count = 20; 1.1034 + if(n >= 2) { 1.1035 + sscanf(args[1], "%x", &pc); 1.1036 + } 1.1037 + if(pc & 3) { 1.1038 + printf("Misaligned address %08x\n", pc); 1.1039 + pc &= 0xfffffffc; 1.1040 + } 1.1041 + if(n >= 3) { 1.1042 + sscanf(args[2], "%d", &count); 1.1043 + } 1.1044 + int i = 0; 1.1045 + int len = 0; 1.1046 + char format[30]; 1.1047 + for(i = 0; i < count; i++) { 1.1048 + int l = strlen(elfGetAddressSymbol(pc+4*i)); 1.1049 + if(l > len) 1.1050 + len = l; 1.1051 + } 1.1052 + sprintf(format, "%%08x %%-%ds %%s\n", len); 1.1053 + for(i = 0; i < count; i++) { 1.1054 + u32 addr = pc; 1.1055 + pc += disArm(pc, buffer, 2); 1.1056 + printf(format, addr, elfGetAddressSymbol(addr), buffer); 1.1057 + } 1.1058 +} 1.1059 + 1.1060 +void debuggerDisassembleThumb(int n, char **args) 1.1061 +{ 1.1062 + char buffer[80]; 1.1063 + u32 pc = reg[15].I; 1.1064 + pc -= 2; 1.1065 + int count = 20; 1.1066 + if(n >= 2) { 1.1067 + sscanf(args[1], "%x", &pc); 1.1068 + } 1.1069 + if(pc & 1) { 1.1070 + printf("Misaligned address %08x\n", pc); 1.1071 + pc &= 0xfffffffe; 1.1072 + } 1.1073 + if(n >= 3) { 1.1074 + sscanf(args[2], "%d", &count); 1.1075 + } 1.1076 + 1.1077 + int i = 0; 1.1078 + int len = 0; 1.1079 + char format[30]; 1.1080 + for(i = 0; i < count; i++) { 1.1081 + int l = strlen(elfGetAddressSymbol(pc+2*i)); 1.1082 + if(l > len) 1.1083 + len = l; 1.1084 + } 1.1085 + sprintf(format, "%%08x %%-%ds %%s\n", len); 1.1086 + 1.1087 + for(i = 0; i < count; i++) { 1.1088 + u32 addr = pc; 1.1089 + pc += disThumb(pc, buffer, 2); 1.1090 + printf(format, addr, elfGetAddressSymbol(addr), buffer); 1.1091 + } 1.1092 +} 1.1093 + 1.1094 +void debuggerDisassemble(int n, char **args) 1.1095 +{ 1.1096 + if(armState) 1.1097 + debuggerDisassembleArm(n, args); 1.1098 + else 1.1099 + debuggerDisassembleThumb(n, args); 1.1100 +} 1.1101 + 1.1102 +void debuggerContinueAfterBreakpoint() 1.1103 +{ 1.1104 + printf("Continuing after breakpoint\n"); 1.1105 + debuggerEnableBreakpoints(true); 1.1106 + theEmulator.emuMain(1); 1.1107 + debuggerAtBreakpoint = false; 1.1108 +} 1.1109 + 1.1110 +void debuggerRegisters(int, char **) 1.1111 +{ 1.1112 + char *command[3]; 1.1113 + char buffer[10]; 1.1114 + 1.1115 + printf("R00=%08x R04=%08x R08=%08x R12=%08x\n", 1.1116 + reg[0].I, reg[4].I, reg[8].I, reg[12].I); 1.1117 + printf("R01=%08x R05=%08x R09=%08x R13=%08x\n", 1.1118 + reg[1].I, reg[5].I, reg[9].I, reg[13].I); 1.1119 + printf("R02=%08x R06=%08x R10=%08x R14=%08x\n", 1.1120 + reg[2].I, reg[6].I, reg[10].I, reg[14].I); 1.1121 + printf("R03=%08x R07=%08x R11=%08x R15=%08x\n", 1.1122 + reg[3].I, reg[7].I, reg[11].I, reg[15].I); 1.1123 + printf("CPSR=%08x (%c%c%c%c%c%c%c Mode: %02x)\n", 1.1124 + reg[16].I, 1.1125 + (N_FLAG ? 'N' : '.'), 1.1126 + (Z_FLAG ? 'Z' : '.'), 1.1127 + (C_FLAG ? 'C' : '.'), 1.1128 + (V_FLAG ? 'V' : '.'), 1.1129 + (armIrqEnable ? '.' : 'I'), 1.1130 + ((!(reg[16].I & 0x40)) ? '.' : 'F'), 1.1131 + (armState ? '.' : 'T'), 1.1132 + armMode); 1.1133 + sprintf(buffer,"%08x", armState ? reg[15].I - 4 : reg[15].I - 2); 1.1134 + command[0]=const_cast<char *>("m"); 1.1135 + command[1]=buffer; 1.1136 + command[2]=const_cast<char *>("1"); 1.1137 + debuggerDisassemble(3, command); 1.1138 +} 1.1139 + 1.1140 +void debuggerIoVideo() 1.1141 +{ 1.1142 + printf("DISPCNT = %04x\n", DISPCNT); 1.1143 + printf("DISPSTAT = %04x\n", DISPSTAT); 1.1144 + printf("VCOUNT = %04x\n", VCOUNT); 1.1145 + printf("BG0CNT = %04x\n", BG0CNT); 1.1146 + printf("BG1CNT = %04x\n", BG1CNT); 1.1147 + printf("BG2CNT = %04x\n", BG2CNT); 1.1148 + printf("BG3CNT = %04x\n", BG3CNT); 1.1149 + printf("WIN0H = %04x\n", WIN0H); 1.1150 + printf("WIN0V = %04x\n", WIN0V); 1.1151 + printf("WIN1H = %04x\n", WIN1H); 1.1152 + printf("WIN1V = %04x\n", WIN1V); 1.1153 + printf("WININ = %04x\n", WININ); 1.1154 + printf("WINOUT = %04x\n", WINOUT); 1.1155 + printf("MOSAIC = %04x\n", MOSAIC); 1.1156 + printf("BLDMOD = %04x\n", BLDMOD); 1.1157 + printf("COLEV = %04x\n", COLEV); 1.1158 + printf("COLY = %04x\n", COLY); 1.1159 +} 1.1160 + 1.1161 +void debuggerIoVideo2() 1.1162 +{ 1.1163 + printf("BG0HOFS = %04x\n", BG0HOFS); 1.1164 + printf("BG0VOFS = %04x\n", BG0VOFS); 1.1165 + printf("BG1HOFS = %04x\n", BG1HOFS); 1.1166 + printf("BG1VOFS = %04x\n", BG1VOFS); 1.1167 + printf("BG2HOFS = %04x\n", BG2HOFS); 1.1168 + printf("BG2VOFS = %04x\n", BG2VOFS); 1.1169 + printf("BG3HOFS = %04x\n", BG3HOFS); 1.1170 + printf("BG3VOFS = %04x\n", BG3VOFS); 1.1171 + printf("BG2PA = %04x\n", BG2PA); 1.1172 + printf("BG2PB = %04x\n", BG2PB); 1.1173 + printf("BG2PC = %04x\n", BG2PC); 1.1174 + printf("BG2PD = %04x\n", BG2PD); 1.1175 + printf("BG2X = %08x\n", (BG2X_H<<16)|BG2X_L); 1.1176 + printf("BG2Y = %08x\n", (BG2Y_H<<16)|BG2Y_L); 1.1177 + printf("BG3PA = %04x\n", BG3PA); 1.1178 + printf("BG3PB = %04x\n", BG3PB); 1.1179 + printf("BG3PC = %04x\n", BG3PC); 1.1180 + printf("BG3PD = %04x\n", BG3PD); 1.1181 + printf("BG3X = %08x\n", (BG3X_H<<16)|BG3X_L); 1.1182 + printf("BG3Y = %08x\n", (BG3Y_H<<16)|BG3Y_L); 1.1183 +} 1.1184 + 1.1185 +void debuggerIoDMA() 1.1186 +{ 1.1187 + printf("DM0SAD = %08x\n", (DM0SAD_H<<16)|DM0SAD_L); 1.1188 + printf("DM0DAD = %08x\n", (DM0DAD_H<<16)|DM0DAD_L); 1.1189 + printf("DM0CNT = %08x\n", (DM0CNT_H<<16)|DM0CNT_L); 1.1190 + printf("DM1SAD = %08x\n", (DM1SAD_H<<16)|DM1SAD_L); 1.1191 + printf("DM1DAD = %08x\n", (DM1DAD_H<<16)|DM1DAD_L); 1.1192 + printf("DM1CNT = %08x\n", (DM1CNT_H<<16)|DM1CNT_L); 1.1193 + printf("DM2SAD = %08x\n", (DM2SAD_H<<16)|DM2SAD_L); 1.1194 + printf("DM2DAD = %08x\n", (DM2DAD_H<<16)|DM2DAD_L); 1.1195 + printf("DM2CNT = %08x\n", (DM2CNT_H<<16)|DM2CNT_L); 1.1196 + printf("DM3SAD = %08x\n", (DM3SAD_H<<16)|DM3SAD_L); 1.1197 + printf("DM3DAD = %08x\n", (DM3DAD_H<<16)|DM3DAD_L); 1.1198 + printf("DM3CNT = %08x\n", (DM3CNT_H<<16)|DM3CNT_L); 1.1199 +} 1.1200 + 1.1201 +void debuggerIoTimer() 1.1202 +{ 1.1203 + printf("TM0D = %04x\n", TM0D); 1.1204 + printf("TM0CNT = %04x\n", TM0CNT); 1.1205 + printf("TM1D = %04x\n", TM1D); 1.1206 + printf("TM1CNT = %04x\n", TM1CNT); 1.1207 + printf("TM2D = %04x\n", TM2D); 1.1208 + printf("TM2CNT = %04x\n", TM2CNT); 1.1209 + printf("TM3D = %04x\n", TM3D); 1.1210 + printf("TM3CNT = %04x\n", TM3CNT); 1.1211 +} 1.1212 + 1.1213 +void debuggerIoMisc() 1.1214 +{ 1.1215 + printf("P1 = %04x\n", P1); 1.1216 + printf("IE = %04x\n", IE); 1.1217 + printf("IF = %04x\n", IF); 1.1218 + printf("IME = %04x\n", IME); 1.1219 +} 1.1220 + 1.1221 +void debuggerIo(int n, char **args) 1.1222 +{ 1.1223 + if(n == 1) { 1.1224 + debuggerIoVideo(); 1.1225 + return; 1.1226 + } 1.1227 + if(!strcmp(args[1], "video")) 1.1228 + debuggerIoVideo(); 1.1229 + else if(!strcmp(args[1], "video2")) 1.1230 + debuggerIoVideo2(); 1.1231 + else if(!strcmp(args[1], "dma")) 1.1232 + debuggerIoDMA(); 1.1233 + else if(!strcmp(args[1], "timer")) 1.1234 + debuggerIoTimer(); 1.1235 + else if(!strcmp(args[1], "misc")) 1.1236 + debuggerIoMisc(); 1.1237 + else printf("Unrecognized option %s\n", args[1]); 1.1238 +} 1.1239 + 1.1240 +void debuggerEditByte(int n, char **args) 1.1241 +{ 1.1242 + if(n == 3) { 1.1243 + u32 address; 1.1244 + u32 byte; 1.1245 + sscanf(args[1], "%x", &address); 1.1246 + sscanf(args[2], "%x", &byte); 1.1247 + debuggerWriteByte(address, (u8)byte); 1.1248 + } else 1.1249 + debuggerUsage("eb"); 1.1250 +} 1.1251 + 1.1252 +void debuggerEditHalfWord(int n, char **args) 1.1253 +{ 1.1254 + if(n == 3) { 1.1255 + u32 address; 1.1256 + u32 byte; 1.1257 + sscanf(args[1], "%x", &address); 1.1258 + if(address & 1) { 1.1259 + printf("Error: address must be half-word aligned\n"); 1.1260 + return; 1.1261 + } 1.1262 + sscanf(args[2], "%x", &byte); 1.1263 + debuggerWriteHalfWord(address, (u16)byte); 1.1264 + } else 1.1265 + debuggerUsage("eh"); 1.1266 +} 1.1267 + 1.1268 +void debuggerEdit(int n, char **args) 1.1269 +{ 1.1270 + if(n == 3) { 1.1271 + u32 address; 1.1272 + u32 byte; 1.1273 + sscanf(args[1], "%x", &address); 1.1274 + if(address & 3) { 1.1275 + printf("Error: address must be word aligned\n"); 1.1276 + return; 1.1277 + } 1.1278 + sscanf(args[2], "%x", &byte); 1.1279 + debuggerWriteMemory(address, (u32)byte); 1.1280 + } else 1.1281 + debuggerUsage("ew"); 1.1282 +} 1.1283 + 1.1284 + 1.1285 +#define ASCII(c) (c) < 32 ? '.' : (c) > 127 ? '.' : (c) 1.1286 + 1.1287 +void debuggerMemoryByte(int n, char **args) 1.1288 +{ 1.1289 + if(n == 2) { 1.1290 + u32 addr = 0; 1.1291 + sscanf(args[1], "%x", &addr); 1.1292 + for(int ii = 0; ii < 16; ii++) { 1.1293 + int a = debuggerReadByte(addr); 1.1294 + int b = debuggerReadByte(addr+1); 1.1295 + int c = debuggerReadByte(addr+2); 1.1296 + int d = debuggerReadByte(addr+3); 1.1297 + int e = debuggerReadByte(addr+4); 1.1298 + int f = debuggerReadByte(addr+5); 1.1299 + int g = debuggerReadByte(addr+6); 1.1300 + int h = debuggerReadByte(addr+7); 1.1301 + int i = debuggerReadByte(addr+8); 1.1302 + int j = debuggerReadByte(addr+9); 1.1303 + int k = debuggerReadByte(addr+10); 1.1304 + int l = debuggerReadByte(addr+11); 1.1305 + int m = debuggerReadByte(addr+12); 1.1306 + int n = debuggerReadByte(addr+13); 1.1307 + int o = debuggerReadByte(addr+14); 1.1308 + int p = debuggerReadByte(addr+15); 1.1309 + 1.1310 + printf("%08x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %02x %c%c%c%c%c%c%c%c%c%c%c%c%c%c%c%c\n", 1.1311 + addr,a,b,c,d,e,f,g,h,i,j,k,l,m,n,o,p, 1.1312 + ASCII(a),ASCII(b),ASCII(c),ASCII(d), 1.1313 + ASCII(e),ASCII(f),ASCII(g),ASCII(h), 1.1314 + ASCII(i),ASCII(j),ASCII(k),ASCII(l), 1.1315 + ASCII(m),ASCII(n),ASCII(o),ASCII(p)); 1.1316 + addr += 16; 1.1317 + } 1.1318 + } else 1.1319 + debuggerUsage("mb"); 1.1320 +} 1.1321 + 1.1322 +void debuggerMemoryHalfWord(int n, char **args) 1.1323 +{ 1.1324 + if(n == 2) { 1.1325 + u32 addr = 0; 1.1326 + sscanf(args[1], "%x", &addr); 1.1327 + addr = addr & 0xfffffffe; 1.1328 + for(int ii = 0; ii < 16; ii++) { 1.1329 + int a = debuggerReadByte(addr); 1.1330 + int b = debuggerReadByte(addr+1); 1.1331 + int c = debuggerReadByte(addr+2); 1.1332 + int d = debuggerReadByte(addr+3); 1.1333 + int e = debuggerReadByte(addr+4); 1.1334 + int f = debuggerReadByte(addr+5); 1.1335 + int g = debuggerReadByte(addr+6); 1.1336 + int h = debuggerReadByte(addr+7); 1.1337 + int i = debuggerReadByte(addr+8); 1.1338 + int j = debuggerReadByte(addr+9); 1.1339 + int k = debuggerReadByte(addr+10); 1.1340 + int l = debuggerReadByte(addr+11); 1.1341 + int m = debuggerReadByte(addr+12); 1.1342 + int n = debuggerReadByte(addr+13); 1.1343 + int o = debuggerReadByte(addr+14); 1.1344 + int p = debuggerReadByte(addr+15); 1.1345 + 1.1346 + printf("%08x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %02x%02x %c%c%c%c%c%c%c%c%c%c%c%c%c%c%c%c\n", 1.1347 + addr,b,a,d,c,f,e,h,g,j,i,l,k,n,m,p,o, 1.1348 + ASCII(a),ASCII(b),ASCII(c),ASCII(d), 1.1349 + ASCII(e),ASCII(f),ASCII(g),ASCII(h), 1.1350 + ASCII(i),ASCII(j),ASCII(k),ASCII(l), 1.1351 + ASCII(m),ASCII(n),ASCII(o),ASCII(p)); 1.1352 + addr += 16; 1.1353 + } 1.1354 + } else 1.1355 + debuggerUsage("mh"); 1.1356 +} 1.1357 + 1.1358 +void debuggerMemory(int n, char **args) 1.1359 +{ 1.1360 + if(n == 2) { 1.1361 + u32 addr = 0; 1.1362 + sscanf(args[1], "%x", &addr); 1.1363 + addr = addr & 0xfffffffc; 1.1364 + for(int ii = 0; ii < 16; ii++) { 1.1365 + int a = debuggerReadByte(addr); 1.1366 + int b = debuggerReadByte(addr+1); 1.1367 + int c = debuggerReadByte(addr+2); 1.1368 + int d = debuggerReadByte(addr+3); 1.1369 + 1.1370 + int e = debuggerReadByte(addr+4); 1.1371 + int f = debuggerReadByte(addr+5); 1.1372 + int g = debuggerReadByte(addr+6); 1.1373 + int h = debuggerReadByte(addr+7); 1.1374 + 1.1375 + int i = debuggerReadByte(addr+8); 1.1376 + int j = debuggerReadByte(addr+9); 1.1377 + int k = debuggerReadByte(addr+10); 1.1378 + int l = debuggerReadByte(addr+11); 1.1379 + 1.1380 + int m = debuggerReadByte(addr+12); 1.1381 + int n = debuggerReadByte(addr+13); 1.1382 + int o = debuggerReadByte(addr+14); 1.1383 + int p = debuggerReadByte(addr+15); 1.1384 + 1.1385 + printf("%08x %02x%02x%02x%02x %02x%02x%02x%02x %02x%02x%02x%02x %02x%02x%02x%02x %c%c%c%c%c%c%c%c%c%c%c%c%c%c%c%c\n", 1.1386 + addr,d,c,b,a,h,g,f,e,l,k,j,i,p,o,n,m, 1.1387 + ASCII(a),ASCII(b),ASCII(c),ASCII(d), 1.1388 + ASCII(e),ASCII(f),ASCII(g),ASCII(h), 1.1389 + ASCII(i),ASCII(j),ASCII(k),ASCII(l), 1.1390 + ASCII(m),ASCII(n),ASCII(o),ASCII(p)); 1.1391 + addr += 16; 1.1392 + } 1.1393 + } else 1.1394 + debuggerUsage("mw"); 1.1395 +} 1.1396 + 1.1397 +void debuggerQuit(int, char **) 1.1398 +{ 1.1399 + char buffer[10]; 1.1400 + printf("Are you sure you want to quit (y/n)? "); 1.1401 + fgets(buffer, 1024, stdin); 1.1402 + 1.1403 + if(buffer[0] == 'y' || buffer[0] == 'Y') { 1.1404 + debugger = false; 1.1405 + emulating = false; 1.1406 + } 1.1407 +} 1.1408 + 1.1409 +void debuggerOutput(char *s, u32 addr) 1.1410 +{ 1.1411 + if(s) 1.1412 + puts(s); 1.1413 + else { 1.1414 + char c; 1.1415 + 1.1416 + c = debuggerReadByte(addr); 1.1417 + addr++; 1.1418 + while(c) { 1.1419 + putchar(c); 1.1420 + c = debuggerReadByte(addr); 1.1421 + addr++; 1.1422 + } 1.1423 + } 1.1424 +} 1.1425 + 1.1426 +void debuggerMain() 1.1427 +{ 1.1428 + char buffer[1024]; 1.1429 + char *commands[10]; 1.1430 + int commandCount = 0; 1.1431 + 1.1432 + if(theEmulator.emuUpdateCPSR) 1.1433 + theEmulator.emuUpdateCPSR(); 1.1434 + debuggerRegisters(0, NULL); 1.1435 + 1.1436 + while(debugger) { 1.1437 + systemSoundPause(); 1.1438 + printf("debugger> "); 1.1439 + commandCount = 0; 1.1440 + char *s = fgets(buffer, 1024, stdin); 1.1441 + 1.1442 + commands[0] = strtok(s, " \t\n"); 1.1443 + if(commands[0] == NULL) 1.1444 + continue; 1.1445 + commandCount++; 1.1446 + while((s = strtok(NULL, " \t\n"))) { 1.1447 + commands[commandCount++] = s; 1.1448 + if(commandCount == 10) 1.1449 + break; 1.1450 + } 1.1451 + 1.1452 + for(int j = 0; ; j++) { 1.1453 + if(debuggerCommands[j].name == NULL) { 1.1454 + printf("Unrecognized command %s. Type h for help.\n", commands[0]); 1.1455 + break; 1.1456 + } 1.1457 + if(!strcmp(commands[0], debuggerCommands[j].name)) { 1.1458 + debuggerCommands[j].function(commandCount, commands); 1.1459 + break; 1.1460 + } 1.1461 + } 1.1462 + } 1.1463 +}