mirror of https://github.com/stella-emu/stella.git
When a ROM doesn't exist and is temporarily inserted into the properties
database, it now has its extension removed (.bin, .a26, etc). Symbol and config files are now loaded when the debugger is first entered, not when the ROM is actually loaded. This speeds up ROM loading, since the user may never need to access these file (ie, they may never enter the debugger). The debugger 'loadconfig' and 'saveconfig' commands are now implemented. Bumped version # for next test release. git-svn-id: svn://svn.code.sf.net/p/stella/code/trunk@2126 8b62c5a3-ac7e-4cc8-8f21-d9a121418aba
This commit is contained in:
parent
2d08d4e459
commit
7f75957e11
|
@ -22,7 +22,7 @@
|
|||
|
||||
#include <cstdlib>
|
||||
|
||||
#define STELLA_VERSION "3.3_test2"
|
||||
#define STELLA_VERSION "3.3_test3"
|
||||
#define STELLA_BUILD atoi("$Rev$" + 6)
|
||||
|
||||
#endif
|
||||
|
|
|
@ -46,10 +46,10 @@ CartDebug::CartDebug(Debugger& dbg, Console& console, const OSystem& osystem)
|
|||
BankInfo info;
|
||||
for(int i = 0; i < myConsole.cartridge().bankCount(); ++i)
|
||||
{
|
||||
info.banksize = banksize; // TODO - get this from Cart class
|
||||
info.size = banksize; // TODO - get this from Cart class
|
||||
myBankInfo.push_back(info);
|
||||
}
|
||||
info.banksize = 128; // ZP RAM
|
||||
info.size = 128; // ZP RAM
|
||||
myBankInfo.push_back(info);
|
||||
|
||||
// We know the address for the startup bank right now
|
||||
|
@ -362,10 +362,13 @@ bool CartDebug::addDirective(CartDebug::DisasmType type,
|
|||
PRINT_TAG((*d)); \
|
||||
cerr << endl;
|
||||
|
||||
if(end < start || start == 0 || end == 0)
|
||||
return false;
|
||||
|
||||
if(bank < 0) // Do we want the current bank or ZP RAM?
|
||||
bank = (myDebugger.cpuDebug().pc() & 0x1000) ? getBank() : myBankInfo.size()-1;
|
||||
|
||||
bank = BSPF_min(bank, bankCount() - 1);
|
||||
bank = BSPF_min(bank, bankCount());
|
||||
BankInfo& info = myBankInfo[bank];
|
||||
DirectiveList& list = info.directiveList;
|
||||
|
||||
|
@ -595,15 +598,20 @@ int CartDebug::getAddress(const string& label) const
|
|||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
string CartDebug::loadSymbolFile(const string& f)
|
||||
string CartDebug::loadSymbolFile(string file)
|
||||
{
|
||||
string file = f;
|
||||
// TODO - use similar load logic as loadconfig command
|
||||
if(file == "")
|
||||
file = myOSystem.romFile();
|
||||
|
||||
string::size_type spos;
|
||||
if( (spos = file.find_last_of('.')) != string::npos )
|
||||
file.replace(spos, file.size(), ".sym");
|
||||
else
|
||||
file += ".sym";
|
||||
|
||||
// TODO - rewrite this to use C++ streams
|
||||
|
||||
int pos = 0, lines = 0, curVal;
|
||||
string curLabel;
|
||||
char line[1024];
|
||||
|
@ -651,7 +659,25 @@ string CartDebug::loadConfigFile(string file)
|
|||
{
|
||||
FilesystemNode node(file);
|
||||
|
||||
if(file == "") // Use config file based on ROM name
|
||||
if(file == "")
|
||||
{
|
||||
// There are three possible locations for loading config files
|
||||
// (in order of decreasing relevance):
|
||||
// 1) ROM dir based on properties entry name
|
||||
// 2) ROM dir based on actual ROM name
|
||||
// 3) CFG dir based on properties entry name
|
||||
|
||||
const string& propsname =
|
||||
myConsole.properties().get(Cartridge_Name) + ".cfg";
|
||||
|
||||
// Case 1
|
||||
FilesystemNode case1(FilesystemNode(myOSystem.romFile()).getParent().getPath() +
|
||||
propsname);
|
||||
if(case1.exists())
|
||||
{
|
||||
node = case1;
|
||||
}
|
||||
else
|
||||
{
|
||||
file = myOSystem.romFile();
|
||||
string::size_type spos;
|
||||
|
@ -659,18 +685,17 @@ string CartDebug::loadConfigFile(string file)
|
|||
file.replace(spos, file.size(), ".cfg");
|
||||
else
|
||||
file += ".cfg";
|
||||
FilesystemNode usercfg(file);
|
||||
if(usercfg.exists())
|
||||
FilesystemNode case2(file);
|
||||
if(case2.exists())
|
||||
{
|
||||
node = usercfg;
|
||||
node = case2;
|
||||
}
|
||||
else // Use global config file based on properties cart name
|
||||
{
|
||||
const string& globalfile = myOSystem.cfgDir() +
|
||||
myConsole.properties().get(Cartridge_Name) + ".cfg";
|
||||
FilesystemNode globalcfg(globalfile);
|
||||
if(globalcfg.exists())
|
||||
node = globalcfg;
|
||||
FilesystemNode case3(myOSystem.cfgDir() + propsname);
|
||||
if(case3.exists())
|
||||
node = case3;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -678,7 +703,14 @@ string CartDebug::loadConfigFile(string file)
|
|||
{
|
||||
ifstream in(node.getPath().c_str());
|
||||
if(!in.is_open())
|
||||
return "Unable to read directives from " + node.getPath();
|
||||
return "Unable to load directives from " + node.getPath();
|
||||
|
||||
// Erase all previous directives
|
||||
for(Common::Array<BankInfo>::iterator bi = myBankInfo.begin();
|
||||
bi != myBankInfo.end(); ++bi)
|
||||
{
|
||||
bi->directiveList.clear();
|
||||
}
|
||||
|
||||
int currentbank = 0;
|
||||
while(!in.eof())
|
||||
|
@ -718,33 +750,26 @@ string CartDebug::loadConfigFile(string file)
|
|||
{
|
||||
buf >> hex >> start;
|
||||
// TODO - figure out what to do with this
|
||||
cerr << "ignoring directive: "
|
||||
<< directive << " " << HEX4 << start << endl;
|
||||
}
|
||||
else if(BSPF_startsWithIgnoreCase(directive, "BLOCK"))
|
||||
{
|
||||
buf >> hex >> start;
|
||||
buf >> hex >> end;
|
||||
// TODO - add directive for this
|
||||
cerr << "ignoring directive: " << directive << " "
|
||||
<< HEX4 << start << " " << HEX4 << end << endl;
|
||||
// addDirective(CartDebug::BLOCK, start, end, currentbank);
|
||||
}
|
||||
else if(BSPF_startsWithIgnoreCase(directive, "CODE"))
|
||||
{
|
||||
buf >> hex >> start;
|
||||
buf >> hex >> end;
|
||||
buf >> hex >> start >> hex >> end;
|
||||
addDirective(CartDebug::CODE, start, end, currentbank);
|
||||
}
|
||||
else if(BSPF_startsWithIgnoreCase(directive, "DATA"))
|
||||
{
|
||||
buf >> hex >> start;
|
||||
buf >> hex >> end;
|
||||
buf >> hex >> start >> hex >> end;
|
||||
addDirective(CartDebug::DATA, start, end, currentbank);
|
||||
}
|
||||
else if(BSPF_startsWithIgnoreCase(directive, "GFX"))
|
||||
{
|
||||
buf >> hex >> start;
|
||||
buf >> hex >> end;
|
||||
buf >> hex >> start >> hex >> end;
|
||||
addDirective(CartDebug::GFX, start, end, currentbank);
|
||||
}
|
||||
}
|
||||
|
@ -760,7 +785,44 @@ string CartDebug::loadConfigFile(string file)
|
|||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
string CartDebug::saveConfigFile(string file)
|
||||
{
|
||||
return "NOT YET IMPLEMENTED";
|
||||
FilesystemNode node(file);
|
||||
|
||||
const string& name = myConsole.properties().get(Cartridge_Name);
|
||||
const string& md5 = myConsole.properties().get(Cartridge_MD5);
|
||||
|
||||
if(file == "")
|
||||
{
|
||||
// There are two possible locations for saving config files
|
||||
// (in order of decreasing relevance):
|
||||
// 1) ROM dir based on properties entry name
|
||||
// 2) ROM dir based on actual ROM name
|
||||
//
|
||||
// In either case, we're using the properties entry, since even ROMs that
|
||||
// don't have a proper entry have a temporary one inserted by OSystem
|
||||
node = FilesystemNode(name + ".cfg");
|
||||
}
|
||||
|
||||
if(!node.isDirectory())
|
||||
{
|
||||
ofstream out(node.getPath().c_str());
|
||||
if(!out.is_open())
|
||||
return "Unable to save directives to " + node.getPath();
|
||||
|
||||
// Store all bank information
|
||||
out << "//Stella.pro: \"" << name << "\"" << endl
|
||||
<< "//MD5: " << md5 << endl
|
||||
<< endl;
|
||||
for(uInt32 b = 0; b < myConsole.cartridge().bankCount(); ++b)
|
||||
{
|
||||
out << "[" << b << "]" << endl;
|
||||
getBankDirectives(out, myBankInfo[b]);
|
||||
}
|
||||
out.close();
|
||||
|
||||
return "saved " + node.getRelativePath() + " OK";
|
||||
}
|
||||
else
|
||||
return DebuggerParser::red("config file not found");
|
||||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
|
@ -782,11 +844,12 @@ string CartDebug::listConfig(int bank)
|
|||
for(DirectiveList::const_iterator i = info.directiveList.begin();
|
||||
i != info.directiveList.end(); ++i)
|
||||
{
|
||||
buf << "(*) "
|
||||
<< (i->type == CartDebug::CODE ? "CODE" :
|
||||
i->type == CartDebug::DATA ? "DATA" :
|
||||
"GFX")
|
||||
<< " " << HEX4 << i->start << " " << HEX4 << i->end << endl;
|
||||
if(i->type != CartDebug::NONE)
|
||||
{
|
||||
buf << "(*) ";
|
||||
disasmTypeAsString(buf, i->type);
|
||||
buf << " " << HEX4 << i->start << " " << HEX4 << i->end << endl;
|
||||
}
|
||||
}
|
||||
getBankDirectives(buf, info);
|
||||
}
|
||||
|
@ -861,7 +924,7 @@ CartDebug::AddrType CartDebug::addressType(uInt16 addr) const
|
|||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
void CartDebug::getBankDirectives(ostringstream& buf, BankInfo& info) const
|
||||
void CartDebug::getBankDirectives(ostream& buf, BankInfo& info) const
|
||||
{
|
||||
// Disassemble the bank, then scan it for an up-to-date description
|
||||
DisassemblyList list;
|
||||
|
@ -890,10 +953,9 @@ void CartDebug::getBankDirectives(ostringstream& buf, BankInfo& info) const
|
|||
if(type == CartDebug::DATA)
|
||||
last = tag.address - 1;
|
||||
|
||||
buf << (type == CartDebug::CODE ? "CODE" :
|
||||
type == CartDebug::DATA ? "DATA" :
|
||||
"GFX")
|
||||
<< " " << HEX4 << start << " " << HEX4 << last << endl;
|
||||
disasmTypeAsString(buf, type);
|
||||
buf << " " << HEX4 << start << " " << HEX4 << last << endl;
|
||||
|
||||
type = tag.type;
|
||||
start = last = tag.address;
|
||||
}
|
||||
|
@ -901,10 +963,30 @@ void CartDebug::getBankDirectives(ostringstream& buf, BankInfo& info) const
|
|||
last = tag.address;
|
||||
}
|
||||
// Grab the last directive, making sure it accounts for all remaining space
|
||||
buf << (type == CartDebug::CODE ? "CODE" :
|
||||
type == CartDebug::DATA ? "DATA" :
|
||||
"GFX")
|
||||
<< " " << HEX4 << start << " " << HEX4 << (info.offset+info.end) << endl;
|
||||
disasmTypeAsString(buf, type);
|
||||
buf << " " << HEX4 << start << " " << HEX4 << (info.offset+info.end) << endl;
|
||||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
void CartDebug::disasmTypeAsString(ostream& buf, DisasmType type) const
|
||||
{
|
||||
switch(type)
|
||||
{
|
||||
case CartDebug::BLOCK:
|
||||
buf << "BLOCK";
|
||||
break;
|
||||
case CartDebug::CODE:
|
||||
buf << "CODE";
|
||||
break;
|
||||
case CartDebug::DATA:
|
||||
buf << "DATA";
|
||||
break;
|
||||
case CartDebug::GFX:
|
||||
buf << "GFX";
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
|
|
|
@ -57,7 +57,8 @@ class CartDebug : public DebuggerSystem
|
|||
GFX = 1 << 0,
|
||||
DATA = 1 << 1,
|
||||
CODE = 1 << 2,
|
||||
NONE = 1 << 3
|
||||
BLOCK = 1 << 3,
|
||||
NONE = 1 << 4
|
||||
};
|
||||
struct DisassemblyTag {
|
||||
DisasmType type;
|
||||
|
@ -212,7 +213,7 @@ class CartDebug : public DebuggerSystem
|
|||
/**
|
||||
Load user equates from the given symbol file (as generated by DASM).
|
||||
*/
|
||||
string loadSymbolFile(const string& file = "");
|
||||
string loadSymbolFile(string file = "");
|
||||
|
||||
/**
|
||||
Load/save Distella config file (Distella directives)
|
||||
|
@ -257,7 +258,7 @@ class CartDebug : public DebuggerSystem
|
|||
uInt16 start; // start of address space
|
||||
uInt16 end; // end of address space
|
||||
uInt16 offset; // ORG value
|
||||
uInt16 banksize; // size of a bank (in bytes)
|
||||
uInt16 size; // size of a bank (in bytes)
|
||||
AddressList addressList; // addresses which PC has hit
|
||||
DirectiveList directiveList; // overrides for automatic code determination
|
||||
} BankInfo;
|
||||
|
@ -268,7 +269,10 @@ class CartDebug : public DebuggerSystem
|
|||
|
||||
// Analyze of bank of ROM, generating a list of Distella directives
|
||||
// based on its disassembly
|
||||
void getBankDirectives(ostringstream& buf, BankInfo& info) const;
|
||||
void getBankDirectives(ostream& buf, BankInfo& info) const;
|
||||
|
||||
// Convert disassembly enum type to corresponding string and append to buf
|
||||
void disasmTypeAsString(ostream& buf, DisasmType type) const;
|
||||
|
||||
// Extract labels and values from the given character stream
|
||||
string extractLabel(const char* c) const;
|
||||
|
|
|
@ -209,11 +209,7 @@ void Debugger::setConsole(Console* console)
|
|||
myCpuDebug = new CpuDebug(*this, *myConsole);
|
||||
|
||||
delete myCartDebug;
|
||||
// Register any RAM areas in the Cartridge
|
||||
// Zero-page RAM is automatically recognized by CartDebug
|
||||
myCartDebug = new CartDebug(*this, *myConsole, *myOSystem);
|
||||
myCartDebug->loadSymbolFile(myOSystem->romFile());
|
||||
myCartDebug->loadConfigFile();
|
||||
|
||||
delete myRiotDebug;
|
||||
myRiotDebug = new RiotDebug(*this, *myConsole);
|
||||
|
@ -770,7 +766,7 @@ string Debugger::builtinHelp() const
|
|||
if(len > i_maxlen) i_maxlen = len;
|
||||
}
|
||||
|
||||
buf << endl << "Built-in functions:" << endl;
|
||||
buf << setfill(' ') << endl << "Built-in functions:" << endl;
|
||||
for(int i = 0; builtin_functions[i][0] != 0; ++i)
|
||||
{
|
||||
buf << setw(c_maxlen) << left << builtin_functions[i][0]
|
||||
|
|
|
@ -39,7 +39,7 @@ DiStella::DiStella(const CartDebug& dbg, CartDebug::DisassemblyList& list,
|
|||
|
||||
if(start & 0x1000)
|
||||
{
|
||||
if(info.banksize == 4096) // 4K ROM space
|
||||
if(info.size == 4096) // 4K ROM space
|
||||
{
|
||||
/*============================================
|
||||
The offset is the address where the code segment
|
||||
|
@ -76,7 +76,7 @@ DiStella::DiStella(const CartDebug& dbg, CartDebug::DisassemblyList& list,
|
|||
|
||||
myOffset = 0;
|
||||
}
|
||||
myAppData.length = info.banksize;
|
||||
myAppData.length = info.size;
|
||||
|
||||
info.start = myAppData.start;
|
||||
info.end = myAppData.end;
|
||||
|
@ -785,6 +785,9 @@ void DiStella::addEntry(CartDebug::DisasmType type)
|
|||
myDisasmBuf.seekg(11, ios::beg);
|
||||
switch(tag.type)
|
||||
{
|
||||
case CartDebug::BLOCK:
|
||||
// TODO - handle this
|
||||
break;
|
||||
case CartDebug::CODE:
|
||||
getline(myDisasmBuf, tag.disasm, '\'');
|
||||
getline(myDisasmBuf, tag.ccount, '\'');
|
||||
|
|
|
@ -497,6 +497,8 @@ void PromptWidget::loadConfig()
|
|||
|
||||
// Take care of one-time debugger stuff
|
||||
print(instance().debugger().autoExec().c_str());
|
||||
print(instance().debugger().cartDebug().loadConfigFile() + "\n");
|
||||
print(instance().debugger().cartDebug().loadSymbolFile() + "\n");
|
||||
print(PROMPT);
|
||||
|
||||
_promptStartPos = _promptEndPos = _currentPos;
|
||||
|
|
|
@ -817,8 +817,12 @@ uInt8* OSystem::openROM(string file, string& md5, uInt32& size)
|
|||
if(!myPropSet->getMD5(md5, props))
|
||||
{
|
||||
// Get the filename from the rom pathname
|
||||
string::size_type pos = file.find_last_of("/\\");
|
||||
if(pos != string::npos) file = file.substr(pos+1);
|
||||
FilesystemNode node(file);
|
||||
file = node.getDisplayName();
|
||||
|
||||
// Remove extension
|
||||
string::size_type pos = file.find_last_of(".");
|
||||
if(pos != string::npos) file = file.substr(0, pos);
|
||||
|
||||
props.set(Cartridge_MD5, md5);
|
||||
props.set(Cartridge_Name, file);
|
||||
|
|
Loading…
Reference in New Issue