Files
qt-creator/src/plugins/debugger/registerhandler.cpp

557 lines
15 KiB
C++
Raw Normal View History

/**************************************************************************
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator
**
2011-01-11 16:28:15 +01:00
** Copyright (c) 2011 Nokia Corporation and/or its subsidiary(-ies).
2008-12-02 12:01:29 +01:00
**
** Contact: Nokia Corporation (qt-info@nokia.com)
2008-12-02 12:01:29 +01:00
**
**
** GNU Lesser General Public License Usage
**
2011-04-13 08:42:33 +02:00
** 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.
**
2010-12-17 16:01:08 +01:00
** In addition, as a special exception, Nokia gives you certain additional
2011-04-13 08:42:33 +02:00
** rights. These rights are described in the Nokia Qt LGPL Exception
2010-12-17 16:01:08 +01:00
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
2011-04-13 08:42:33 +02:00
** Other Usage
**
** Alternatively, this file may be used in accordance with the terms and
** conditions contained in a signed written agreement between you and Nokia.
**
2010-12-17 16:01:08 +01:00
** If you have questions regarding the use of this file, please contact
** Nokia at qt-info@nokia.com.
2008-12-02 12:01:29 +01:00
**
**************************************************************************/
2008-12-02 15:08:31 +01:00
2008-12-02 12:01:29 +01:00
#include "registerhandler.h"
#include "watchdelegatewidgets.h"
2008-12-02 12:01:29 +01:00
#if USE_REGISTER_MODEL_TEST
#include "modeltest.h"
#endif
#include <utils/qtcassert.h>
namespace Debugger {
namespace Internal {
2008-12-02 12:01:29 +01:00
//////////////////////////////////////////////////////////////////
//
// Register
2008-12-02 12:01:29 +01:00
//
//////////////////////////////////////////////////////////////////
enum RegisterType
{
RegisterUnknown,
//RegisterDummy, // like AH if EAX is present.
RegisterI8,
RegisterI16,
RegisterI32,
RegisterI64,
RegisterI128,
RegisterF32,
RegisterF64,
RegisterF80,
RegisterXMM,
RegisterMMX,
RegisterNeon,
RegisterFlags32
};
static struct RegisterNameAndType
{
const char *name;
RegisterType type;
} theNameAndType[] = {
// ARM
{ "r0", RegisterI32 },
{ "r1", RegisterI32 },
{ "r2", RegisterI32 },
{ "r3", RegisterI32 },
{ "r4", RegisterI32 },
{ "r5", RegisterI32 },
{ "r6", RegisterI32 },
{ "r7", RegisterI32 },
{ "r8", RegisterI32 },
{ "r9", RegisterI32 },
{ "r10", RegisterI32 },
{ "r11", RegisterI32 },
{ "r12", RegisterI32 },
{ "sp", RegisterI32 },
{ "lr", RegisterI32 },
{ "pc", RegisterI32 },
{ "cpsr", RegisterFlags32 },
{ "d0", RegisterI64 },
{ "d1", RegisterI64 },
{ "d2", RegisterI64 },
{ "d3", RegisterI64 },
{ "d4", RegisterI64 },
{ "d5", RegisterI64 },
{ "d6", RegisterI64 },
{ "d7", RegisterI64 },
{ "d8", RegisterI64 },
{ "d9", RegisterI64 },
{ "d10", RegisterI64 },
{ "d11", RegisterI64 },
{ "d12", RegisterI64 },
{ "d13", RegisterI64 },
{ "d14", RegisterI64 },
{ "d15", RegisterI64 },
{ "d16", RegisterI64 },
{ "d17", RegisterI64 },
{ "d18", RegisterI64 },
{ "d19", RegisterI64 },
{ "d20", RegisterI64 },
{ "d21", RegisterI64 },
{ "d22", RegisterI64 },
{ "d23", RegisterI64 },
{ "d24", RegisterI64 },
{ "d25", RegisterI64 },
{ "d26", RegisterI64 },
{ "d27", RegisterI64 },
{ "d28", RegisterI64 },
{ "d29", RegisterI64 },
{ "d30", RegisterI64 },
{ "d31", RegisterI64 },
{ "fpscr", RegisterFlags32 },
{ "s0", RegisterI32 },
{ "s1", RegisterI32 },
{ "s2", RegisterI32 },
{ "s3", RegisterI32 },
{ "s4", RegisterI32 },
{ "s5", RegisterI32 },
{ "s6", RegisterI32 },
{ "s7", RegisterI32 },
{ "s8", RegisterI32 },
{ "s9", RegisterI32 },
{ "s10", RegisterI32 },
{ "s11", RegisterI32 },
{ "s12", RegisterI32 },
{ "s13", RegisterI32 },
{ "s14", RegisterI32 },
{ "s15", RegisterI32 },
{ "s16", RegisterI32 },
{ "s17", RegisterI32 },
{ "s18", RegisterI32 },
{ "s19", RegisterI32 },
{ "s20", RegisterI32 },
{ "s21", RegisterI32 },
{ "s22", RegisterI32 },
{ "s23", RegisterI32 },
{ "s24", RegisterI32 },
{ "s25", RegisterI32 },
{ "s26", RegisterI32 },
{ "s27", RegisterI32 },
{ "s28", RegisterI32 },
{ "s29", RegisterI32 },
{ "s30", RegisterI32 },
{ "s31", RegisterI32 },
{ "q0", RegisterI128 },
{ "q1", RegisterI128 },
{ "q2", RegisterI128 },
{ "q3", RegisterI128 },
{ "q4", RegisterI128 },
{ "q5", RegisterI128 },
{ "q6", RegisterI128 },
{ "q7", RegisterI128 },
{ "q8", RegisterI128 },
{ "q9", RegisterI128 },
{ "q10", RegisterI128 },
{ "q11", RegisterI128 },
{ "q12", RegisterI128 },
{ "q13", RegisterI128 },
{ "q14", RegisterI128 },
{ "q15", RegisterI128 },
// Intel
{ "eax", RegisterI32 },
{ "ecx", RegisterI32 },
{ "edx", RegisterI32 },
{ "ebx", RegisterI32 },
{ "esp", RegisterI32 },
{ "ebp", RegisterI32 },
{ "esi", RegisterI32 },
{ "edi", RegisterI32 },
{ "eip", RegisterI32 },
{ "eflags", RegisterFlags32 },
{ "cs", RegisterI32 },
{ "ss", RegisterI32 },
{ "ds", RegisterI32 },
{ "es", RegisterI32 },
{ "fs", RegisterI32 },
{ "gs", RegisterI32 },
{ "st0", RegisterF80 },
{ "st1", RegisterF80 },
{ "st2", RegisterF80 },
{ "st3", RegisterF80 },
{ "st4", RegisterF80 },
{ "st5", RegisterF80 },
{ "st6", RegisterF80 },
{ "st7", RegisterF80 },
{ "fctrl", RegisterFlags32 },
{ "fstat", RegisterFlags32 },
{ "ftag", RegisterFlags32 },
{ "fiseg", RegisterFlags32 },
{ "fioff", RegisterFlags32 },
{ "foseg", RegisterFlags32 },
{ "fooff", RegisterFlags32 },
{ "fop", RegisterFlags32 },
{ "xmm0", RegisterXMM },
{ "xmm1", RegisterXMM },
{ "xmm2", RegisterXMM },
{ "xmm3", RegisterXMM },
{ "xmm4", RegisterXMM },
{ "xmm5", RegisterXMM },
{ "xmm6", RegisterXMM },
{ "xmm7", RegisterXMM },
{ "mxcsr", RegisterFlags32 },
{ "orig_eax", RegisterI32 },
{ "al", RegisterI8 },
{ "cl", RegisterI8 },
{ "dl", RegisterI8 },
{ "bl", RegisterI8 },
{ "ah", RegisterI8 },
{ "ch", RegisterI8 },
{ "dh", RegisterI8 },
{ "bh", RegisterI8 },
{ "ax", RegisterI16 },
{ "cx", RegisterI16 },
{ "dx", RegisterI16 },
{ "bx", RegisterI16 },
{ "bp", RegisterI16 },
{ "si", RegisterI16 },
{ "di", RegisterI16 },
{ "mm0", RegisterMMX },
{ "mm1", RegisterMMX },
{ "mm2", RegisterMMX },
{ "mm3", RegisterMMX },
{ "mm4", RegisterMMX },
{ "mm5", RegisterMMX },
{ "mm6", RegisterMMX },
{ "mm7", RegisterMMX }
};
static RegisterType guessType(const QByteArray &name)
{
static QHash<QByteArray, RegisterType> theTypes;
if (theTypes.isEmpty()) {
for (int i = 0; i != sizeof(theNameAndType) / sizeof(theNameAndType[0]); ++i)
theTypes[theNameAndType[i].name] = theNameAndType[i].type;
}
return theTypes.value(name, RegisterUnknown);
}
static int childCountFromType(int type)
{
switch (type) {
case RegisterUnknown: return 0;
case RegisterI8: return 0;
case RegisterI16: return 1;
case RegisterI32: return 2;
case RegisterI64: return 3;
case RegisterI128: return 4;
case RegisterF32: return 0;
case RegisterF64: return 0;
case RegisterF80: return 0;
case RegisterXMM: return 3;
case RegisterMMX: return 3;
case RegisterNeon: return 3;
case RegisterFlags32: return 0;
}
QTC_ASSERT(false, /**/);
return 0;
}
static int bitWidthFromType(int type, int subType)
{
const uint integer[] = { 8, 16, 32, 64, 128 };
const uint xmm[] = { 8, 16, 32, 64, 128 };
const uint mmx[] = { 8, 16, 32, 64, 128 };
const uint neon[] = { 8, 16, 32, 64, 128 };
switch (type) {
case RegisterUnknown: return 0;
case RegisterI8: return 8;
case RegisterI16: return integer[subType];
case RegisterI32: return integer[subType];
case RegisterI64: return integer[subType];
case RegisterI128: return integer[subType];
case RegisterF32: return 0;
case RegisterF64: return 0;
case RegisterF80: return 0;
case RegisterXMM: return xmm[subType];
case RegisterMMX: return mmx[subType];
case RegisterNeon: return neon[subType];
case RegisterFlags32: return 0;
}
QTC_ASSERT(false, /**/);
return 0;
}
Register::Register(const QByteArray &name_)
: name(name_), changed(true)
{
type = guessType(name);
}
//////////////////////////////////////////////////////////////////
//
// RegisterHandler
//
//////////////////////////////////////////////////////////////////
RegisterHandler::RegisterHandler()
{
m_base = 16;
calculateWidth();
#if USE_REGISTER_MODEL_TEST
new ModelTest(this, 0);
#endif
}
int RegisterHandler::rowCount(const QModelIndex &idx) const
{
if (idx.column() > 0)
return 0;
if (!idx.isValid())
return m_registers.size(); // Top level.
if (idx.internalId() >= 0)
return 0; // Sub-Items don't have children.
if (idx.row() >= m_registers.size())
return 0;
return childCountFromType(m_registers[idx.row()].type);
}
int RegisterHandler::columnCount(const QModelIndex &idx) const
{
if (idx.column() > 0)
return 0;
if (!idx.isValid())
return 2;
if (idx.internalId() >= 0)
return 0; // Sub-Items don't have children.
return 2;
}
2008-12-02 12:01:29 +01:00
QModelIndex RegisterHandler::index(int row, int col, const QModelIndex &parent) const
2008-12-02 12:01:29 +01:00
{
if (row < 0 || col < 0 || col >= 2)
return QModelIndex();
if (!parent.isValid()) // Top level.
return createIndex(row, col, -1);
if (parent.internalId() >= 0) // Sub-Item has no children.
return QModelIndex();
if (parent.column() > 0)
return QModelIndex();
return createIndex(row, col, parent.row());
2008-12-02 12:01:29 +01:00
}
QModelIndex RegisterHandler::parent(const QModelIndex &idx) const
2008-12-02 12:01:29 +01:00
{
if (!idx.isValid())
return QModelIndex();
if (idx.internalId() >= 0)
return createIndex(idx.internalId(), 0, -1);
return QModelIndex();
2008-12-02 12:01:29 +01:00
}
// Editor value: Preferably number, else string.
QVariant Register::editValue() const
{
bool ok = true;
// Try to convert to number?
const qulonglong v = value.toULongLong(&ok, 0); // Autodetect format
if (ok)
return QVariant(v);
return QVariant(value);
}
// Editor value: Preferably padded number, else padded string.
QString Register::displayValue(int base, int strlen) const
{
const QVariant editV = editValue();
if (editV.type() == QVariant::ULongLong)
return QString::fromAscii("%1").arg(editV.toULongLong(), strlen, base);
const QString stringValue = editV.toString();
if (stringValue.size() < strlen)
return QString(strlen - stringValue.size(), QLatin1Char(' ')) + QLatin1String(value);
return stringValue;
}
2008-12-02 12:01:29 +01:00
QVariant RegisterHandler::data(const QModelIndex &index, int role) const
{
if (!index.isValid())
return QVariant();
QModelIndex topLevel = index.parent();
const int mainRow = topLevel.isValid() ? topLevel.row() : index.row();
if (mainRow >= m_registers.size())
2008-12-02 12:01:29 +01:00
return QVariant();
const Register &reg = m_registers.at(mainRow);
if (topLevel.isValid()) {
//
// Nested
//
int subType = index.row();
int bitWidth = bitWidthFromType(reg.type, subType);
switch (role) {
case Qt::DisplayRole:
switch (index.column()) {
case 0: {
switch (bitWidth) {
case 8: return QLatin1String("[Bytes]");
case 16: return QLatin1String("[Words]");
case 32: return QLatin1String("[DWords]");
case 64: return QLatin1String("[QWords]");
case 128: return QLatin1String("[TWords]");
case -32: return QLatin1String("[Single]");
case -64: return QLatin1String("[Double]");
return QVariant(bitWidth);
}
}
}
default:
break;
}
} else {
//
// Toplevel
//
switch (role) {
case Qt::DisplayRole:
switch (index.column()) {
case 0: {
const QString padding = QLatin1String(" ");
return QVariant(padding + QLatin1String(reg.name) + padding);
//return QVariant(reg.name);
}
case 1: // Display: Pad value for alignment
return reg.displayValue(m_base, m_strlen);
} // switch column
case Qt::EditRole: // Edit: Unpadded for editing
return reg.editValue();
case Qt::TextAlignmentRole:
return index.column() == 1 ? QVariant(Qt::AlignRight) : QVariant();
default:
break;
2008-12-02 12:01:29 +01:00
}
}
return QVariant();
}
QVariant RegisterHandler::headerData(int section, Qt::Orientation orientation,
int role) const
{
if (orientation == Qt::Horizontal && role == Qt::DisplayRole) {
switch (section) {
case 0: return tr("Name");
case 1: return tr("Value (Base %1)").arg(m_base);
2008-12-02 12:01:29 +01:00
};
}
return QVariant();
}
Qt::ItemFlags RegisterHandler::flags(const QModelIndex &idx) const
{
if (!idx.isValid())
return Qt::ItemFlags();
const Qt::ItemFlags notEditable = Qt::ItemIsSelectable|Qt::ItemIsEnabled;
// Can edit registers if they are hex numbers and not arrays.
if (idx.column() == 1
&& IntegerWatchLineEdit::isUnsignedHexNumber(QLatin1String(m_registers.at(idx.row()).value)))
return notEditable | Qt::ItemIsEditable;
return notEditable;
}
2008-12-02 12:01:29 +01:00
void RegisterHandler::removeAll()
{
m_registers.clear();
reset();
}
bool RegisterHandler::isEmpty() const
{
return m_registers.isEmpty();
}
// Compare register sets by name
static inline bool compareRegisterSet(const Registers &r1, const Registers &r2)
{
if (r1.size() != r2.size())
return false;
const int size = r1.size();
for (int r = 0; r < size; r++)
if (r1.at(r).name != r2.at(r).name)
return false;
return true;
}
void RegisterHandler::setRegisters(const Registers &registers)
2008-12-02 12:01:29 +01:00
{
m_registers = registers;
const int size = m_registers.size();
for (int r = 0; r < size; r++)
m_registers[r].changed = false;
calculateWidth();
2008-12-02 12:01:29 +01:00
reset();
}
void RegisterHandler::setAndMarkRegisters(const Registers &registers)
{
if (!compareRegisterSet(m_registers, registers)) {
setRegisters(registers);
return;
}
const int size = m_registers.size();
for (int r = 0; r != size; ++r) {
const QModelIndex regIndex = index(r, 1, QModelIndex());
if (m_registers.at(r).value != registers.at(r).value) {
// Indicate red if values change, keep changed.
m_registers[r].changed = m_registers.at(r).changed
|| !m_registers.at(r).value.isEmpty();
m_registers[r].value = registers.at(r).value;
emit dataChanged(regIndex, regIndex);
}
emit registerSet(regIndex); // Notify attached memory views.
}
}
Registers RegisterHandler::registers() const
2008-12-02 12:01:29 +01:00
{
return m_registers;
}
void RegisterHandler::calculateWidth()
{
m_strlen = (m_base == 2 ? 64 : m_base == 8 ? 32 : m_base == 10 ? 26 : 16);
}
void RegisterHandler::setNumberBase(int base)
{
if (m_base != base) {
m_base = base;
calculateWidth();
emit reset();
}
}
} // namespace Internal
} // namespace Debugger