Merge branch 'trapif_attempt_1'

This commit is contained in:
thrust26 2017-10-08 18:31:26 +02:00
commit 20babc832a
11 changed files with 563 additions and 148 deletions

View File

@ -30,6 +30,7 @@ class CartDebugWidget;
#include "Cart.hxx" #include "Cart.hxx"
#include "DebuggerSystem.hxx" #include "DebuggerSystem.hxx"
#include "System.hxx" #include "System.hxx"
#include "M6502.hxx"
// Function type for CartDebug instance methods // Function type for CartDebug instance methods
class CartDebug; class CartDebug;
@ -127,6 +128,11 @@ class CartDebug : public DebuggerSystem
// write port area. // write port area.
int readFromWritePort(); int readFromWritePort();
// Return the base (= non-mirrored) address of the last CPU read
int lastReadBaseAddress() { return mySystem.m6502().lastReadBaseAddress(); }
// Return the base (= non-mirrored) address of the last CPU write
int lastWriteBaseAddress() { return mySystem.m6502().lastWriteBaseAddress(); }
// The following two methods are meant to be used together // The following two methods are meant to be used together
// First, a call is made to disassemble(), which updates the disassembly // First, a call is made to disassemble(), which updates the disassembly
// list; it will figure out when an actual complete disassembly is // list; it will figure out when an actual complete disassembly is

View File

