mirror of https://github.com/stella-emu/stella.git
Implemented "disassemble" command in prompt, in case you want to see more
code than the current line. On ROM load, we replace the file extension with .sym and attempt to load that as a symbol file for the debugger. No harm is done if the file is missing. git-svn-id: svn://svn.code.sf.net/p/stella/code/trunk@521 8b62c5a3-ac7e-4cc8-8f21-d9a121418aba
This commit is contained in:
parent
09bb356f52
commit
eaa8fa898b
|
@ -13,7 +13,7 @@
|
||||||
// See the file "license" for information on usage and redistribution of
|
// See the file "license" for information on usage and redistribution of
|
||||||
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||||||
//
|
//
|
||||||
// $Id: Debugger.cxx,v 1.13 2005-06-17 03:49:08 urchlay Exp $
|
// $Id: Debugger.cxx,v 1.14 2005-06-17 21:59:53 urchlay Exp $
|
||||||
//============================================================================
|
//============================================================================
|
||||||
|
|
||||||
#include "bspf.hxx"
|
#include "bspf.hxx"
|
||||||
|
@ -89,6 +89,22 @@ void Debugger::setConsole(Console* console)
|
||||||
// Create a new 6502 debugger for this console
|
// Create a new 6502 debugger for this console
|
||||||
delete myDebugger;
|
delete myDebugger;
|
||||||
myDebugger = new D6502(mySystem);
|
myDebugger = new D6502(mySystem);
|
||||||
|
|
||||||
|
autoLoadSymbols(myOSystem->romFile());
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
void Debugger::autoLoadSymbols(string fileName) {
|
||||||
|
string file = fileName;
|
||||||
|
|
||||||
|
string::size_type pos;
|
||||||
|
if( (pos = file.find_last_of('.')) != string::npos ) {
|
||||||
|
file.replace(pos, file.size(), ".sym");
|
||||||
|
} else {
|
||||||
|
file += ".sym";
|
||||||
|
}
|
||||||
|
string ret = equateList->loadFile(file);
|
||||||
|
// cerr << "loading syms from file " << file << ": " << ret << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
@ -101,7 +117,7 @@ const string Debugger::run(const string& command)
|
||||||
const string Debugger::state()
|
const string Debugger::state()
|
||||||
{
|
{
|
||||||
string result;
|
string result;
|
||||||
char buf[255], bbuf[255];
|
char buf[255];
|
||||||
|
|
||||||
//cerr << "state(): pc is " << myDebugger->pc() << endl;
|
//cerr << "state(): pc is " << myDebugger->pc() << endl;
|
||||||
result += "\nPC=";
|
result += "\nPC=";
|
||||||
|
@ -123,21 +139,8 @@ const string Debugger::state()
|
||||||
sprintf(buf, "%d", mySystem->cycles());
|
sprintf(buf, "%d", mySystem->cycles());
|
||||||
result += buf;
|
result += buf;
|
||||||
result += "\n ";
|
result += "\n ";
|
||||||
char *label = equateList->getLabel(myDebugger->pc());
|
|
||||||
if(label != NULL) {
|
|
||||||
result += label;
|
|
||||||
result += ": ";
|
|
||||||
}
|
|
||||||
int count = myDebugger->disassemble(myDebugger->pc(), buf, equateList);
|
|
||||||
for(int i=0; i<count; i++) {
|
|
||||||
sprintf(bbuf, "%02x ", readRAM(myDebugger->pc() + i));
|
|
||||||
result += bbuf;
|
|
||||||
}
|
|
||||||
if(count < 3) result += " ";
|
|
||||||
if(count < 2) result += " ";
|
|
||||||
result += " ";
|
|
||||||
result += buf;
|
|
||||||
|
|
||||||
|
result += disassemble(myDebugger->pc(), 1);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -389,3 +392,36 @@ bool Debugger::breakPoint(int bp) {
|
||||||
return breakPoints->isSet(bp) != 0;
|
return breakPoints->isSet(bp) != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
int Debugger::getPC() {
|
||||||
|
return myDebugger->pc();
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
string Debugger::disassemble(int start, int lines) {
|
||||||
|
char buf[255], bbuf[255];
|
||||||
|
string result;
|
||||||
|
|
||||||
|
do {
|
||||||
|
char *label = equateList->getFormatted(start, 4);
|
||||||
|
|
||||||
|
result += label;
|
||||||
|
result += ": ";
|
||||||
|
|
||||||
|
int count = myDebugger->disassemble(start, buf, equateList);
|
||||||
|
|
||||||
|
for(int i=0; i<count; i++) {
|
||||||
|
sprintf(bbuf, "%02x ", readRAM(start++));
|
||||||
|
result += bbuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(count < 3) result += " ";
|
||||||
|
if(count < 2) result += " ";
|
||||||
|
|
||||||
|
result += " ";
|
||||||
|
result += buf;
|
||||||
|
result += "\n";
|
||||||
|
} while(--lines > 0);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
// See the file "license" for information on usage and redistribution of
|
// See the file "license" for information on usage and redistribution of
|
||||||
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||||||
//
|
//
|
||||||
// $Id: Debugger.hxx,v 1.11 2005-06-17 17:34:01 stephena Exp $
|
// $Id: Debugger.hxx,v 1.12 2005-06-17 21:59:53 urchlay Exp $
|
||||||
//============================================================================
|
//============================================================================
|
||||||
|
|
||||||
#ifndef DEBUGGER_HXX
|
#ifndef DEBUGGER_HXX
|
||||||
|
@ -49,7 +49,7 @@ enum {
|
||||||
for all debugging operations in Stella (parser, 6502 debugger, etc).
|
for all debugging operations in Stella (parser, 6502 debugger, etc).
|
||||||
|
|
||||||
@author Stephen Anthony
|
@author Stephen Anthony
|
||||||
@version $Id: Debugger.hxx,v 1.11 2005-06-17 17:34:01 stephena Exp $
|
@version $Id: Debugger.hxx,v 1.12 2005-06-17 21:59:53 urchlay Exp $
|
||||||
*/
|
*/
|
||||||
class Debugger : public DialogContainer
|
class Debugger : public DialogContainer
|
||||||
{
|
{
|
||||||
|
@ -108,6 +108,7 @@ class Debugger : public DialogContainer
|
||||||
|
|
||||||
void toggleBreakPoint(int bp);
|
void toggleBreakPoint(int bp);
|
||||||
bool breakPoint(int bp);
|
bool breakPoint(int bp);
|
||||||
|
string disassemble(int start, int lines);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/**
|
/**
|
||||||
|
@ -144,12 +145,14 @@ class Debugger : public DialogContainer
|
||||||
void setY(int y);
|
void setY(int y);
|
||||||
void setS(int sp);
|
void setS(int sp);
|
||||||
void setPC(int pc);
|
void setPC(int pc);
|
||||||
|
int getPC();
|
||||||
void toggleC();
|
void toggleC();
|
||||||
void toggleZ();
|
void toggleZ();
|
||||||
void toggleN();
|
void toggleN();
|
||||||
void toggleV();
|
void toggleV();
|
||||||
void toggleD();
|
void toggleD();
|
||||||
void reset();
|
void reset();
|
||||||
|
void autoLoadSymbols(string file);
|
||||||
|
|
||||||
void formatFlags(int f, char *out);
|
void formatFlags(int f, char *out);
|
||||||
EquateList *equates();
|
EquateList *equates();
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
// See the file "license" for information on usage and redistribution of
|
// See the file "license" for information on usage and redistribution of
|
||||||
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||||||
//
|
//
|
||||||
// $Id: DebuggerParser.cxx,v 1.9 2005-06-17 03:49:08 urchlay Exp $
|
// $Id: DebuggerParser.cxx,v 1.10 2005-06-17 21:59:53 urchlay Exp $
|
||||||
//============================================================================
|
//============================================================================
|
||||||
|
|
||||||
#include "bspf.hxx"
|
#include "bspf.hxx"
|
||||||
|
@ -188,6 +188,23 @@ string DebuggerParser::listBreaks() {
|
||||||
return "no breakpoints set";
|
return "no breakpoints set";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
string DebuggerParser::disasm() {
|
||||||
|
int start, lines = 20;
|
||||||
|
|
||||||
|
if(argCount == 0) {
|
||||||
|
start = debugger->getPC();
|
||||||
|
} else if(argCount == 1) {
|
||||||
|
start = args[0];
|
||||||
|
} else if(argCount == 2) {
|
||||||
|
start = args[0];
|
||||||
|
lines = args[1];
|
||||||
|
} else {
|
||||||
|
return "wrong number of arguments";
|
||||||
|
}
|
||||||
|
|
||||||
|
return debugger->disassemble(start, lines);
|
||||||
|
}
|
||||||
|
|
||||||
string DebuggerParser::run(const string& command) {
|
string DebuggerParser::run(const string& command) {
|
||||||
string result;
|
string result;
|
||||||
|
|
||||||
|
@ -302,6 +319,8 @@ string DebuggerParser::run(const string& command) {
|
||||||
return "Cleared breakpoint";
|
return "Cleared breakpoint";
|
||||||
} else if(subStringMatch(verb, "listbreaks")) {
|
} else if(subStringMatch(verb, "listbreaks")) {
|
||||||
return listBreaks();
|
return listBreaks();
|
||||||
|
} else if(subStringMatch(verb, "disasm")) {
|
||||||
|
return disasm();
|
||||||
} else if(subStringMatch(verb, "clearbreaks")) {
|
} else if(subStringMatch(verb, "clearbreaks")) {
|
||||||
//debugger->clearAllBreakPoints();
|
//debugger->clearAllBreakPoints();
|
||||||
return "cleared all breakpoints";
|
return "cleared all breakpoints";
|
||||||
|
@ -309,12 +328,15 @@ string DebuggerParser::run(const string& command) {
|
||||||
// please leave each option on its own line so they're
|
// please leave each option on its own line so they're
|
||||||
// easy to sort - bkw
|
// easy to sort - bkw
|
||||||
return
|
return
|
||||||
|
"Commands are case-insensitive and may be abbreviated.\n"
|
||||||
"a xx - Set Accumulator to xx\n"
|
"a xx - Set Accumulator to xx\n"
|
||||||
"break - Set/clear breakpoint at current PC\n"
|
"break - Set/clear breakpoint at current PC\n"
|
||||||
"break xx - Set/clear breakpoint at address xx\n"
|
"break xx - Set/clear breakpoint at address xx\n"
|
||||||
"c - Toggle Carry Flag\n"
|
"c - Toggle Carry Flag\n"
|
||||||
"clearbreaks - Clear all breakpoints\n"
|
"clearbreaks - Clear all breakpoints\n"
|
||||||
"d - Toggle Decimal Flag\n"
|
"d - Toggle Decimal Flag\n"
|
||||||
|
"disasm - Disassemble (from current PC)\n"
|
||||||
|
"disasm xx - Disassemble (from address xx)\n"
|
||||||
"listbreaks - List all breakpoints\n"
|
"listbreaks - List all breakpoints\n"
|
||||||
"loadsym f - Load DASM symbols from file f\n"
|
"loadsym f - Load DASM symbols from file f\n"
|
||||||
"n - Toggle Negative Flag\n"
|
"n - Toggle Negative Flag\n"
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
// See the file "license" for information on usage and redistribution of
|
// See the file "license" for information on usage and redistribution of
|
||||||
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||||||
//
|
//
|
||||||
// $Id: DebuggerParser.hxx,v 1.5 2005-06-17 03:49:08 urchlay Exp $
|
// $Id: DebuggerParser.hxx,v 1.6 2005-06-17 21:59:53 urchlay Exp $
|
||||||
//============================================================================
|
//============================================================================
|
||||||
|
|
||||||
#ifndef DEBUGGER_PARSER_HXX
|
#ifndef DEBUGGER_PARSER_HXX
|
||||||
|
@ -34,12 +34,13 @@ class DebuggerParser
|
||||||
void setDone();
|
void setDone();
|
||||||
string run(const string& command);
|
string run(const string& command);
|
||||||
bool getArgs(const string& command);
|
bool getArgs(const string& command);
|
||||||
string listBreaks();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int DebuggerParser::conv_hex_digit(char d);
|
int DebuggerParser::conv_hex_digit(char d);
|
||||||
bool DebuggerParser::subStringMatch(const string& needle, const string& haystack);
|
bool DebuggerParser::subStringMatch(const string& needle, const string& haystack);
|
||||||
int decipher_arg(string &arg);
|
int decipher_arg(string &arg);
|
||||||
|
string disasm();
|
||||||
|
string listBreaks();
|
||||||
|
|
||||||
Debugger* debugger;
|
Debugger* debugger;
|
||||||
EquateList *equateList;
|
EquateList *equateList;
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
// See the file "license" for information on usage and redistribution of
|
// See the file "license" for information on usage and redistribution of
|
||||||
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||||||
//
|
//
|
||||||
// $Id: OSystem.hxx,v 1.21 2005-06-16 01:11:28 stephena Exp $
|
// $Id: OSystem.hxx,v 1.22 2005-06-17 21:59:53 urchlay Exp $
|
||||||
//============================================================================
|
//============================================================================
|
||||||
|
|
||||||
#ifndef OSYSTEM_HXX
|
#ifndef OSYSTEM_HXX
|
||||||
|
@ -44,7 +44,7 @@ class Debugger;
|
||||||
other objects belong.
|
other objects belong.
|
||||||
|
|
||||||
@author Stephen Anthony
|
@author Stephen Anthony
|
||||||
@version $Id: OSystem.hxx,v 1.21 2005-06-16 01:11:28 stephena Exp $
|
@version $Id: OSystem.hxx,v 1.22 2005-06-17 21:59:53 urchlay Exp $
|
||||||
*/
|
*/
|
||||||
class OSystem
|
class OSystem
|
||||||
{
|
{
|
||||||
|
@ -282,6 +282,8 @@ class OSystem
|
||||||
*/
|
*/
|
||||||
bool openROM(const string& rom, uInt8** image, int* size);
|
bool openROM(const string& rom, uInt8** image, int* size);
|
||||||
|
|
||||||
|
const string& romFile() { return myRomFile; }
|
||||||
|
|
||||||
public:
|
public:
|
||||||
//////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////
|
||||||
// The following methods are system-specific and must be implemented
|
// The following methods are system-specific and must be implemented
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
// See the file "license" for information on usage and redistribution of
|
// See the file "license" for information on usage and redistribution of
|
||||||
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
||||||
//
|
//
|
||||||
// $Id: PromptWidget.cxx,v 1.8 2005-06-17 03:49:10 urchlay Exp $
|
// $Id: PromptWidget.cxx,v 1.9 2005-06-17 21:59:54 urchlay Exp $
|
||||||
//
|
//
|
||||||
// Based on code from ScummVM - Scumm Interpreter
|
// Based on code from ScummVM - Scumm Interpreter
|
||||||
// Copyright (C) 2002-2004 The ScummVM project
|
// Copyright (C) 2002-2004 The ScummVM project
|
||||||
|
@ -138,7 +138,7 @@ void PromptWidget::handleMouseWheel(int x, int y, int direction)
|
||||||
}
|
}
|
||||||
|
|
||||||
void PromptWidget::printPrompt() {
|
void PromptWidget::printPrompt() {
|
||||||
print( instance()->debugger().state() + "\n");
|
print( instance()->debugger().state() );
|
||||||
print(PROMPT);
|
print(PROMPT);
|
||||||
_promptStartPos = _promptEndPos = _currentPos;
|
_promptStartPos = _promptEndPos = _currentPos;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue