forked from qt-creator/qt-creator
Change-Id: Ibb186cabba8fecedfdbd04da000459cf358e56e4 Reviewed-by: Eike Ziller <eike.ziller@digia.com>
278 lines
9.8 KiB
C++
278 lines
9.8 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
|
** Contact: http://www.qt-project.org/legal
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and Digia. For licensing terms and
|
|
** conditions see http://qt.digia.com/licensing. For further information
|
|
** use the contact form at http://qt.digia.com/contact-us.
|
|
**
|
|
** GNU Lesser General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
** General Public License version 2.1 as published by the Free Software
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
** packaging of this file. Please review the following information to
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
|
**
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "registerwindow.h"
|
|
#include "memoryview.h"
|
|
#include "debuggeractions.h"
|
|
#include "debuggerdialogs.h"
|
|
#include "debuggercore.h"
|
|
#include "debuggerengine.h"
|
|
#include "registerhandler.h"
|
|
#include "watchdelegatewidgets.h"
|
|
#include "memoryagent.h"
|
|
|
|
#include <utils/savedaction.h>
|
|
#include <utils/qtcassert.h>
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QItemDelegate>
|
|
#include <QMenu>
|
|
#include <QPainter>
|
|
|
|
|
|
namespace Debugger {
|
|
namespace Internal {
|
|
|
|
static DebuggerEngine *currentEngine()
|
|
{
|
|
return debuggerCore()->currentEngine();
|
|
}
|
|
|
|
static RegisterHandler *currentHandler()
|
|
{
|
|
DebuggerEngine *engine = currentEngine();
|
|
QTC_ASSERT(engine, return 0);
|
|
return engine->registerHandler();
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
//
|
|
// RegisterDelegate
|
|
//
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
class RegisterDelegate : public QItemDelegate
|
|
{
|
|
public:
|
|
RegisterDelegate(QObject *parent)
|
|
: QItemDelegate(parent)
|
|
{}
|
|
|
|
QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem &,
|
|
const QModelIndex &index) const
|
|
{
|
|
Register reg = currentHandler()->registerAt(index.row());
|
|
IntegerWatchLineEdit *lineEdit = new IntegerWatchLineEdit(parent);
|
|
const int base = currentHandler()->numberBase();
|
|
const bool big = reg.value.size() > 16;
|
|
// Big integers are assumed to be hexadecimal.
|
|
lineEdit->setBigInt(big);
|
|
lineEdit->setBase(big ? 16 : base);
|
|
lineEdit->setSigned(false);
|
|
lineEdit->setAlignment(Qt::AlignRight);
|
|
lineEdit->setFrame(false);
|
|
return lineEdit;
|
|
}
|
|
|
|
void setEditorData(QWidget *editor, const QModelIndex &index) const
|
|
{
|
|
IntegerWatchLineEdit *lineEdit = qobject_cast<IntegerWatchLineEdit *>(editor);
|
|
QTC_ASSERT(lineEdit, return);
|
|
lineEdit->setModelData(index.data(Qt::EditRole));
|
|
}
|
|
|
|
void setModelData(QWidget *editor, QAbstractItemModel *,
|
|
const QModelIndex &index) const
|
|
{
|
|
if (index.column() != 1)
|
|
return;
|
|
IntegerWatchLineEdit *lineEdit = qobject_cast<IntegerWatchLineEdit*>(editor);
|
|
QTC_ASSERT(lineEdit, return);
|
|
const int base = currentHandler()->numberBase();
|
|
QString value = lineEdit->text();
|
|
if (base == 16 && !value.startsWith(QLatin1String("0x")))
|
|
value.insert(0, QLatin1String("0x"));
|
|
currentEngine()->setRegisterValue(index.row(), value);
|
|
}
|
|
|
|
void updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option,
|
|
const QModelIndex &) const
|
|
{
|
|
editor->setGeometry(option.rect);
|
|
}
|
|
|
|
void paint(QPainter *painter, const QStyleOptionViewItem &option,
|
|
const QModelIndex &index) const
|
|
{
|
|
if (index.column() == 1) {
|
|
bool paintRed = currentHandler()->registerAt(index.row()).changed;
|
|
QPen oldPen = painter->pen();
|
|
if (paintRed)
|
|
painter->setPen(QColor(200, 0, 0));
|
|
// FIXME: performance? this changes only on real font changes.
|
|
QFontMetrics fm(option.font);
|
|
int charWidth = fm.width(QLatin1Char('x'));
|
|
for (int i = '1'; i <= '9'; ++i)
|
|
charWidth = qMax(charWidth, fm.width(QLatin1Char(i)));
|
|
for (int i = 'a'; i <= 'f'; ++i)
|
|
charWidth = qMax(charWidth, fm.width(QLatin1Char(i)));
|
|
QString str = index.data(Qt::DisplayRole).toString();
|
|
int x = option.rect.x();
|
|
for (int i = 0; i < str.size(); ++i) {
|
|
QRect r = option.rect;
|
|
r.setX(x);
|
|
r.setWidth(charWidth);
|
|
x += charWidth;
|
|
painter->drawText(r, Qt::AlignHCenter, QString(str.at(i)));
|
|
}
|
|
if (paintRed)
|
|
painter->setPen(oldPen);
|
|
} else {
|
|
QItemDelegate::paint(painter, option, index);
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
//
|
|
// RegisterWindow
|
|
//
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
RegisterTreeView::RegisterTreeView(QWidget *parent)
|
|
: BaseTreeView(parent)
|
|
{
|
|
setItemDelegate(new RegisterDelegate(this));
|
|
}
|
|
|
|
void RegisterTreeView::contextMenuEvent(QContextMenuEvent *ev)
|
|
{
|
|
QMenu menu;
|
|
|
|
DebuggerEngine *engine = currentEngine();
|
|
QTC_ASSERT(engine, return);
|
|
RegisterHandler *handler = currentHandler();
|
|
const bool actionsEnabled = engine->debuggerActionsEnabled();
|
|
const DebuggerState state = engine->state();
|
|
|
|
QAction *actReload = menu.addAction(tr("Reload Register Listing"));
|
|
actReload->setEnabled(engine->hasCapability(RegisterCapability)
|
|
&& (state == InferiorStopOk || state == InferiorUnrunnable));
|
|
|
|
menu.addSeparator();
|
|
|
|
Register aRegister;
|
|
const QModelIndex idx = indexAt(ev->pos());
|
|
if (idx.isValid())
|
|
aRegister = handler->registers().at(idx.row());
|
|
const QVariant addressV = aRegister.editValue();
|
|
const quint64 address = addressV.type() == QVariant::ULongLong
|
|
? addressV.toULongLong() : 0;
|
|
QAction *actViewMemory = menu.addAction(QString());
|
|
QAction *actEditMemory = menu.addAction(QString());
|
|
|
|
QAction *actShowDisassemblerAt = menu.addAction(QString());
|
|
QAction *actShowDisassembler = menu.addAction(tr("Open Disassembler..."));
|
|
actShowDisassembler->setEnabled(engine->hasCapability(DisassemblerCapability));
|
|
|
|
if (address) {
|
|
const bool canShow = actionsEnabled && engine->hasCapability(ShowMemoryCapability);
|
|
actEditMemory->setText(tr("Open Memory Editor at 0x%1").arg(address, 0, 16));
|
|
actEditMemory->setEnabled(canShow);
|
|
actViewMemory->setText(tr("Open Memory View at Value of Register %1 0x%2")
|
|
.arg(QString::fromLatin1(aRegister.name)).arg(address, 0, 16));
|
|
actShowDisassemblerAt->setText(tr("Open Disassembler at 0x%1")
|
|
.arg(address, 0, 16));
|
|
actShowDisassemblerAt->setEnabled(engine->hasCapability(DisassemblerCapability));
|
|
} else {
|
|
actEditMemory->setText(tr("Open Memory Editor"));
|
|
actViewMemory->setText(tr("Open Memory View at Value of Register"));
|
|
actEditMemory->setEnabled(false);
|
|
actViewMemory->setEnabled(false);
|
|
actShowDisassemblerAt->setText(tr("Open Disassembler"));
|
|
actShowDisassemblerAt->setEnabled(false);
|
|
}
|
|
menu.addSeparator();
|
|
|
|
const int base = handler->numberBase();
|
|
QAction *act16 = menu.addAction(tr("Hexadecimal"));
|
|
act16->setCheckable(true);
|
|
act16->setChecked(base == 16);
|
|
QAction *act10 = menu.addAction(tr("Decimal"));
|
|
act10->setCheckable(true);
|
|
act10->setChecked(base == 10);
|
|
QAction *act8 = menu.addAction(tr("Octal"));
|
|
act8->setCheckable(true);
|
|
act8->setChecked(base == 8);
|
|
QAction *act2 = menu.addAction(tr("Binary"));
|
|
act2->setCheckable(true);
|
|
act2->setChecked(base == 2);
|
|
|
|
menu.addSeparator();
|
|
menu.addAction(debuggerCore()->action(SettingsDialog));
|
|
|
|
const QPoint position = ev->globalPos();
|
|
QAction *act = menu.exec(position);
|
|
|
|
if (act == actReload) {
|
|
engine->reloadRegisters();
|
|
} else if (act == actEditMemory) {
|
|
const QString registerName = QString::fromLatin1(aRegister.name);
|
|
engine->openMemoryView(address, 0,
|
|
RegisterMemoryView::registerMarkup(address, registerName),
|
|
QPoint(), RegisterMemoryView::title(registerName), 0);
|
|
} else if (act == actViewMemory) {
|
|
engine->openMemoryView(idx.row(),
|
|
DebuggerEngine::MemoryTrackRegister|DebuggerEngine::MemoryView,
|
|
QList<MemoryMarkup>(), position, QString(), this);
|
|
} else if (act == actShowDisassembler) {
|
|
AddressDialog dialog;
|
|
if (address)
|
|
dialog.setAddress(address);
|
|
if (dialog.exec() == QDialog::Accepted)
|
|
currentEngine()->openDisassemblerView(Location(dialog.address()));
|
|
} else if (act == actShowDisassemblerAt) {
|
|
engine->openDisassemblerView(Location(address));
|
|
} else if (act == act16)
|
|
handler->setNumberBase(16);
|
|
else if (act == act10)
|
|
handler->setNumberBase(10);
|
|
else if (act == act8)
|
|
handler->setNumberBase(8);
|
|
else if (act == act2)
|
|
handler->setNumberBase(2);
|
|
}
|
|
|
|
void RegisterTreeView::reloadRegisters()
|
|
{
|
|
// FIXME: Only trigger when becoming visible?
|
|
currentEngine()->reloadRegisters();
|
|
}
|
|
|
|
RegisterWindow::RegisterWindow()
|
|
: BaseWindow(new RegisterTreeView)
|
|
{
|
|
setWindowTitle(tr("Registers"));
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace Debugger
|