@ -85,7 +85,6 @@ static const char* const builtin_functions[][3] = {
{ "_diff0a", "*SWCHB & $40", "Left diff. set to A (hard)" }, { "_diff0a", "*SWCHB & $40", "Left diff. set to A (hard)" },
{ "_diff1b", "!(*SWCHB & $80)", "Right diff. set to B (easy)" }, { "_diff1b", "!(*SWCHB & $80)", "Right diff. set to B (easy)" },
{ "_diff1a", "*SWCHB & $80", "Right diff. set to A (hard)" }, { "_diff1a", "*SWCHB & $80", "Right diff. set to A (hard)" },
// empty string marks end of list, do not remove // empty string marks end of list, do not remove
{ 0, 0, 0 } { 0, 0, 0 }
}; };
@ -105,6 +104,9 @@ static const char* const pseudo_registers[][2] = {
{ "_scan", "Current scanline count" }, { "_scan", "Current scanline count" },
{ "_vblank", "Whether vertical blank is enabled (1 or 0)" }, { "_vblank", "Whether vertical blank is enabled (1 or 0)" },
{ "_vsync", "Whether vertical sync is enabled (1 or 0)" }, { "_vsync", "Whether vertical sync is enabled (1 or 0)" },
// CPU address access functions:
/*{ "__lastread", "last CPU read address" },
{ "__lastwrite", "last CPU write address" },*/
// empty string marks end of list, do not remove // empty string marks end of list, do not remove
{ 0, 0 } { 0, 0 }
@ -166,7 +168,7 @@ FBInitStatus Debugger::initializeVideo()
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
bool Debugger::start(const string& message, int address) bool Debugger::start(const string& message, int address, bool read)
{ {
if(myOSystem.eventHandler().enterDebugMode()) if(myOSystem.eventHandler().enterDebugMode())
{ {
@ -175,8 +177,7 @@ bool Debugger::start(const string& message, int address)
ostringstream buf; ostringstream buf;
buf << message; buf << message;
if(address > -1) if(address > -1)
buf << Common::Base::HEX4 << address; buf << cartDebug().getLabel(address, read);
myDialog->message().setText(buf.str()); myDialog->message().setText(buf.str());
return true; return true;
} }
@ -367,24 +368,43 @@ bool Debugger::breakPoint(uInt16 bp)
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Debugger::toggleReadTrap(uInt16 t) void Debugger::addReadTrap(uInt16 t)
{ {
readTraps().initialize(); readTraps().initialize();
readTraps().toggle(t); readTraps().add(t);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Debugger::addWriteTrap(uInt16 t)
void Debugger::toggleWriteTrap(uInt16 t)
{ {
writeTraps().initialize(); writeTraps().initialize();
writeTraps().toggle(t); writeTraps().add(t);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Debugger::toggleTrap(uInt16 t) void Debugger::addTrap(uInt16 t)
{ {
toggleReadTrap(t); addReadTrap(t);
toggleWriteTrap(t); addWriteTrap(t);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Debugger::removeReadTrap(uInt16 t)
{
readTraps().initialize();
readTraps().remove(t);
}
void Debugger::removeWriteTrap(uInt16 t)
{
writeTraps().initialize();
writeTraps().remove(t);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Debugger::removeTrap(uInt16 t)
{
removeReadTrap(t);
removeWriteTrap(t);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -399,6 +419,48 @@ bool Debugger::writeTrap(uInt16 t)
return writeTraps().isInitialized() && writeTraps().isSet(t); return writeTraps().isInitialized() && writeTraps().isSet(t);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt32 Debugger::getBaseAddress(uInt32 addr, bool read)
{
if((addr & 0x1080) == 0x0000) // (addr & 0b 0001 0000 1000 0000) == 0b 0000 0000 0000 0000
if(read)
// ADDR_TIA read (%xxx0 xxxx 0xxx ????)
return addr & 0x000f; // 0b 0000 0000 0000 1111
else
// ADDR_TIA write (%xxx0 xxxx 0x?? ????)
return addr & 0x003f; // 0b 0000 0000 0011 1111
// ADDR_ZPRAM (%xxx0 xx0x 1??? ????)
if((addr & 0x1280) == 0x0080) // (addr & 0b 0001 0010 1000 0000) == 0b 0000 0000 1000 0000
return addr & 0x00ff; // 0b 0000 0000 1111 1111
// ADDR_ROM
if(addr & 0x1000)
return addr & 0x1fff; // 0b 0001 1111 1111 1111
// ADDR_IO read/write I/O registers (%xxx0 xx1x 1xxx x0??)
if((addr & 0x1284) == 0x0280) // (addr & 0b 0001 0010 1000 0100) == 0b 0000 0010 1000 0000
return addr & 0x0283; // 0b 0000 0010 1000 0011
// ADDR_IO write timers (%xxx0 xx1x 1xx1 ?1??)
if(!read && (addr & 0x1294) == 0x0294) // (addr & 0b 0001 0010 1001 0100) == 0b 0000 0010 1001 0100
return addr & 0x029f; // 0b 0000 0010 1001 1111
// ADDR_IO read timers (%xxx0 xx1x 1xxx ?1x0)
if(read && (addr & 0x1285) == 0x0284) // (addr & 0b 0001 0010 1000 0101) == 0b 0000 0010 1000 0100
return addr & 0x028c; // 0b 0000 0010 1000 1100
// ADDR_IO read timer/PA7 interrupt (%xxx0 xx1x 1xxx x1x1)
if(read && (addr & 0x1285) == 0x0285) // (addr & 0b 0001 0010 1000 0101) == 0b 0000 0010 1000 0101
return addr & 0x0285; // 0b 0000 0010 1000 0101
// ADDR_IO write PA7 edge control (%xxx0 xx1x 1xx0 x1??)
if(!read && (addr & 0x1294) == 0x0284) // (addr & 0b 0001 0010 1001 0100) == 0b 0000 0010 1000 0100
return addr & 0x0287; // 0b 0000 0010 1000 0111
return 0;
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void Debugger::nextScanline(int lines) void Debugger::nextScanline(int lines)
{ {

View File

@ -88,7 +88,7 @@ class Debugger : public DialogContainer
@param message Message to display when entering debugger @param message Message to display when entering debugger
@param address An address associated with the message @param address An address associated with the message
*/ */
bool start(const string& message = "", int address = -1); bool start(const string& message = "", int address = -1, bool read = true);
bool startWithFatalError(const string& message = ""); bool startWithFatalError(const string& message = "");
/** /**
@ -145,8 +145,8 @@ class Debugger : public DialogContainer
TiaOutputWidget& tiaOutput() const { return myDialog->tiaOutput(); } TiaOutputWidget& tiaOutput() const { return myDialog->tiaOutput(); }
PackedBitArray& breakPoints() const { return mySystem.m6502().breakPoints(); } PackedBitArray& breakPoints() const { return mySystem.m6502().breakPoints(); }
PackedBitArray& readTraps() const { return mySystem.m6502().readTraps(); } TrapArray& readTraps() const { return mySystem.m6502().readTraps(); }
PackedBitArray& writeTraps() const { return mySystem.m6502().writeTraps(); } TrapArray& writeTraps() const { return mySystem.m6502().writeTraps(); }
/** /**
Run the debugger command and return the result. Run the debugger command and return the result.
@ -214,6 +214,7 @@ class Debugger : public DialogContainer
{ mySystem.setAccessFlags(addr, flags); } { mySystem.setAccessFlags(addr, flags); }
void setBreakPoint(uInt16 bp, bool set); void setBreakPoint(uInt16 bp, bool set);
uInt32 getBaseAddress(uInt32 addr, bool read);
bool patchROM(uInt16 addr, uInt8 value); bool patchROM(uInt16 addr, uInt8 value);
@ -256,9 +257,12 @@ class Debugger : public DialogContainer
void toggleBreakPoint(uInt16 bp); void toggleBreakPoint(uInt16 bp);
bool breakPoint(uInt16 bp); bool breakPoint(uInt16 bp);
void toggleReadTrap(uInt16 t); void addReadTrap(uInt16 t);
void toggleWriteTrap(uInt16 t); void addWriteTrap(uInt16 t);
void toggleTrap(uInt16 t); void addTrap(uInt16 t);
void removeReadTrap(uInt16 t);
void removeWriteTrap(uInt16 t);
void removeTrap(uInt16 t);
bool readTrap(uInt16 t); bool readTrap(uInt16 t);
bool writeTrap(uInt16 t); bool writeTrap(uInt16 t);
void clearAllTraps(); void clearAllTraps();

View File

@ -543,31 +543,32 @@ string DebuggerParser::eval()
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
string DebuggerParser::trapStatus(uInt32 addr, bool& enabled) string DebuggerParser::trapStatus(const Trap& trap)
{ {
string result; stringstream result;
result += Base::toString(addr); string lblb = debugger.cartDebug().getLabel(trap.begin, !trap.write);
result += ": "; string lble = debugger.cartDebug().getLabel(trap.end, !trap.write);
bool r = debugger.readTrap(addr);
bool w = debugger.writeTrap(addr);
enabled = r || w;
if(r && w)
result += "read|write";
else if(r)
result += "read";
else if(w)
result += "write";
else
result += "none";
const string& l = debugger.cartDebug().getLabel(addr, !w); if(lblb != "") {
if(l != "") { result << " (";
result += " ("; result << lblb;
result += l;
result += ")";
} }
return result; if(trap.begin != trap.end)
{
if(lble != "")
{
if (lblb != "")
result << " ";
else
result << " (";
result << lble;
}
}
if (lblb != "" || lble != "")
result << ")";
return result.str();
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -589,6 +590,7 @@ bool DebuggerParser::saveScriptFile(string file)
if(debugger.breakPoint(i)) if(debugger.breakPoint(i))
out << "break #" << i << endl; out << "break #" << i << endl;
// TODO: new trapif
for(uInt32 i = 0; i < 0x10000; ++i) for(uInt32 i = 0; i < 0x10000; ++i)
{ {
bool r = debugger.readTrap(i); bool r = debugger.readTrap(i);
@ -744,8 +746,9 @@ void DebuggerParser::executeClearconfig()
// "cleartraps" // "cleartraps"
void DebuggerParser::executeCleartraps() void DebuggerParser::executeCleartraps()
{ {
myTraps.clear();
debugger.clearAllTraps(); debugger.clearAllTraps();
debugger.cpuDebug().m6502().clearCondTraps();
myTraps.clear();
commandResult << "all traps cleared"; commandResult << "all traps cleared";
} }
@ -861,6 +864,24 @@ void DebuggerParser::executeDelfunction()
commandResult << "function " << argStrings[0] << " built-in or not found"; commandResult << "function " << argStrings[0] << " built-in or not found";
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// "deltrap"
void DebuggerParser::executeDeltrap()
{
int index = args[0];
if(debugger.cpuDebug().m6502().delCondTrap(index))
{
for(uInt32 addr = myTraps[index]->begin; addr <= myTraps[index]->end; ++addr)
executeTrapRW(addr, myTraps[index]->read, myTraps[index]->write, false);
// @sa666666: please check this:
Vec::removeAt(myTraps, index);
commandResult << "removed trap " << Base::toString(index);
}
else
commandResult << "no such trap";
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// "delwatch" // "delwatch"
void DebuggerParser::executeDelwatch() void DebuggerParser::executeDelwatch()
@ -1089,7 +1110,7 @@ void DebuggerParser::executeListbreaks()
commandResult << "\nbreakifs:\n"; commandResult << "\nbreakifs:\n";
for(uInt32 i = 0; i < conds.size(); i++) for(uInt32 i = 0; i < conds.size(); i++)
{ {
commandResult << i << ": " << conds[i]; commandResult << Base::toString(i) << ": " << conds[i];
if(i != (conds.size() - 1)) commandResult << endl; if(i != (conds.size() - 1)) commandResult << endl;
} }
} }
@ -1127,11 +1148,37 @@ void DebuggerParser::executeListfunctions()
// "listtraps" // "listtraps"
void DebuggerParser::executeListtraps() void DebuggerParser::executeListtraps()
{ {
if(myTraps.size() > 0) StringList names = debugger.cpuDebug().m6502().getCondTrapNames();
if(myTraps.size() != names.size())
{ {
bool enabled = true; commandResult << "Internal error! Different trap sizes.";
for(const auto& trap: myTraps) return;
commandResult << trapStatus(trap, enabled) << " + mirrors" << endl; }
if (names.size() > 0)
{
for(uInt32 i = 0; i < names.size(); i++)
{
commandResult << Base::toString(i) << ": ";
if(myTraps[i]->read && myTraps[i]->write)
commandResult << "read|write";
else if(myTraps[i]->read)
commandResult << "read ";
else if(myTraps[i]->write)
commandResult << " write";
else
commandResult << "none";
commandResult << " " << names[i];
commandResult << " " << Base::toString(myTraps[i]->begin);
if (myTraps[i]->begin != myTraps[i]->end)
commandResult << " " << Base::toString(myTraps[i]->end);
commandResult << trapStatus(*myTraps[i]);
commandResult << " + mirrors";
if(i != (names.size() - 1)) commandResult << endl;
}
} }
else else
commandResult << "no traps set"; commandResult << "no traps set";
@ -1482,71 +1529,153 @@ void DebuggerParser::executeTrace()
// "trap" // "trap"
void DebuggerParser::executeTrap() void DebuggerParser::executeTrap()
{ {
if(argCount > 2) executeTraps(true, true, "trap");
{ }
commandResult << red("Command takes one or two arguments") << endl;
return;
}
uInt32 beg = args[0]; // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt32 end = argCount == 2 ? args[1] : beg; // "trapif"
if(beg > 0xFFFF || end > 0xFFFF) void DebuggerParser::executeTrapif()
{ {
commandResult << red("One or more addresses are invalid") << endl; executeTraps(true, true, "trapif", true);
return;
}
for(uInt32 addr = beg; addr <= end; ++addr)
executeTrapRW(addr, true, true);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// "trapread" // "trapread"
void DebuggerParser::executeTrapread() void DebuggerParser::executeTrapread()
{ {
if(argCount > 2) executeTraps(true, false, "trapread");
{ }
commandResult << red("Command takes one or two arguments") << endl;
return;
}
uInt32 beg = args[0]; // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt32 end = argCount == 2 ? args[1] : beg; // "trapreadif"
if(beg > 0xFFFF || end > 0xFFFF) void DebuggerParser::executeTrapreadif()
{ {
commandResult << red("One or more addresses are invalid") << endl; executeTraps(true, false, "trapreadif", true);
return;
}
for(uInt32 addr = beg; addr <= end; ++addr)
executeTrapRW(addr, true, false);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// "trapwrite" // "trapwrite"
void DebuggerParser::executeTrapwrite() void DebuggerParser::executeTrapwrite()
{ {
if(argCount > 2) executeTraps(false, true, "trapwrite");
{
commandResult << red("Command takes one or two arguments") << endl;
return;
}
uInt32 beg = args[0];
uInt32 end = argCount == 2 ? args[1] : beg;
if(beg > 0xFFFF || end > 0xFFFF)
{
commandResult << red("One or more addresses are invalid") << endl;
return;
}
for(uInt32 addr = beg; addr <= end; ++addr)
executeTrapRW(addr, false, true);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// wrapper function for trap/trapread/trapwrite commands // "trapwriteif"
void DebuggerParser::executeTrapRW(uInt32 addr, bool read, bool write) void DebuggerParser::executeTrapwriteif()
{
executeTraps(false, true, "trapwriteif", true);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// Wrapper function for trap(if)s
void DebuggerParser::executeTraps(bool read, bool write, string command, bool hasCond)
{
if(hasCond)
{
if(argCount < 1 || argCount > 3)
{
commandResult << red("Command takes one to three arguments");
return;
}
}
else
{
if(argCount > 2)
{
commandResult << red("Command takes one or two arguments");
return;
}
}
int ofs = hasCond ? 1 : 0;
uInt32 begin = args[ofs];
uInt32 end = argCount == ofs + 2 ? args[ofs + 1] : begin;
// mirrors
uInt32 beginRead = debugger.getBaseAddress(begin, true);
uInt32 endRead = debugger.getBaseAddress(end, true);
uInt32 beginWrite = debugger.getBaseAddress(begin, false);
uInt32 endWrite = debugger.getBaseAddress(end, false);
if(begin > 0xFFFF || end > 0xFFFF || begin > end)
{
commandResult << red("One or more addresses are invalid");
return;
}
// parenthesize provided and address range condition(s) (begin)
stringstream parserBuf, displayBuf;
if(hasCond)
parserBuf << "(" << argStrings[0] << ")&&(";
// add address range condition(s) to provided condition
if(read)
{
if(beginRead != endRead)
parserBuf << "__lastread>=" << Base::toString(beginRead) << "&&__lastread<=" << Base::toString(endRead);
else
parserBuf << "__lastread==" << Base::toString(beginRead);
}
if(read && write)
parserBuf << "||";
if(write)
{
if(beginWrite != endWrite)
parserBuf << "__lastwrite>=" << Base::toString(beginWrite) << "&&__lastwrite<=" << Base::toString(endWrite);
else
parserBuf << "__lastwrite==" << Base::toString(beginWrite);
}
// parenthesize provided condition (end)
if(hasCond)
parserBuf << ")";
const string parserCondition = parserBuf.str();
int res = YaccParser::parse(parserCondition.c_str());
if(res == 0)
{
// duplicates will remove each other
bool add = true;
for(uInt32 i = 0; i < myTraps.size(); i++)
{
if(myTraps[i]->begin == begin && myTraps[i]->end == end &&
myTraps[i]->read == read && myTraps[i]->write == write &&
myTraps[i]->condition == parserCondition)
{
if(debugger.cpuDebug().m6502().delCondTrap(i))
{
add = false;
// @sa666666: please check this:
Vec::removeAt(myTraps, i);
commandResult << "Removed trap " << Base::toString(i);
break;
}
commandResult << "Internal error! Duplicate trap removal failed!";
return;
}
}
if(add)
{
uInt32 ret = debugger.cpuDebug().m6502().addCondTrap(
YaccParser::getResult(), hasCond ? argStrings[0] : " ");
commandResult << "Added trap " << Base::toString(ret);
// @sa666666: please check this:
myTraps.emplace_back(new Trap{ read, write, begin, end, parserCondition });
}
for(uInt32 addr = begin; addr <= end; ++addr)
executeTrapRW(addr, read, write, add);
}
else
{
commandResult << red("invalid expression");
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
// wrapper function for trap(if)/trapread(if)/trapwrite(if) commands
void DebuggerParser::executeTrapRW(uInt32 addr, bool read, bool write, bool add)
{ {
switch(debugger.cartDebug().addressType(addr)) switch(debugger.cartDebug().addressType(addr))
{ {
@ -1556,10 +1685,11 @@ void DebuggerParser::executeTrapRW(uInt32 addr, bool read, bool write)
{ {
if((i & 0x1080) == 0x0000) if((i & 0x1080) == 0x0000)
{ {
if(read && (i & 0x000F) == addr) // @sa666666: This seems wrong. E.g. trapread 40 4f will never trigger
debugger.toggleReadTrap(i); if(read && (i & 0x000F) == (addr & 0x000F))
if(write && (i & 0x003F) == addr) add ? debugger.addReadTrap(i) : debugger.removeReadTrap(i);
debugger.toggleWriteTrap(i); if(write && (i & 0x003F) == (addr & 0x003F))
add ? debugger.addWriteTrap(i) : debugger.removeWriteTrap(i);
} }
} }
break; break;
@ -1568,10 +1698,12 @@ void DebuggerParser::executeTrapRW(uInt32 addr, bool read, bool write)
{ {
for(uInt32 i = 0; i <= 0xFFFF; ++i) for(uInt32 i = 0; i <= 0xFFFF; ++i)
{ {
if((i & 0x1080) == 0x0080 && (i & 0x0200) != 0x0000 && (i & 0x02FF) == addr) if((i & 0x1280) == 0x0280 && (i & 0x029F) == (addr & 0x029F))
{ {
if(read) debugger.toggleReadTrap(i); if(read)
if(write) debugger.toggleWriteTrap(i); add ? debugger.addReadTrap(i) : debugger.removeReadTrap(i);
if(write)
add ? debugger.addWriteTrap(i) : debugger.removeWriteTrap(i);
} }
} }
break; break;
@ -1580,10 +1712,12 @@ void DebuggerParser::executeTrapRW(uInt32 addr, bool read, bool write)
{ {
for(uInt32 i = 0; i <= 0xFFFF; ++i) for(uInt32 i = 0; i <= 0xFFFF; ++i)
{ {
if((i & 0x1080) == 0x0080 && (i & 0x0200) == 0x0000 && (i & 0x00FF) == addr) if((i & 0x1280) == 0x0080 && (i & 0x00FF) == (addr & 0x00FF))
{ {
if(read) debugger.toggleReadTrap(i); if(read)
if(write) debugger.toggleWriteTrap(i); add ? debugger.addReadTrap(i) : debugger.removeReadTrap(i);
if(write)
add ? debugger.addWriteTrap(i) : debugger.removeWriteTrap(i);
} }
} }
break; break;
@ -1596,21 +1730,16 @@ void DebuggerParser::executeTrapRW(uInt32 addr, bool read, bool write)
{ {
if((i % 0x2000 >= 0x1000) && (i & 0x0FFF) == (addr & 0x0FFF)) if((i % 0x2000 >= 0x1000) && (i & 0x0FFF) == (addr & 0x0FFF))
{ {
if(read) debugger.toggleReadTrap(i); if(read)
if(write) debugger.toggleWriteTrap(i); add ? debugger.addReadTrap(i) : debugger.removeReadTrap(i);
if(write)
add ? debugger.addWriteTrap(i) : debugger.removeWriteTrap(i);
} }
} }
} }
break; break;
} }
} }
bool trapEnabled = false;
const string& result = trapStatus(addr, trapEnabled);
if(trapEnabled) myTraps.insert(addr);
else myTraps.erase(addr);
commandResult << result << " + mirrors" << endl;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -1891,6 +2020,16 @@ DebuggerParser::Command DebuggerParser::commands[kNumCommands] = {
std::mem_fn(&DebuggerParser::executeDelfunction) std::mem_fn(&DebuggerParser::executeDelfunction)
}, },
{
"deltrap",
"Delete trap <xx>",
"Example: deltrap 0",
true,
false,
{ kARG_WORD, kARG_END_ARGS },
std::mem_fn(&DebuggerParser::executeDeltrap)
},
{ {
"delwatch", "delwatch",
"Delete watch <xx>", "Delete watch <xx>",
@ -2333,6 +2472,17 @@ DebuggerParser::Command DebuggerParser::commands[kNumCommands] = {
std::mem_fn(&DebuggerParser::executeTrap) std::mem_fn(&DebuggerParser::executeTrap)
}, },
{
"trapif",
"On <condition> trap R/W access to address(es) xx [yy]",
"Set a conditional R/W trap on the given address(es) and all mirrors\nCondition can include multiple items.\n"
"Example: trapif _scan>100 GRP0, trapif _bank==1 f000 f100",
true,
false,
{ kARG_WORD, kARG_MULTI_BYTE },
std::mem_fn(&DebuggerParser::executeTrapif)
},
{ {
"trapread", "trapread",
"Trap read access to address(es) xx [yy]", "Trap read access to address(es) xx [yy]",
@ -2344,6 +2494,17 @@ DebuggerParser::Command DebuggerParser::commands[kNumCommands] = {
std::mem_fn(&DebuggerParser::executeTrapread) std::mem_fn(&DebuggerParser::executeTrapread)
}, },
{
"trapreadif",
"On <condition> trap read access to address(es) xx [yy]",
"Set a conditional read trap on the given address(es) and all mirrors\nCondition can include multiple items.\n"
"Example: trapreadif _scan>100 GRP0, trapreadif _bank==1 f000 f100",
true,
false,
{ kARG_WORD, kARG_MULTI_BYTE },
std::mem_fn(&DebuggerParser::executeTrapreadif)
},
{ {
"trapwrite", "trapwrite",
"Trap write access to address(es) xx [yy]", "Trap write access to address(es) xx [yy]",
@ -2355,6 +2516,17 @@ DebuggerParser::Command DebuggerParser::commands[kNumCommands] = {
std::mem_fn(&DebuggerParser::executeTrapwrite) std::mem_fn(&DebuggerParser::executeTrapwrite)
}, },
{
"trapwriteif",
"On <condition> trap write access to address(es) xx [yy]",
"Set a conditional write trap on the given address(es) and all mirrors\nCondition can include multiple items.\n"
"Example: trapwriteif _scan>100 GRP0, trapwriteif _bank==1 f000 f100",
true,
false,
{ kARG_WORD, kARG_MULTI_BYTE },
std::mem_fn(&DebuggerParser::executeTrapwriteif)
},
{ {
"type", "type",
"Show disassembly type for address xx [yy]", "Show disassembly type for address xx [yy]",

View File

@ -68,7 +68,7 @@ class DebuggerParser
bool saveScriptFile(string file); bool saveScriptFile(string file);
private: private:
enum { kNumCommands = 72 }; enum { kNumCommands = 76 };
// Constants for argument processing // Constants for argument processing
enum { enum {
@ -100,6 +100,14 @@ class DebuggerParser
parameters parms[10]; parameters parms[10];
std::function<void (DebuggerParser*)> executor; std::function<void (DebuggerParser*)> executor;
}; };
struct Trap
{
bool read;
bool write;
uInt32 begin;
uInt32 end;
string condition;
};
// Reference to our debugger object // Reference to our debugger object
Debugger& debugger; Debugger& debugger;
@ -117,9 +125,12 @@ class DebuggerParser
StringList myWatches; StringList myWatches;
// Keep track of traps (read and/or write) // Keep track of traps (read and/or write)
std::set<uInt32> myTraps; vector<unique_ptr<Trap>> myTraps;
string trapStatus(uInt32 addr, bool& enabled); /*std::set<uInt32> myTraps;
std::vector<uInt32> myTrapIfs;*/
string trapStatus(const Trap& trap);
// List of available command methods // List of available command methods
void executeA(); void executeA();
@ -141,6 +152,7 @@ class DebuggerParser
void executeDefine(); void executeDefine();
void executeDelbreakif(); void executeDelbreakif();
void executeDelfunction(); void executeDelfunction();
void executeDeltrap();
void executeDelwatch(); void executeDelwatch();
void executeDisasm(); void executeDisasm();
void executeDump(); void executeDump();
@ -184,9 +196,13 @@ class DebuggerParser
void executeTia(); void executeTia();
void executeTrace(); void executeTrace();
void executeTrap(); void executeTrap();
void executeTrapif();
void executeTrapread(); void executeTrapread();
void executeTrapreadif();
void executeTrapwrite(); void executeTrapwrite();
void executeTrapRW(uInt32 addr, bool read, bool write); // not exposed by debugger void executeTrapwriteif();
void executeTraps(bool read, bool write, string command, bool cond = false);
void executeTrapRW(uInt32 addr, bool read, bool write, bool add = true); // not exposed by debugger
void executeType(); void executeType();
void executeUHex(); void executeUHex();
void executeUndef(); void executeUndef();

View File

@ -0,0 +1,59 @@
//============================================================================
//
// SSSS tt lll lll
// SS SS tt ll ll
// SS tttttt eeee ll ll aaaa
// SSSS tt ee ee ll ll aa
// SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator"
// SS SS tt ee ll ll aa aa
// SSSS ttt eeeee llll llll aaaaa
//
// Copyright (c) 1995-2017 by Bradford W. Mott, Stephen Anthony
// and the Stella Team
//
// See the file "License.txt" for information on usage and redistribution of
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
//============================================================================
#ifndef TRAP_ARRAY_HXX
#define TRAP_ARRAY_HXX
#include "bspf.hxx"
class TrapArray
{
public:
TrapArray() : myInitialized(false) {}
bool isSet(const uInt16 address) const { return myCount[address]; }
bool isClear(const uInt16 address) const { return myCount[address] == 0; }
void add(const uInt16 address) { myCount[address]++; }
void remove(const uInt16 address) { myCount[address]--; }
//void toggle(uInt16 address) { myCount[address] ? remove(address) : add(address); } // TODO condition
void initialize() {
if(!myInitialized)
memset(myCount, 0, sizeof(myCount));
myInitialized = true;
}
void clearAll() { myInitialized = false; memset(myCount, 0, sizeof(myCount)); }
bool isInitialized() const { return myInitialized; }
private:
// The actual counts
uInt8 myCount[0x10000];
// Indicates whether we should treat this array as initialized
bool myInitialized;
private:
// Following constructors and assignment operators not supported
TrapArray(const TrapArray&) = delete;
TrapArray(TrapArray&&) = delete;
TrapArray& operator=(const TrapArray&) = delete;
TrapArray& operator=(TrapArray&&) = delete;
};
#endif

View File

@ -55,6 +55,8 @@ M6502::M6502(const Settings& settings)
myLastAddress(0), myLastAddress(0),
myLastPeekAddress(0), myLastPeekAddress(0),
myLastPokeAddress(0), myLastPokeAddress(0),
myLastPeekBaseAddress(0),
myLastPokeBaseAddress(0),
myLastSrcAddressS(-1), myLastSrcAddressS(-1),
myLastSrcAddressA(-1), myLastSrcAddressA(-1),
myLastSrcAddressX(-1), myLastSrcAddressX(-1),
@ -65,7 +67,7 @@ M6502::M6502(const Settings& settings)
{ {
#ifdef DEBUGGER_SUPPORT #ifdef DEBUGGER_SUPPORT
myDebugger = nullptr; myDebugger = nullptr;
myJustHitTrapFlag = false; myJustHitReadTrapFlag = myJustHitWriteTrapFlag = false;
#endif #endif
} }
@ -98,7 +100,7 @@ void M6502::reset()
// Load PC from the reset vector // Load PC from the reset vector
PC = uInt16(mySystem->peek(0xfffc)) | (uInt16(mySystem->peek(0xfffd)) << 8); PC = uInt16(mySystem->peek(0xfffc)) | (uInt16(mySystem->peek(0xfffd)) << 8);
myLastAddress = myLastPeekAddress = myLastPokeAddress = 0; myLastAddress = myLastPeekAddress = myLastPokeAddress = myLastPeekBaseAddress = myLastPokeBaseAddress;
myLastSrcAddressS = myLastSrcAddressA = myLastSrcAddressS = myLastSrcAddressA =
myLastSrcAddressX = myLastSrcAddressY = -1; myLastSrcAddressX = myLastSrcAddressY = -1;
myDataAddressForPoke = 0; myDataAddressForPoke = 0;
@ -120,18 +122,23 @@ inline uInt8 M6502::peek(uInt16 address, uInt8 flags)
} }
//////////////////////////////////////////////// ////////////////////////////////////////////////
mySystem->incrementCycles(SYSTEM_CYCLES_PER_CPU); mySystem->incrementCycles(SYSTEM_CYCLES_PER_CPU);
uInt8 result = mySystem->peek(address, flags);
myLastPeekAddress = address;
#ifdef DEBUGGER_SUPPORT #ifdef DEBUGGER_SUPPORT
if(myReadTraps.isInitialized() && myReadTraps.isSet(address)) if(myReadTraps.isInitialized() && myReadTraps.isSet(address))
{ {
myJustHitTrapFlag = true; myLastPeekBaseAddress = myDebugger->getBaseAddress(myLastPeekAddress, true); // mirror handling
myHitTrapInfo.message = "RTrap: "; int cond = evalCondTraps();
myHitTrapInfo.address = address; if(cond > -1)
{
myJustHitReadTrapFlag = true;
myHitTrapInfo.message = "RTrap(" + myTrapCondNames[cond] + "): ";
myHitTrapInfo.address = address;
}
} }
#endif // DEBUGGER_SUPPORT #endif // DEBUGGER_SUPPORT
uInt8 result = mySystem->peek(address, flags);
myLastPeekAddress = address;
return result; return result;
} }
@ -147,18 +154,22 @@ inline void M6502::poke(uInt16 address, uInt8 value, uInt8 flags)
} }
//////////////////////////////////////////////// ////////////////////////////////////////////////
mySystem->incrementCycles(SYSTEM_CYCLES_PER_CPU); mySystem->incrementCycles(SYSTEM_CYCLES_PER_CPU);
mySystem->poke(address, value, flags);
myLastPokeAddress = address;
#ifdef DEBUGGER_SUPPORT #ifdef DEBUGGER_SUPPORT
if(myWriteTraps.isInitialized() && myWriteTraps.isSet(address)) if(myWriteTraps.isInitialized() && myWriteTraps.isSet(address))
{ {
myJustHitTrapFlag = true; myLastPokeBaseAddress = myDebugger->getBaseAddress(myLastPokeAddress, false); // mirror handling
myHitTrapInfo.message = "WTrap: "; int cond = evalCondTraps();
myHitTrapInfo.address = address; if(cond > -1)
{
myJustHitWriteTrapFlag = true;
myHitTrapInfo.message = "WTrap(" + myTrapCondNames[cond] + "): ";
myHitTrapInfo.address = address;
}
} }
#endif // DEBUGGER_SUPPORT #endif // DEBUGGER_SUPPORT
mySystem->poke(address, value, flags);
myLastPokeAddress = address;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -189,11 +200,11 @@ bool M6502::execute(uInt32 number)
for(; !myExecutionStatus && (number != 0); --number) for(; !myExecutionStatus && (number != 0); --number)
{ {
#ifdef DEBUGGER_SUPPORT #ifdef DEBUGGER_SUPPORT
if(myJustHitTrapFlag) if(myJustHitReadTrapFlag || myJustHitWriteTrapFlag)
{ {
if(myDebugger && myDebugger->start(myHitTrapInfo.message, myHitTrapInfo.address)) myJustHitReadTrapFlag = myJustHitWriteTrapFlag = false;
if(myDebugger && myDebugger->start(myHitTrapInfo.message, myHitTrapInfo.address, myJustHitReadTrapFlag))
{ {
myJustHitTrapFlag = false;
return true; return true;
} }
} }
@ -416,13 +427,15 @@ uInt32 M6502::addCondBreak(Expression* e, const string& name)
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void M6502::delCondBreak(uInt32 brk) bool M6502::delCondBreak(uInt32 brk)
{ {
if(brk < myBreakConds.size()) if(brk < myBreakConds.size())
{ {
Vec::removeAt(myBreakConds, brk); Vec::removeAt(myBreakConds, brk);
Vec::removeAt(myBreakCondNames, brk); Vec::removeAt(myBreakCondNames, brk);
return true;
} }
return false;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -437,4 +450,37 @@ const StringList& M6502::getCondBreakNames() const
{ {
return myBreakCondNames; return myBreakCondNames;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt32 M6502::addCondTrap(Expression* e, const string& name)
{
myTrapConds.emplace_back(e);
myTrapCondNames.push_back(name);
return uInt32(myTrapConds.size() - 1);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
bool M6502::delCondTrap(uInt32 brk)
{
if(brk < myTrapConds.size())
{
Vec::removeAt(myTrapConds, brk);
Vec::removeAt(myTrapCondNames, brk);
return true;
}
return false;
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void M6502::clearCondTraps()
{
myTrapConds.clear();
myTrapCondNames.clear();
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
const StringList& M6502::getCondTrapNames() const
{
return myTrapCondNames;
}
#endif // DEBUGGER_SUPPORT #endif // DEBUGGER_SUPPORT

View File

@ -24,6 +24,7 @@
#include "Expression.hxx" #include "Expression.hxx"
#include "PackedBitArray.hxx" #include "PackedBitArray.hxx"
#include "TrapArray.hxx"
#endif #endif
class Settings; class Settings;
@ -150,6 +151,20 @@ class M6502 : public Serializable
myLastPeekAddress; myLastPeekAddress;
} }
/**
Return the last address that was part of a read/peek.
@return The address of the last read
*/
uInt16 lastReadBaseAddress() const { return myLastPeekBaseAddress; }
/**
Return the last address that was part of a write/poke.
@return The address of the last write
*/
uInt16 lastWriteBaseAddress() const { return myLastPokeBaseAddress; }
/** /**
Return the source of the address that was used for a write/poke. Return the source of the address that was used for a write/poke.
Note that this isn't the same as the address that is poked, but Note that this isn't the same as the address that is poked, but
@ -207,13 +222,24 @@ class M6502 : public Serializable
void attach(Debugger& debugger); void attach(Debugger& debugger);
PackedBitArray& breakPoints() { return myBreakPoints; } PackedBitArray& breakPoints() { return myBreakPoints; }
PackedBitArray& readTraps() { return myReadTraps; } //PackedBitArray& readTraps() { return myReadTraps; }
PackedBitArray& writeTraps() { return myWriteTraps; } //PackedBitArray& writeTraps() { return myWriteTraps; }
//PackedBitArray& readTrapIfs() { return myReadTrapIfs; }
//PackedBitArray& writeTrapIfs() { return myWriteTrapIfs; }
TrapArray& readTraps() { return myReadTraps; }
TrapArray& writeTraps() { return myWriteTraps; }
// methods for 'breakif' handling
uInt32 addCondBreak(Expression* e, const string& name); uInt32 addCondBreak(Expression* e, const string& name);
void delCondBreak(uInt32 brk); bool delCondBreak(uInt32 brk);
void clearCondBreaks(); void clearCondBreaks();
const StringList& getCondBreakNames() const; const StringList& getCondBreakNames() const;
// methods for 'trapif' handling
uInt32 addCondTrap(Expression* e, const string& name);
bool delCondTrap(uInt32 brk);
void clearCondTraps();
const StringList& getCondTrapNames() const;
#endif // DEBUGGER_SUPPORT #endif // DEBUGGER_SUPPORT
private: private:
@ -329,6 +355,9 @@ class M6502 : public Serializable
/// Indicates the last address which was accessed specifically /// Indicates the last address which was accessed specifically
/// by a peek or poke command /// by a peek or poke command
uInt16 myLastPeekAddress, myLastPokeAddress; uInt16 myLastPeekAddress, myLastPokeAddress;
/// Indicates the last base (= non-mirrored) address which was
/// accessed specifically by a peek or poke command
uInt16 myLastPeekBaseAddress, myLastPokeBaseAddress;
/// Indicates the last address used to access data by a peek command /// Indicates the last address used to access data by a peek command
/// for the CPU registers (S/A/X/Y) /// for the CPU registers (S/A/X/Y)
@ -359,14 +388,25 @@ class M6502 : public Serializable
return -1; // no break hit return -1; // no break hit
} }
Int32 evalCondTraps()
{
for(uInt32 i = 0; i < myTrapConds.size(); i++)
if(myTrapConds[i]->evaluate())
return i;
return -1; // no trapif hit
}
/// Pointer to the debugger for this processor or the null pointer /// Pointer to the debugger for this processor or the null pointer
Debugger* myDebugger; Debugger* myDebugger;
// Addresses for which the specified action should occur // Addresses for which the specified action should occur
PackedBitArray myBreakPoints, myReadTraps, myWriteTraps; PackedBitArray myBreakPoints;// , myReadTraps, myWriteTraps, myReadTrapIfs, myWriteTrapIfs;
TrapArray myReadTraps, myWriteTraps;
// Did we just now hit a trap? // Did we just now hit a trap?
bool myJustHitTrapFlag; bool myJustHitReadTrapFlag;
bool myJustHitWriteTrapFlag;
struct HitTrapInfo { struct HitTrapInfo {
string message; string message;
int address; int address;
@ -375,6 +415,8 @@ class M6502 : public Serializable
vector<unique_ptr<Expression>> myBreakConds; vector<unique_ptr<Expression>> myBreakConds;
StringList myBreakCondNames; StringList myBreakCondNames;
vector<unique_ptr<Expression>> myTrapConds;
StringList myTrapCondNames;
#endif // DEBUGGER_SUPPORT #endif // DEBUGGER_SUPPORT
private: private:

View File

@ -575,6 +575,7 @@
<ClInclude Include="..\debugger\gui\RomListSettings.hxx" /> <ClInclude Include="..\debugger\gui\RomListSettings.hxx" />
<ClInclude Include="..\debugger\gui\SaveKeyWidget.hxx" /> <ClInclude Include="..\debugger\gui\SaveKeyWidget.hxx" />
<ClInclude Include="..\debugger\gui\TrakBallWidget.hxx" /> <ClInclude Include="..\debugger\gui\TrakBallWidget.hxx" />
<ClInclude Include="..\debugger\TrapArray.hxx" />
<ClInclude Include="..\emucore\AmigaMouse.hxx" /> <ClInclude Include="..\emucore\AmigaMouse.hxx" />
<ClInclude Include="..\emucore\AtariMouse.hxx" /> <ClInclude Include="..\emucore\AtariMouse.hxx" />
<ClInclude Include="..\emucore\BSType.hxx" /> <ClInclude Include="..\emucore\BSType.hxx" />

View File

@ -1751,6 +1751,9 @@
<ClInclude Include="..\debugger\gui\TrakBallWidget.hxx"> <ClInclude Include="..\debugger\gui\TrakBallWidget.hxx">
<Filter>Header Files\debugger</Filter> <Filter>Header Files\debugger</Filter>
</ClInclude> </ClInclude>
<ClInclude Include="..\debugger\TrapArray.hxx">
<Filter>Header Files\debugger</Filter>
</ClInclude>
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<None Include="stella.ico"> <None Include="stella.ico">

View File

@ -206,6 +206,10 @@ CartMethod getCartSpecial(char* ch)
return &CartDebug::getBank; return &CartDebug::getBank;
else if(BSPF::equalsIgnoreCase(ch, "_rwport")) else if(BSPF::equalsIgnoreCase(ch, "_rwport"))
return &CartDebug::readFromWritePort; return &CartDebug::readFromWritePort;
else if(BSPF::equalsIgnoreCase(ch, "__lastread"))
return &CartDebug::lastReadBaseAddress;
else if(BSPF::equalsIgnoreCase(ch, "__lastwrite"))
return &CartDebug::lastWriteBaseAddress;
else else
return 0; return 0;
} }