Created a new class RiotRamWidget, which takes the place of the

old RamWidget.  Class RawWidget is now an abstract base class
to RiotRamWidget and CartRamWidget, allowing to share quite a lot
of code.  In the process, both classes now implement input from
the binary and decimal widgets, not just the DataGridWidget.


git-svn-id: svn://svn.code.sf.net/p/stella/code/trunk@3195 8b62c5a3-ac7e-4cc8-8f21-d9a121418aba
This commit is contained in:
stephena 2015-09-07 23:12:00 +00:00
parent 62fa17b7fa
commit cc7943610d
15 changed files with 465 additions and 639 deletions

View File

@ -12,7 +12,7 @@
Release History Release History
=========================================================================== ===========================================================================
4.6.1 to 4.7: (July dd, 2015) 4.6.1 to 4.6.5: (September dd, 2015)
* Added mappable events for toggling TV color/BW, left difficulty A/B * Added mappable events for toggling TV color/BW, left difficulty A/B
and right difficulty A/B. This means that one key, joystick button, and right difficulty A/B. This means that one key, joystick button,

View File

@ -40,7 +40,7 @@ using CartMethod = int (CartDebug::*)();
class CartState : public DebuggerState class CartState : public DebuggerState
{ {
public: public:
IntArray ram; // The actual data values ByteArray ram; // The actual data values
IntArray rport; // Address for reading from RAM IntArray rport; // Address for reading from RAM
IntArray wport; // Address for writing to RAM IntArray wport; // Address for writing to RAM
string bank; // Current banking layout string bank; // Current banking layout

View File

@ -17,8 +17,12 @@
// $Id$ // $Id$
//============================================================================ //============================================================================
#include "PopUpWidget.hxx" #include "EditTextWidget.hxx"
#include "InputTextDialog.hxx" #include "GuiObject.hxx"
#include "OSystem.hxx"
#include "CartDebug.hxx"
#include "Widget.hxx"
#include "CartRamWidget.hxx" #include "CartRamWidget.hxx"
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -31,27 +35,23 @@ CartRamWidget::CartRamWidget(
myFontWidth(lfont.getMaxCharWidth()), myFontWidth(lfont.getMaxCharWidth()),
myFontHeight(lfont.getFontHeight()), myFontHeight(lfont.getFontHeight()),
myLineHeight(lfont.getLineHeight()), myLineHeight(lfont.getLineHeight()),
myButtonHeight(myLineHeight + 4), myButtonHeight(myLineHeight + 4)
myCart(cartDebug),
myCurrentRamBank(0)
{ {
int lwidth = lfont.getStringWidth("Description: "), int lwidth = lfont.getStringWidth("Description: "),
fwidth = w - lwidth - 20; fwidth = w - lwidth - 20;
const int bwidth = lfont.getStringWidth("Compare "),
bheight = myLineHeight + 2;
EditTextWidget* etw = nullptr; EditTextWidget* etw = nullptr;
ostringstream buf; ostringstream buf;
int xpos = 2, ypos = 5; int xpos = 2, ypos = 5;
// Add RAM size // Add RAM size
myRamSize = myCart.internalRamSize();
new StaticTextWidget(_boss, _font, xpos, ypos, lwidth, new StaticTextWidget(_boss, _font, xpos, ypos, lwidth,
myFontHeight, "RAM Size: ", kTextAlignLeft); myFontHeight, "RAM Size: ", kTextAlignLeft);
buf << myRamSize << " bytes"; uInt32 ramsize = cartDebug.internalRamSize();
if(myRamSize >= 1024) buf << ramsize << " bytes";
buf << " / " << (myRamSize/1024) << "KB"; if(ramsize >= 1024)
buf << " / " << (ramsize/1024) << "KB";
etw = new EditTextWidget(boss, nfont, xpos+lwidth, ypos, etw = new EditTextWidget(boss, nfont, xpos+lwidth, ypos,
fwidth, myLineHeight, buf.str()); fwidth, myLineHeight, buf.str());
@ -59,7 +59,7 @@ CartRamWidget::CartRamWidget(
ypos += myLineHeight + 4; ypos += myLineHeight + 4;
// Add Description // Add Description
const string& desc = myCart.internalRamDescription(); const string& desc = cartDebug.internalRamDescription();
const uInt16 maxlines = 6; const uInt16 maxlines = 6;
StringParser bs(desc, (fwidth - kScrollBarWidth) / myFontWidth); StringParser bs(desc, (fwidth - kScrollBarWidth) / myFontWidth);
const StringList& sl = bs.stringList(); const StringList& sl = bs.stringList();
@ -77,437 +77,97 @@ CartRamWidget::CartRamWidget(
ypos += myDesc->getHeight() + myLineHeight + 4; ypos += myDesc->getHeight() + myLineHeight + 4;
// Add RAM grid // Add RAM widget
xpos = _font.getStringWidth("xxxx"); xpos = x + _font.getStringWidth("xxxx");
myNumRows = myRamSize / 16; myRam = new InternalRamWidget(boss, lfont, nfont, 2, ypos, w, h-ypos, cartDebug);
if (myNumRows > 16) myNumRows = 16; addToFocusList(myRam->getFocusList());
myPageSize = myNumRows * 16; }
myRamGrid = new DataGridWidget(_boss, _nfont, xpos, ypos,
16, myNumRows, 2, 8, Common::Base::F_16, true);
myRamGrid->setTarget(this);
addFocusWidget(myRamGrid);
// Create actions buttons to the left of the RAM grid // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
int bx = xpos + myRamGrid->getWidth() + 4; CartRamWidget::~CartRamWidget()
int by = ypos; {
myUndoButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Undo", kUndoCmd);
myUndoButton->setTarget(this);
by += bheight + 4;
myRevertButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Revert", kRevertCmd);
myRevertButton->setTarget(this);
by += 2 * bheight + 2;
mySearchButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Search", kSearchCmd);
mySearchButton->setTarget(this);
by += bheight + 4;
myCompareButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Compare", kCmpCmd);
myCompareButton->setTarget(this);
by += bheight + 4;
myRestartButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Reset", kRestartCmd);
myRestartButton->setTarget(this);
// Labels for RAM grid
for(int col = 0; col < 16; ++col)
{
new StaticTextWidget(_boss, _font, xpos + col*myRamGrid->colWidth() + 8,
ypos - myLineHeight,
myFontWidth, myFontHeight,
Common::Base::toString(col, Common::Base::F_16_1),
kTextAlignLeft);
}
myRamStart =
new StaticTextWidget(_boss, _font, xpos - _font.getStringWidth("xxxx"), ypos - myLineHeight,
_font.getStringWidth("xxxx"), myFontHeight,
"F0xx", kTextAlignLeft);
uInt32 row;
for(row = 0; row < myNumRows; ++row)
{
myRamLabels[row] =
new StaticTextWidget(_boss, _font, xpos - _font.getStringWidth("x "),
ypos + row*myLineHeight + 2,
myFontWidth, myFontHeight, "", kTextAlignLeft);
}
// for smaller grids, make sure RAM cell detail fields are below the RESET button
if (myNumRows < 8)
row = 8 + 1;
else
row = myNumRows + 1;
ypos += myLineHeight * row;
// We need to define these widgets from right to left since the leftmost
// one resizes as much as possible
// Add Binary display of selected RAM cell
xpos = w - 13*myFontWidth - 20;
new StaticTextWidget(boss, lfont, xpos, ypos, 4*myFontWidth, myFontHeight,
"Bin:", kTextAlignLeft);
myBinValue = new EditTextWidget(boss, nfont, xpos + 4*myFontWidth + 5,
ypos-2, 9*myFontWidth, myLineHeight, "");
myBinValue->setEditable(false);
// Add Decimal display of selected RAM cell
xpos -= 8*myFontWidth + 5 + 20;
new StaticTextWidget(boss, lfont, xpos, ypos, 4*myFontWidth, myFontHeight,
"Dec:", kTextAlignLeft);
myDecValue = new EditTextWidget(boss, nfont, xpos + 4*myFontWidth + 5, ypos-2,
4*myFontWidth, myLineHeight, "");
myDecValue->setEditable(false);
// Add Label of selected RAM cell
int xpos_r = xpos - 20;
xpos = x + 10;
new StaticTextWidget(boss, lfont, xpos, ypos, 6*myFontWidth, myFontHeight,
"Label:", kTextAlignLeft);
xpos += 6*myFontWidth + 5;
myLabel = new EditTextWidget(boss, nfont, xpos, ypos-2, xpos_r-xpos,
myLineHeight, "");
myLabel->setEditable(false);
// Inputbox which will pop up when searching RAM
StringList labels;
labels.push_back("Search: ");
myInputBox = new InputTextDialog(boss, lfont, nfont, labels);
myInputBox->setTarget(this);
// Start with these buttons disabled
myCompareButton->clearFlags(WIDGET_ENABLED);
myRestartButton->clearFlags(WIDGET_ENABLED);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::loadConfig() void CartRamWidget::loadConfig()
{ {
fillGrid(true); myRam->loadConfig();
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::fillGrid(bool updateOld) void CartRamWidget::setOpsWidget(DataGridOpsWidget* w)
{ {
IntArray alist; myRam->setOpsWidget(w);
IntArray vlist; }
BoolArray changed;
uInt32 start = myCurrentRamBank * myPageSize;
const ByteArray& oldRam = myCart.internalRamOld(start, myPageSize);
const ByteArray& currentRam = myCart.internalRamCurrent(start, myPageSize);
for(uInt32 i = 0; i < myPageSize; i++) // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::handleCommand(CommandSender* sender, int cmd, int data, int id)
{
myRam->handleCommand(sender, cmd, data, id);
}
///////////////////////////////////
// Internal RAM implementation
///////////////////////////////////
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
CartRamWidget::InternalRamWidget::InternalRamWidget(GuiObject* boss,
const GUI::Font& lfont, const GUI::Font& nfont,
int x, int y, int w, int h,
CartDebugWidget& dbg)
: RamWidget(boss, lfont, nfont, x, y, w, h,
dbg.internalRamSize(), BSPF_min(dbg.internalRamSize() / 16, 16u),
BSPF_min(dbg.internalRamSize() / 16, 16u) * 16),
myCart(dbg)
{
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
CartRamWidget::InternalRamWidget::~InternalRamWidget()
{
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt8 CartRamWidget::InternalRamWidget::getValue(int addr) const
{
return myCart.internalRamGetValue(addr);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::InternalRamWidget::setValue(int addr, uInt8 value)
{
myCart.internalRamSetValue(addr, value);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
string CartRamWidget::InternalRamWidget::getLabel(int addr) const
{
return myCart.internalRamLabel(addr);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::InternalRamWidget::fillList(uInt32 start, uInt32 size,
IntArray& alist, IntArray& vlist, BoolArray& changed) const
{
const ByteArray& oldRam = myCart.internalRamOld(start, size);
const ByteArray& currRam = myCart.internalRamCurrent(start, size);
for(uInt32 i = 0; i < size; i++)
{ {
alist.push_back(i+start); alist.push_back(i+start);
vlist.push_back(currentRam[i]); vlist.push_back(currRam[i]);
changed.push_back(currentRam[i] != oldRam[i]); changed.push_back(currRam[i] != oldRam[i]);
} }
if(updateOld)
{
myOldValueList.clear();
myOldValueList = myCart.internalRamCurrent(start, myCart.internalRamSize());
}
myRamGrid->setNumRows(myRamSize / myPageSize);
myRamGrid->setList(alist, vlist, changed);
if(updateOld)
{
myRevertButton->setEnabled(false);
myUndoButton->setEnabled(false);
}
// Update RAM labels
uInt32 rport = myCart.internalRamRPort(start);
uInt32 page = rport & 0xf0;
char buf[5];
BSPF_snprintf(buf, 5, "%04X", rport);
buf[2] = buf[3] = 'x';
myRamStart->setLabel(buf);
for(uInt32 row = 0; row < myNumRows; ++row, page += 0x10)
myRamLabels[row]->setLabel(Common::Base::toString(page>>4, Common::Base::F_16_1));
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::showInputBox(int cmd) uInt32 CartRamWidget::InternalRamWidget::readPort(uInt32 start) const
{ {
// Add inputbox in the middle of the RAM widget return myCart.internalRamRPort(start);
uInt32 x = getAbsX() + ((getWidth() - myInputBox->getWidth()) >> 1);
uInt32 y = getAbsY() + ((getHeight() - myInputBox->getHeight()) >> 1);
myInputBox->show(x, y);
myInputBox->setText("");
myInputBox->setTitle("");
myInputBox->setFocus(0);
myInputBox->setEmitSignal(cmd);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
string CartRamWidget::doSearch(const string& str) const ByteArray& CartRamWidget::InternalRamWidget::currentRam(uInt32 start) const
{ {
bool comparisonSearch = true; return myCart.internalRamCurrent(start, myCart.internalRamSize());
if(str.length() == 0)
{
// An empty field means return all memory locations
comparisonSearch = false;
}
else if(str.find_first_of("+-", 0) != string::npos)
{
// Don't accept these characters here, only in compare
return "Invalid input +|-";
}
int searchVal = instance().debugger().stringToValue(str);
// Clear the search array of previous items
mySearchAddr.clear();
mySearchValue.clear();
mySearchState.clear();
// Now, search all memory locations for this value, and add it to the
// search array
bool hitfound = false;
// CartDebug& dbg = instance().debugger().cartDebug();
// const CartState& state = (CartState&) dbg.getState();
const ByteArray& currentRam = myCart.internalRamCurrent(0, myCart.internalRamSize());
for(uInt32 addr = 0; addr < myCart.internalRamSize(); ++addr)
{
int value = currentRam[addr];
if(comparisonSearch && searchVal != value)
{
mySearchState.push_back(false);
}
else
{
mySearchAddr.push_back(addr);
mySearchValue.push_back(value);
mySearchState.push_back(true);
hitfound = true;
}
}
// If we have some hits, enable the comparison methods
if(hitfound)
{
mySearchButton->setEnabled(false);
myCompareButton->setEnabled(true);
myRestartButton->setEnabled(true);
}
// Finally, show the search results in the list
showSearchResults();
return EmptyString;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
string CartRamWidget::doCompare(const string& str)
{
bool comparitiveSearch = false;
int searchVal = 0, offset = 0;
if(str.length() == 0)
return "Enter an absolute or comparitive value";
// Do some pre-processing on the string
string::size_type pos = str.find_first_of("+-", 0);
if(pos > 0 && pos != string::npos)
{
// Only accept '+' or '-' at the start of the string
return "Input must be [+|-]NUM";
}
// A comparitive search searches memory for locations that have changed by
// the specified amount, vs. for exact values
if(str[0] == '+' || str[0] == '-')
{
comparitiveSearch = true;
bool negative = false;
if(str[0] == '-')
negative = true;
string tmp = str;
tmp.erase(0, 1); // remove the operator
offset = instance().debugger().stringToValue(tmp);
if(negative)
offset = -offset;
}
else
searchVal = instance().debugger().stringToValue(str);
// Now, search all memory locations previously 'found' for this value
bool hitfound = false;
const ByteArray& currentRam = myCart.internalRamCurrent(0, myCart.internalRamSize());
IntArray tempAddrList, tempValueList;
mySearchState.clear();
for(uInt32 i = 0; i < myCart.internalRamSize(); ++i)
mySearchState.push_back(false);
for(uInt32 i = 0; i < mySearchAddr.size(); ++i)
{
if(comparitiveSearch)
{
searchVal = mySearchValue[i] + offset;
if(searchVal < 0 || searchVal > 255)
continue;
}
int addr = mySearchAddr[i];
if(currentRam[addr] == searchVal)
{
tempAddrList.push_back(addr);
tempValueList.push_back(searchVal);
mySearchState[addr] = hitfound = true;
}
}
// Update the searchArray for the new addresses and data
mySearchAddr = tempAddrList;
mySearchValue = tempValueList;
// If we have some hits, enable the comparison methods
if(hitfound)
{
myCompareButton->setEnabled(true);
myRestartButton->setEnabled(true);
}
// Finally, show the search results in the list
showSearchResults();
return EmptyString;
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::handleCommand(CommandSender* sender,
int cmd, int data, int id)
{
// We simply change the values in the DataGridWidget
// It will then send the 'kDGItemDataChangedCmd' signal to change the actual
// memory location
int addr, value;
switch(cmd)
{
case DataGridWidget::kItemDataChangedCmd:
{
addr = myRamGrid->getSelectedAddr();
value = myRamGrid->getSelectedValue();
uInt8 oldval = myCart.internalRamGetValue(addr);
myCart.internalRamSetValue(addr, value);
myUndoAddress = addr;
myUndoValue = oldval;
myDecValue->setText(Common::Base::toString(value, Common::Base::F_10));
myBinValue->setText(Common::Base::toString(value, Common::Base::F_2));
myRevertButton->setEnabled(true);
myUndoButton->setEnabled(true);
break;
}
case DataGridWidget::kSelectionChangedCmd:
{
addr = myRamGrid->getSelectedAddr();
value = myRamGrid->getSelectedValue();
myLabel->setText(myCart.internalRamLabel(addr));
myDecValue->setText(Common::Base::toString(value, Common::Base::F_10));
myBinValue->setText(Common::Base::toString(value, Common::Base::F_2));
break;
}
case kRevertCmd:
for(uInt32 i = 0; i < myOldValueList.size(); ++i)
myCart.internalRamSetValue(i, myOldValueList[i]);
fillGrid(true);
break;
case kUndoCmd:
myCart.internalRamSetValue(myUndoAddress, myUndoValue);
myUndoButton->setEnabled(false);
fillGrid(false);
break;
case kSearchCmd:
showInputBox(kSValEntered);
break;
case kCmpCmd:
showInputBox(kCValEntered);
break;
case kRestartCmd:
doRestart();
break;
case kSValEntered:
{
const string& result = doSearch(myInputBox->getResult());
if(result != "")
myInputBox->setTitle(result);
else
myInputBox->close();
break;
}
case kCValEntered:
{
const string& result = doCompare(myInputBox->getResult());
if(result != "")
myInputBox->setTitle(result);
else
myInputBox->close();
break;
}
case kSetPositionCmd:
myCurrentRamBank = data;
showSearchResults();
fillGrid(false);
break;
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::doRestart()
{
// Erase all search buffers, reset to start mode
mySearchAddr.clear();
mySearchValue.clear();
mySearchState.clear();
showSearchResults();
mySearchButton->setEnabled(true);
myCompareButton->setEnabled(false);
myRestartButton->setEnabled(false);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void CartRamWidget::showSearchResults()
{
// Only update the search results for the bank currently being shown
BoolArray temp;
uInt32 start = myCurrentRamBank * myPageSize;
if(mySearchState.size() == 0 || start > mySearchState.size())
{
for(uInt32 i = 0; i < myPageSize; ++i)
temp.push_back(false);
}
else
{
for(uInt32 i = start; i < start + myPageSize; ++i)
temp.push_back(mySearchState[i]);
}
myRamGrid->setHiliteList(temp);
}

View File

@ -21,21 +21,14 @@
#define CART_RAM_WIDGET_HXX #define CART_RAM_WIDGET_HXX
class GuiObject; class GuiObject;
class InputTextDialog; class DataGridOpsWidget;
class ButtonWidget; class StringListWidget;
class DataGridWidget; class InternalRamWidget;
#include "Base.hxx"
#include "Font.hxx"
#include "CartDebugWidget.hxx" #include "CartDebugWidget.hxx"
#include "Command.hxx" #include "RamWidget.hxx"
#include "DataGridWidget.hxx"
#include "Debugger.hxx"
#include "RomWidget.hxx"
#include "Widget.hxx" #include "Widget.hxx"
#include "EditTextWidget.hxx" #include "Command.hxx"
#include "StringListWidget.hxx"
#include "StringParser.hxx"
class CartRamWidget : public Widget, public CommandSender class CartRamWidget : public Widget, public CommandSender
{ {
@ -43,19 +36,13 @@ class CartRamWidget : public Widget, public CommandSender
CartRamWidget(GuiObject* boss, const GUI::Font& lfont, CartRamWidget(GuiObject* boss, const GUI::Font& lfont,
const GUI::Font& nfont, const GUI::Font& nfont,
int x, int y, int w, int h, CartDebugWidget& cartDebug); int x, int y, int w, int h, CartDebugWidget& cartDebug);
virtual ~CartRamWidget();
virtual ~CartRamWidget() { }; void loadConfig() override;
void setOpsWidget(DataGridOpsWidget* w);
private: private:
void loadConfig() override;
void handleCommand(CommandSender* sender, int cmd, int data, int id) override; void handleCommand(CommandSender* sender, int cmd, int data, int id) override;
void fillGrid(bool updateOld);
void showInputBox(int cmd);
string doSearch(const string& str);
string doCompare(const string& str);
void doRestart();
void showSearchResults();
protected: protected:
// Font used for 'normal' text; _font is for 'label' text // Font used for 'normal' text; _font is for 'label' text
@ -65,48 +52,6 @@ class CartRamWidget : public Widget, public CommandSender
// we may as well make them protected variables // we may as well make them protected variables
int myFontWidth, myFontHeight, myLineHeight, myButtonHeight; int myFontWidth, myFontHeight, myLineHeight, myButtonHeight;
private:
enum {
kUndoCmd = 'RWud',
kRevertCmd = 'RWrv',
kSearchCmd = 'RWse',
kCmpCmd = 'RWcp',
kRestartCmd = 'RWrs',
kSValEntered = 'RWsv',
kCValEntered = 'RWcv'
};
int myUndoAddress;
int myUndoValue;
CartDebugWidget& myCart;
StringListWidget* myDesc;
DataGridWidget* myRamGrid;
StaticTextWidget* myRamStart;
StaticTextWidget* myRamLabels[16];
EditTextWidget* myBinValue;
EditTextWidget* myDecValue;
EditTextWidget* myLabel;
uInt32 myCurrentRamBank;
uInt32 myRamSize;
uInt32 myPageSize;
uInt32 myNumRows;
ButtonWidget* myRevertButton;
ButtonWidget* myUndoButton;
ButtonWidget* mySearchButton;
ButtonWidget* myCompareButton;
ButtonWidget* myRestartButton;
InputTextDialog* myInputBox;
ByteArray myOldValueList;
IntArray mySearchAddr;
IntArray mySearchValue;
BoolArray mySearchState;
private: private:
// Following constructors and assignment operators not supported // Following constructors and assignment operators not supported
CartRamWidget() = delete; CartRamWidget() = delete;
@ -114,6 +59,42 @@ class CartRamWidget : public Widget, public CommandSender
CartRamWidget(CartRamWidget&&) = delete; CartRamWidget(CartRamWidget&&) = delete;
CartRamWidget& operator=(const CartRamWidget&) = delete; CartRamWidget& operator=(const CartRamWidget&) = delete;
CartRamWidget& operator=(CartRamWidget&&) = delete; CartRamWidget& operator=(CartRamWidget&&) = delete;
private:
class InternalRamWidget : public RamWidget
{
public:
InternalRamWidget(GuiObject* boss, const GUI::Font& lfont,
const GUI::Font& nfont,
int x, int y, int w, int h,
CartDebugWidget& cartDebug);
virtual ~InternalRamWidget();
private:
uInt8 getValue(int addr) const;
void setValue(int addr, uInt8 value);
string getLabel(int addr) const;
void fillList(uInt32 start, uInt32 size, IntArray& alist,
IntArray& vlist, BoolArray& changed) const;
uInt32 readPort(uInt32 start) const;
const ByteArray& currentRam(uInt32 start) const;
private:
CartDebugWidget& myCart;
private:
// Following constructors and assignment operators not supported
InternalRamWidget() = delete;
InternalRamWidget(const InternalRamWidget&) = delete;
InternalRamWidget(InternalRamWidget&&) = delete;
InternalRamWidget& operator=(const InternalRamWidget&) = delete;
InternalRamWidget& operator=(InternalRamWidget&&) = delete;
};
private:
StringListWidget* myDesc;
InternalRamWidget* myRam;
}; };
#endif #endif

View File

@ -215,9 +215,9 @@ void DataGridWidget::setValue(int position, int value)
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void DataGridWidget::setValueInternal(int position, int value) void DataGridWidget::setValueInternal(int position, int value, bool changed)
{ {
setValue(position, value, true, false); setValue(position, value, changed, false);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -

View File

@ -64,7 +64,7 @@ class DataGridWidget : public EditableWidget
/** Set value at given position */ /** Set value at given position */
void setValue(int position, int value); void setValue(int position, int value);
/** Set value at given position, do not emit any signals */ /** Set value at given position, do not emit any signals */
void setValueInternal(int position, int value); void setValueInternal(int position, int value, bool changed = true);
/** Set value at given position, manually specifying if the value changed */ /** Set value at given position, manually specifying if the value changed */
void setValue(int position, int value, bool changed, bool emitSignal = true); void setValue(int position, int value, bool changed, bool emitSignal = true);

View File

@ -26,7 +26,7 @@
#include "AudioWidget.hxx" #include "AudioWidget.hxx"
#include "PromptWidget.hxx" #include "PromptWidget.hxx"
#include "CpuWidget.hxx" #include "CpuWidget.hxx"
#include "RamWidget.hxx" #include "RiotRamWidget.hxx"
#include "RiotWidget.hxx" #include "RiotWidget.hxx"
#include "RomWidget.hxx" #include "RomWidget.hxx"
#include "TiaWidget.hxx" #include "TiaWidget.hxx"
@ -390,7 +390,7 @@ void DebuggerDialog::addRomArea()
addToFocusList(myCpu->getFocusList()); addToFocusList(myCpu->getFocusList());
xpos = r.left + 10; ypos += myCpu->getHeight() + 10; xpos = r.left + 10; ypos += myCpu->getHeight() + 10;
myRam = new RamWidget(this, *myLFont, *myNFont, xpos, ypos); myRam = new RiotRamWidget(this, *myLFont, *myNFont, xpos, ypos, r.width() - 10);
addToFocusList(myRam->getFocusList()); addToFocusList(myRam->getFocusList());
// Add the DataGridOpsWidget to any widgets which contain a // Add the DataGridOpsWidget to any widgets which contain a
@ -434,12 +434,14 @@ void DebuggerDialog::addRomArea()
if (myCartDebug->internalRamSize() > 0) if (myCartDebug->internalRamSize() > 0)
{ {
tabID = myRomTab->addTab(" Cartridge RAM "); tabID = myRomTab->addTab(" Cartridge RAM ");
myCartRam = new CartRamWidget(myRomTab, *myLFont, *myNFont, 2, 2, tabWidth - 1, myCartRam =
tabHeight - myRomTab->getTabHeight() - 2, *myCartDebug); new CartRamWidget(myRomTab, *myLFont, *myNFont, 2, 2, tabWidth - 1,
tabHeight - myRomTab->getTabHeight() - 2, *myCartDebug);
if(myCartRam) // TODO - make this always non-null if(myCartRam) // TODO - make this always non-null
{ {
myRomTab->setParentWidget(tabID, myCartRam); myRomTab->setParentWidget(tabID, myCartRam);
addToFocusList(myCartRam->getFocusList(), myRomTab, tabID); addToFocusList(myCartRam->getFocusList(), myRomTab, tabID);
myCartRam->setOpsWidget(ops);
} }
} }
} }

View File

@ -17,11 +17,8 @@
// $Id$ // $Id$
//============================================================================ //============================================================================
#include <sstream>
#include "DataGridWidget.hxx" #include "DataGridWidget.hxx"
#include "EditTextWidget.hxx" #include "EditTextWidget.hxx"
#include "FrameBuffer.hxx"
#include "GuiObject.hxx" #include "GuiObject.hxx"
#include "InputTextDialog.hxx" #include "InputTextDialog.hxx"
#include "OSystem.hxx" #include "OSystem.hxx"
@ -32,119 +29,135 @@
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
RamWidget::RamWidget(GuiObject* boss, const GUI::Font& lfont, const GUI::Font& nfont, RamWidget::RamWidget(GuiObject* boss, const GUI::Font& lfont, const GUI::Font& nfont,
int x, int y) int x, int y, int w, int h,
: Widget(boss, lfont, x, y, 16, 16), uInt32 ramsize, uInt32 numrows, uInt32 pagesize)
: Widget(boss, lfont, x, y, w, h),
CommandSender(boss), CommandSender(boss),
myUndoAddress(-1), _nfont(nfont),
myUndoValue(-1), myFontWidth(lfont.getMaxCharWidth()),
myCurrentRamBank(0) myFontHeight(lfont.getFontHeight()),
myLineHeight(lfont.getLineHeight()),
myButtonHeight(myLineHeight + 4),
myCurrentRamBank(0),
myRamSize(ramsize),
myNumRows(numrows),
myPageSize(pagesize)
{ {
const int fontWidth = lfont.getMaxCharWidth(),
fontHeight = lfont.getFontHeight(),
lineHeight = lfont.getLineHeight(),
bwidth = lfont.getStringWidth("Compare "),
bheight = lineHeight + 2;
int xpos, ypos, lwidth; int xpos, ypos, lwidth;
const int bwidth = lfont.getStringWidth("Compare "),
bheight = myLineHeight + 2;
// Create a 16x8 grid holding byte values (16 x 8 = 128 RAM bytes) with labels ypos = y + myLineHeight; lwidth = 4 * myFontWidth;
// Add a scrollbar, since there may be more than 128 bytes of RAM available
xpos = x; ypos = y + lineHeight; lwidth = 4 * fontWidth; // Add RAM grid (with scrollbar)
myRamGrid = new DataGridWidget(boss, nfont, xpos + lwidth, ypos, xpos = x + _font.getStringWidth("xxxx");
16, 8, 2, 8, Common::Base::F_16, true); myRamGrid = new DataGridWidget(_boss, _nfont, xpos, ypos,
16, myNumRows, 2, 8, Common::Base::F_16, true);
myRamGrid->setTarget(this); myRamGrid->setTarget(this);
myRamGrid->setID(kRamHexID);
addFocusWidget(myRamGrid); addFocusWidget(myRamGrid);
// Create actions buttons to the left of the RAM grid // Create actions buttons to the left of the RAM grid
xpos += lwidth + myRamGrid->getWidth() + 4; int bx = xpos + myRamGrid->getWidth() + 4;
myUndoButton = new ButtonWidget(boss, lfont, xpos, ypos, bwidth, bheight, int by = ypos;
myUndoButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Undo", kUndoCmd); "Undo", kUndoCmd);
myUndoButton->setTarget(this); myUndoButton->setTarget(this);
ypos += bheight + 4; by += bheight + 4;
myRevertButton = new ButtonWidget(boss, lfont, xpos, ypos, bwidth, bheight, myRevertButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Revert", kRevertCmd); "Revert", kRevertCmd);
myRevertButton->setTarget(this); myRevertButton->setTarget(this);
ypos += 2 * bheight + 2; by += 2 * bheight + 2;
mySearchButton = new ButtonWidget(boss, lfont, xpos, ypos, bwidth, bheight, mySearchButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Search", kSearchCmd); "Search", kSearchCmd);
mySearchButton->setTarget(this); mySearchButton->setTarget(this);
ypos += bheight + 4; by += bheight + 4;
myCompareButton = new ButtonWidget(boss, lfont, xpos, ypos, bwidth, bheight, myCompareButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Compare", kCmpCmd); "Compare", kCmpCmd);
myCompareButton->setTarget(this); myCompareButton->setTarget(this);
ypos += bheight + 4; by += bheight + 4;
myRestartButton = new ButtonWidget(boss, lfont, xpos, ypos, bwidth, bheight, myRestartButton = new ButtonWidget(boss, lfont, bx, by, bwidth, bheight,
"Reset", kRestartCmd); "Reset", kRestartCmd);
myRestartButton->setTarget(this); myRestartButton->setTarget(this);
// Remember position of right side of buttons
int xpos_r = xpos + bwidth ;
// Labels for RAM grid // Labels for RAM grid
xpos = x; ypos = y + lineHeight;
myRamStart = myRamStart =
new StaticTextWidget(boss, lfont, xpos, ypos - lineHeight, new StaticTextWidget(_boss, lfont, xpos - _font.getStringWidth("xxxx"),
lfont.getStringWidth("xxxx"), fontHeight, ypos - myLineHeight,
"00xx", kTextAlignLeft); lfont.getStringWidth("xxxx"), myFontHeight,
"00xx", kTextAlignLeft);
for(int col = 0; col < 16; ++col) for(int col = 0; col < 16; ++col)
{ {
new StaticTextWidget(boss, lfont, xpos + col*myRamGrid->colWidth() + lwidth + 8, new StaticTextWidget(_boss, lfont, xpos + col*myRamGrid->colWidth() + 8,
ypos - lineHeight, ypos - myLineHeight,
fontWidth, fontHeight, myFontWidth, myFontHeight,
Common::Base::toString(col, Common::Base::F_16_1), Common::Base::toString(col, Common::Base::F_16_1),
kTextAlignLeft); kTextAlignLeft);
} }
for(int row = 0; row < 8; ++row)
uInt32 row;
for(row = 0; row < myNumRows; ++row)
{ {
myRamLabels[row] = myRamLabels[row] =
new StaticTextWidget(boss, lfont, xpos + 8, ypos + row*lineHeight + 2, new StaticTextWidget(_boss, _font, xpos - _font.getStringWidth("x "),
3*fontWidth, fontHeight, "", kTextAlignLeft); ypos + row*myLineHeight + 2,
myFontWidth, myFontHeight, "", kTextAlignLeft);
} }
ypos += 9 * lineHeight; // For smaller grids, make sure RAM cell detail fields are below the RESET button
row = myNumRows < 8 ? 9 : myNumRows + 1;
ypos += row * myLineHeight;
// We need to define these widgets from right to left since the leftmost // We need to define these widgets from right to left since the leftmost
// one resizes as much as possible // one resizes as much as possible
xpos = xpos_r - 13*fontWidth - 5;
new StaticTextWidget(boss, lfont, xpos, ypos, 4*fontWidth, fontHeight, // Add Binary display of selected RAM cell
xpos = x + w - 13*myFontWidth - 20;
new StaticTextWidget(boss, lfont, xpos, ypos, 4*myFontWidth, myFontHeight,
"Bin:", kTextAlignLeft); "Bin:", kTextAlignLeft);
myBinValue = new EditTextWidget(boss, nfont, xpos + 4*fontWidth + 5, myBinValue = new DataGridWidget(boss, nfont, xpos + 4*myFontWidth + 5, ypos-2,
ypos-2, 9*fontWidth, lineHeight, ""); 1, 1, 8, 8, Common::Base::F_2);
myBinValue->setEditable(false); myBinValue->setTarget(this);
myBinValue->setID(kRamBinID);
xpos -= 8*fontWidth + 5 + 20; // Add Decimal display of selected RAM cell
new StaticTextWidget(boss, lfont, xpos, ypos, 4*fontWidth, fontHeight, xpos -= 8*myFontWidth + 5 + 20;
new StaticTextWidget(boss, lfont, xpos, ypos, 4*myFontWidth, myFontHeight,
"Dec:", kTextAlignLeft); "Dec:", kTextAlignLeft);
myDecValue = new EditTextWidget(boss, nfont, xpos + 4*fontWidth + 5, ypos-2, myDecValue = new DataGridWidget(boss, nfont, xpos + 4*myFontWidth + 5, ypos-2,
4*fontWidth, lineHeight, ""); 1, 1, 3, 8, Common::Base::F_10);
myDecValue->setEditable(false); myDecValue->setTarget(this);
myDecValue->setID(kRamDecID);
xpos_r = xpos - 20; addFocusWidget(myDecValue);
addFocusWidget(myBinValue);
// Add Label of selected RAM cell
int xpos_r = xpos - 20;
xpos = x + 10; xpos = x + 10;
new StaticTextWidget(boss, lfont, xpos, ypos, 6*fontWidth, fontHeight, new StaticTextWidget(boss, lfont, xpos, ypos, 6*myFontWidth, myFontHeight,
"Label:", kTextAlignLeft); "Label:", kTextAlignLeft);
xpos += 6*fontWidth + 5; xpos += 6*myFontWidth + 5;
myLabel = new EditTextWidget(boss, nfont, xpos, ypos-2, xpos_r-xpos, myLabel = new EditTextWidget(boss, nfont, xpos, ypos-2, xpos_r-xpos,
lineHeight, ""); myLineHeight);
myLabel->setEditable(false); myLabel->setEditable(false);
// Calculate real dimensions
_w = lwidth + myRamGrid->getWidth();
_h = ypos + lineHeight - y;
// Inputbox which will pop up when searching RAM // Inputbox which will pop up when searching RAM
StringList labels; StringList labels = { "Search: " };
labels.push_back("Search: ");
myInputBox = make_ptr<InputTextDialog>(boss, lfont, nfont, labels); myInputBox = make_ptr<InputTextDialog>(boss, lfont, nfont, labels);
myInputBox->setTarget(this); myInputBox->setTarget(this);
// Start with these buttons disabled // Start with these buttons disabled
myCompareButton->clearFlags(WIDGET_ENABLED); myCompareButton->clearFlags(WIDGET_ENABLED);
myRestartButton->clearFlags(WIDGET_ENABLED); myRestartButton->clearFlags(WIDGET_ENABLED);
// Calculate final height
if(_h == 0) _h = ypos + myLineHeight - y;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -160,30 +173,38 @@ void RamWidget::handleCommand(CommandSender* sender, int cmd, int data, int id)
// memory location // memory location
int addr, value; int addr, value;
CartDebug& dbg = instance().debugger().cartDebug();
const CartState& state = (CartState&) dbg.getState();
switch(cmd) switch(cmd)
{ {
case DataGridWidget::kItemDataChangedCmd: case DataGridWidget::kItemDataChangedCmd:
{ {
addr = myRamGrid->getSelectedAddr(); switch(id)
value = myRamGrid->getSelectedValue();
// Attempt the write, and revert if it didn't succeed
uInt8 oldval = dbg.peek(state.rport[addr]);
dbg.poke(state.wport[addr], value);
uInt8 newval = dbg.peek(state.rport[addr]);
if(value != newval)
{ {
myRamGrid->setValue(addr - myCurrentRamBank*128, newval, false); case kRamHexID:
break; addr = myRamGrid->getSelectedAddr();
value = myRamGrid->getSelectedValue();
break;
case kRamDecID:
addr = myRamGrid->getSelectedAddr();
value = myDecValue->getSelectedValue();
break;
case kRamBinID:
addr = myRamGrid->getSelectedAddr();
value = myBinValue->getSelectedValue();
break;
} }
uInt8 oldval = getValue(addr);
setValue(addr, value);
myUndoAddress = addr; myUndoAddress = addr;
myUndoValue = oldval; myUndoValue = oldval;
myDecValue->setText(Common::Base::toString(value, Common::Base::F_10)); myRamGrid->setValueInternal(addr - myCurrentRamBank*myPageSize, value, true);
myBinValue->setText(Common::Base::toString(value, Common::Base::F_2)); myDecValue->setValueInternal(0, value, true);
myBinValue->setValueInternal(0, value, true);
myRevertButton->setEnabled(true); myRevertButton->setEnabled(true);
myUndoButton->setEnabled(true); myUndoButton->setEnabled(true);
break; break;
@ -194,20 +215,20 @@ void RamWidget::handleCommand(CommandSender* sender, int cmd, int data, int id)
addr = myRamGrid->getSelectedAddr(); addr = myRamGrid->getSelectedAddr();
value = myRamGrid->getSelectedValue(); value = myRamGrid->getSelectedValue();
myLabel->setText(dbg.getLabel(state.rport[addr], true)); myLabel->setText(getLabel(addr));
myDecValue->setText(Common::Base::toString(value, Common::Base::F_10)); myDecValue->setValueInternal(0, value, false);
myBinValue->setText(Common::Base::toString(value, Common::Base::F_2)); myBinValue->setValueInternal(0, value, false);
break; break;
} }
case kRevertCmd: case kRevertCmd:
for(uInt32 i = 0; i < myOldValueList.size(); ++i) for(uInt32 i = 0; i < myOldValueList.size(); ++i)
dbg.poke(state.wport[i], myOldValueList[i]); setValue(i, myOldValueList[i]);
fillGrid(true); fillGrid(true);
break; break;
case kUndoCmd: case kUndoCmd:
dbg.poke(state.wport[myUndoAddress], myUndoValue); setValue(myUndoAddress, myUndoValue);
myUndoButton->setEnabled(false); myUndoButton->setEnabled(false);
fillGrid(false); fillGrid(false);
break; break;
@ -256,6 +277,8 @@ void RamWidget::handleCommand(CommandSender* sender, int cmd, int data, int id)
void RamWidget::setOpsWidget(DataGridOpsWidget* w) void RamWidget::setOpsWidget(DataGridOpsWidget* w)
{ {
myRamGrid->setOpsWidget(w); myRamGrid->setOpsWidget(w);
myBinValue->setOpsWidget(w);
myDecValue->setOpsWidget(w);
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -271,29 +294,13 @@ void RamWidget::fillGrid(bool updateOld)
IntArray vlist; IntArray vlist;
BoolArray changed; BoolArray changed;
if(updateOld) myOldValueList.clear(); uInt32 start = myCurrentRamBank * myPageSize;
fillList(start, myPageSize, alist, vlist, changed);
CartDebug& dbg = instance().debugger().cartDebug(); if(updateOld)
myOldValueList = currentRam(start);
const CartState& state = (CartState&) dbg.getState(); myRamGrid->setNumRows(myRamSize / myPageSize);
const CartState& oldstate = (CartState&) dbg.getOldState();
// Jump to the correct 128 byte 'window' in the RAM area
// This assumes that the RAM areas are aligned on 128 byte boundaries
// TODO - the boundary restriction may not always apply ...
uInt32 start = myCurrentRamBank * 128;
assert(start+128 <= state.ram.size());
if(updateOld) myOldValueList = state.ram;
for(uInt32 i = start; i < start + 16*8; ++i)
{
alist.push_back(i);
vlist.push_back(state.ram[i]);
changed.push_back(state.ram[i] != oldstate.ram[i]);
}
myRamGrid->setNumRows((int)state.ram.size() / 128);
myRamGrid->setList(alist, vlist, changed); myRamGrid->setList(alist, vlist, changed);
if(updateOld) if(updateOld)
{ {
@ -302,15 +309,13 @@ void RamWidget::fillGrid(bool updateOld)
} }
// Update RAM labels // Update RAM labels
uInt32 rport = readPort(start), page = rport & 0xf0;
char buf[5]; char buf[5];
BSPF_snprintf(buf, 5, "%04X", state.rport[start] & 0xff00); BSPF_snprintf(buf, 5, "%04X", rport);
buf[2] = buf[3] = 'x'; buf[2] = buf[3] = 'x';
myRamStart->setLabel(buf); myRamStart->setLabel(buf);
for(uInt32 i = start, row = 0; i < start + 16*8; i += 16, ++row) for(uInt32 row = 0; row < myNumRows; ++row, page += 0x10)
{ myRamLabels[row]->setLabel(Common::Base::toString(page>>4, Common::Base::F_16_1));
BSPF_snprintf(buf, 3, "%02X", state.rport[i] & 0x00ff);
myRamLabels[row]->setLabel(buf);
}
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -351,12 +356,11 @@ string RamWidget::doSearch(const string& str)
// Now, search all memory locations for this value, and add it to the // Now, search all memory locations for this value, and add it to the
// search array // search array
const ByteArray& ram = currentRam(0);
bool hitfound = false; bool hitfound = false;
CartDebug& dbg = instance().debugger().cartDebug(); for(uInt32 addr = 0; addr < ram.size(); ++addr)
const CartState& state = (CartState&) dbg.getState();
for(uInt32 addr = 0; addr < state.ram.size(); ++addr)
{ {
int value = state.ram[addr]; int value = ram[addr];
if(comparisonSearch && searchVal != value) if(comparisonSearch && searchVal != value)
{ {
mySearchState.push_back(false); mySearchState.push_back(false);
@ -381,7 +385,7 @@ string RamWidget::doSearch(const string& str)
// Finally, show the search results in the list // Finally, show the search results in the list
showSearchResults(); showSearchResults();
return ""; return EmptyString;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -420,12 +424,11 @@ string RamWidget::doCompare(const string& str)
searchVal = instance().debugger().stringToValue(str); searchVal = instance().debugger().stringToValue(str);
// Now, search all memory locations previously 'found' for this value // Now, search all memory locations previously 'found' for this value
const ByteArray& ram = currentRam(0);
bool hitfound = false; bool hitfound = false;
CartDebug& dbg = instance().debugger().cartDebug();
const CartState& state = (CartState&) dbg.getState();
IntArray tempAddrList, tempValueList; IntArray tempAddrList, tempValueList;
mySearchState.clear(); mySearchState.clear();
for(uInt32 i = 0; i < state.rport.size(); ++i) for(uInt32 i = 0; i < ram.size(); ++i)
mySearchState.push_back(false); mySearchState.push_back(false);
for(uInt32 i = 0; i < mySearchAddr.size(); ++i) for(uInt32 i = 0; i < mySearchAddr.size(); ++i)
@ -438,7 +441,7 @@ string RamWidget::doCompare(const string& str)
} }
int addr = mySearchAddr[i]; int addr = mySearchAddr[i];
if(dbg.peek(state.rport[addr]) == searchVal) if(ram[addr] == searchVal)
{ {
tempAddrList.push_back(addr); tempAddrList.push_back(addr);
tempValueList.push_back(searchVal); tempValueList.push_back(searchVal);
@ -460,7 +463,7 @@ string RamWidget::doCompare(const string& str)
// Finally, show the search results in the list // Finally, show the search results in the list
showSearchResults(); showSearchResults();
return ""; return EmptyString;
} }
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
@ -482,15 +485,15 @@ void RamWidget::showSearchResults()
{ {
// Only update the search results for the bank currently being shown // Only update the search results for the bank currently being shown
BoolArray temp; BoolArray temp;
uInt32 start = myCurrentRamBank * 128; uInt32 start = myCurrentRamBank * myPageSize;
if(mySearchState.size() == 0 || start > mySearchState.size()) if(mySearchState.size() == 0 || start > mySearchState.size())
{ {
for(uInt32 i = 0; i < 128; ++i) for(uInt32 i = 0; i < myPageSize; ++i)
temp.push_back(false); temp.push_back(false);
} }
else else
{ {
for(uInt32 i = start; i < start + 128; ++i) for(uInt32 i = start; i < start + myPageSize; ++i)
temp.push_back(mySearchState[i]); temp.push_back(mySearchState[i]);
} }
myRamGrid->setHiliteList(temp); myRamGrid->setHiliteList(temp);

View File

@ -23,6 +23,7 @@
class GuiObject; class GuiObject;
class InputTextDialog; class InputTextDialog;
class ButtonWidget; class ButtonWidget;
class DataGridWidget;
class DataGridOpsWidget; class DataGridOpsWidget;
class EditTextWidget; class EditTextWidget;
class StaticTextWidget; class StaticTextWidget;
@ -34,11 +35,24 @@ class RamWidget : public Widget, public CommandSender
{ {
public: public:
RamWidget(GuiObject* boss, const GUI::Font& lfont, const GUI::Font& nfont, RamWidget(GuiObject* boss, const GUI::Font& lfont, const GUI::Font& nfont,
int x, int y); int x, int y, int w, int h,
uInt32 ramsize, uInt32 numrows, uInt32 pagesize);
virtual ~RamWidget(); virtual ~RamWidget();
void loadConfig() override; void loadConfig() override;
void setOpsWidget(DataGridOpsWidget* w); void setOpsWidget(DataGridOpsWidget* w);
void handleCommand(CommandSender* sender, int cmd, int data, int id) override;
private:
// To be implemented by derived classes
virtual uInt8 getValue(int addr) const = 0;
virtual void setValue(int addr, uInt8 value) = 0;
virtual string getLabel(int addr) const = 0;
virtual void fillList(uInt32 start, uInt32 size,
IntArray& alist, IntArray& vlist, BoolArray& changed) const = 0;
virtual uInt32 readPort(uInt32 start) const = 0;
virtual const ByteArray& currentRam(uInt32 start) const = 0;
private: private:
void fillGrid(bool updateOld); void fillGrid(bool updateOld);
@ -49,7 +63,13 @@ class RamWidget : public Widget, public CommandSender
void doRestart(); void doRestart();
void showSearchResults(); void showSearchResults();
void handleCommand(CommandSender* sender, int cmd, int data, int id) override; protected:
// Font used for 'normal' text; _font is for 'label' text
const GUI::Font& _nfont;
// These will be needed by most of the child classes;
// we may as well make them protected variables
int myFontWidth, myFontHeight, myLineHeight, myButtonHeight;
private: private:
enum { enum {
@ -59,21 +79,26 @@ class RamWidget : public Widget, public CommandSender
kCmpCmd = 'RWcp', kCmpCmd = 'RWcp',
kRestartCmd = 'RWrs', kRestartCmd = 'RWrs',
kSValEntered = 'RWsv', kSValEntered = 'RWsv',
kCValEntered = 'RWcv' kCValEntered = 'RWcv',
kRamHexID,
kRamDecID,
kRamBinID
}; };
int myUndoAddress; uInt32 myUndoAddress, myUndoValue;
int myUndoValue; uInt32 myCurrentRamBank;
int myCurrentRamBank; uInt32 myRamSize;
uInt32 myNumRows;
uInt32 myPageSize;
unique_ptr<InputTextDialog> myInputBox; unique_ptr<InputTextDialog> myInputBox;
StaticTextWidget* myRamStart; StaticTextWidget* myRamStart;
StaticTextWidget* myRamLabels[8]; StaticTextWidget* myRamLabels[16];
DataGridWidget* myRamGrid;
EditTextWidget* myBinValue; DataGridWidget* myRamGrid;
EditTextWidget* myDecValue; DataGridWidget* myDecValue;
DataGridWidget* myBinValue;
EditTextWidget* myLabel; EditTextWidget* myLabel;
ButtonWidget* myRevertButton; ButtonWidget* myRevertButton;
@ -82,7 +107,7 @@ class RamWidget : public Widget, public CommandSender
ButtonWidget* myCompareButton; ButtonWidget* myCompareButton;
ButtonWidget* myRestartButton; ButtonWidget* myRestartButton;
IntArray myOldValueList; ByteArray myOldValueList;
IntArray mySearchAddr; IntArray mySearchAddr;
IntArray mySearchValue; IntArray mySearchValue;
BoolArray mySearchState; BoolArray mySearchState;

View File

@ -0,0 +1,86 @@
//============================================================================
//
// 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-2015 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.
//
// $Id$
//============================================================================
#include "Debugger.hxx"
#include "CartDebug.hxx"
#include "RiotRamWidget.hxx"
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
RiotRamWidget::RiotRamWidget(GuiObject* boss, const GUI::Font& lfont,
const GUI::Font& nfont, int x, int y, int w)
: RamWidget(boss, lfont, nfont, x, y, w, 0, 128, 8, 128),
myDbg(instance().debugger().cartDebug())
{
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
RiotRamWidget::~RiotRamWidget()
{
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt8 RiotRamWidget::getValue(int addr) const
{
const CartState& state = (CartState&) myDbg.getState();
return myDbg.peek(state.rport[addr]);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void RiotRamWidget::setValue(int addr, uInt8 value)
{
const CartState& state = (CartState&) myDbg.getState();
myDbg.poke(state.wport[addr], value);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
string RiotRamWidget::getLabel(int addr) const
{
const CartState& state = (CartState&) myDbg.getState();
return myDbg.getLabel(state.rport[addr], true);
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
void RiotRamWidget::fillList(uInt32 start, uInt32 size, IntArray& alist,
IntArray& vlist, BoolArray& changed) const
{
const CartState& state = (CartState&) myDbg.getState();
const CartState& oldstate = (CartState&) myDbg.getOldState();
for(uInt32 i = 0; i < size; i++)
{
alist.push_back(i+start);
vlist.push_back(state.ram[i]);
changed.push_back(state.ram[i] != oldstate.ram[i]);
}
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
uInt32 RiotRamWidget::readPort(uInt32 start) const
{
const CartState& state = (CartState&) myDbg.getState();
return state.rport[start];
}
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
const ByteArray& RiotRamWidget::currentRam(uInt32) const
{
const CartState& state = (CartState&) myDbg.getState();
return state.ram;
}

View File

@ -0,0 +1,63 @@
//============================================================================
//
// 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-2015 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.
//
// $Id$
//============================================================================
#ifndef RIOT_RAM_WIDGET_HXX
#define RIOT_RAM_WIDGET_HXX
class GuiObject;
class InputTextDialog;
class ButtonWidget;
class DataGridWidget;
class DataGridOpsWidget;
class EditTextWidget;
class StaticTextWidget;
#include "CartDebug.hxx"
#include "RamWidget.hxx"
class RiotRamWidget : public RamWidget
{
public:
RiotRamWidget(GuiObject* boss, const GUI::Font& lfont, const GUI::Font& nfont,
int x, int y, int w);
virtual ~RiotRamWidget();
private:
uInt8 getValue(int addr) const;
void setValue(int addr, uInt8 value);
string getLabel(int addr) const;
void fillList(uInt32 start, uInt32 size, IntArray& alist,
IntArray& vlist, BoolArray& changed) const;
uInt32 readPort(uInt32 start) const;
const ByteArray& currentRam(uInt32 start) const;
private:
CartDebug& myDbg;
private:
// Following constructors and assignment operators not supported
RiotRamWidget() = delete;
RiotRamWidget(const RiotRamWidget&) = delete;
RiotRamWidget(RiotRamWidget&&) = delete;
RiotRamWidget& operator=(const RiotRamWidget&) = delete;
RiotRamWidget& operator=(RiotRamWidget&&) = delete;
};
#endif

View File

@ -6,6 +6,7 @@ MODULE_OBJS := \
src/debugger/gui/PromptWidget.o \ src/debugger/gui/PromptWidget.o \
src/debugger/gui/RamWidget.o \ src/debugger/gui/RamWidget.o \
src/debugger/gui/RiotWidget.o \ src/debugger/gui/RiotWidget.o \
src/debugger/gui/RiotRamWidget.o \
src/debugger/gui/RomWidget.o \ src/debugger/gui/RomWidget.o \
src/debugger/gui/RomListWidget.o \ src/debugger/gui/RomListWidget.o \
src/debugger/gui/RomListSettings.o \ src/debugger/gui/RomListSettings.o \

View File

@ -64,7 +64,7 @@ class Dialog : public GuiObject
virtual void setDefaults() { } virtual void setDefaults() { }
void addFocusWidget(Widget* w) override; void addFocusWidget(Widget* w) override;
void addToFocusList(WidgetArray& list); void addToFocusList(WidgetArray& list) override;
void addToFocusList(WidgetArray& list, TabWidget* w, int tabId); void addToFocusList(WidgetArray& list, TabWidget* w, int tabId);
void addBGroupToFocusList(WidgetArray& list) { _buttonGroup = list; } void addBGroupToFocusList(WidgetArray& list) { _buttonGroup = list; }
void addTabWidget(TabWidget* w); void addTabWidget(TabWidget* w);

View File

@ -29,6 +29,7 @@ class Widget;
#include "Command.hxx" #include "Command.hxx"
#include "OSystem.hxx" #include "OSystem.hxx"
#include "Vec.hxx"
using WidgetArray = vector<Widget*>; using WidgetArray = vector<Widget*>;
@ -87,8 +88,9 @@ class GuiObject : public CommandReceiver
virtual bool isVisible() const = 0; virtual bool isVisible() const = 0;
/** Add given widget to the focus list */ /** Add given widget(s) to the focus list */
virtual void addFocusWidget(Widget* w) = 0; virtual void addFocusWidget(Widget* w) = 0;
virtual void addToFocusList(WidgetArray& list) = 0;
/** Return focus list for this object */ /** Return focus list for this object */
WidgetArray& getFocusList() { return _focusList; } WidgetArray& getFocusList() { return _focusList; }

View File

@ -86,6 +86,9 @@ class Widget : public GuiObject
void receivedFocus(); void receivedFocus();
void lostFocus(); void lostFocus();
void addFocusWidget(Widget* w) override { _focusList.push_back(w); } void addFocusWidget(Widget* w) override { _focusList.push_back(w); }
void addToFocusList(WidgetArray& list) override {
Vec::append(_focusList, list);
};
/** Set/clear WIDGET_ENABLED flag */ /** Set/clear WIDGET_ENABLED flag */
void setEnabled(bool e); void setEnabled(bool e);