2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
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
|
|
|
**
|
2011-11-02 15:59:12 +01:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
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.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
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
|
2011-11-02 15:59:12 +01:00
|
|
|
** Nokia at qt-info@nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 15:08:31 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "registerhandler.h"
|
2010-09-22 17:30:22 +02:00
|
|
|
#include "watchdelegatewidgets.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2011-08-31 16:45:44 +02: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
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
2011-08-31 16:45:44 +02:00
|
|
|
// Register
|
2008-12-02 12:01:29 +01:00
|
|
|
//
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2011-08-31 16:45:44 +02:00
|
|
|
enum RegisterType
|
|
|
|
|
{
|
|
|
|
|
RegisterUnknown,
|
|
|
|
|
//RegisterDummy, // like AH if EAX is present.
|
|
|
|
|
RegisterI8,
|
|
|
|
|
RegisterI16,
|
|
|
|
|
RegisterI32,
|
|
|
|
|
RegisterI64,
|
2011-09-01 18:56:07 +02:00
|
|
|
RegisterI128,
|
2011-08-31 16:45:44 +02:00
|
|
|
RegisterF32,
|
|
|
|
|
RegisterF64,
|
|
|
|
|
RegisterF80,
|
|
|
|
|
RegisterXMM,
|
|
|
|
|
RegisterMMX,
|
|
|
|
|
RegisterNeon,
|
|
|
|
|
RegisterFlags32
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static struct RegisterNameAndType
|
|
|
|
|
{
|
|
|
|
|
const char *name;
|
|
|
|
|
RegisterType type;
|
|
|
|
|
} theNameAndType[] = {
|
2011-09-01 18:56:07 +02:00
|
|
|
// 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
|
2011-08-31 16:45:44 +02:00
|
|
|
{ "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;
|
2011-09-01 18:56:07 +02:00
|
|
|
case RegisterI128: return 4;
|
2011-08-31 16:45:44 +02:00
|
|
|
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)
|
|
|
|
|
{
|
2011-09-01 18:56:07 +02:00
|
|
|
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 };
|
2011-08-31 16:45:44 +02:00
|
|
|
|
|
|
|
|
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];
|
2011-09-01 18:56:07 +02:00
|
|
|
case RegisterI128: return integer[subType];
|
2011-08-31 16:45:44 +02:00
|
|
|
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
|
|
|
|
|
//
|
|
|
|
|
//////////////////////////////////////////////////////////////////
|
2010-11-05 13:35:31 +01:00
|
|
|
|
|
|
|
|
RegisterHandler::RegisterHandler()
|
2009-08-13 16:16:19 +02:00
|
|
|
{
|
2011-08-31 16:45:44 +02:00
|
|
|
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;
|
2009-08-13 16:16:19 +02:00
|
|
|
}
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2011-08-31 16:45:44 +02:00
|
|
|
QModelIndex RegisterHandler::index(int row, int col, const QModelIndex &parent) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2011-08-31 16:45:44 +02: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
|
|
|
}
|
|
|
|
|
|
2011-08-31 16:45:44 +02:00
|
|
|
QModelIndex RegisterHandler::parent(const QModelIndex &idx) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2011-08-31 16:45:44 +02: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
|
|
|
}
|
|
|
|
|
|
2010-09-22 17:30:22 +02:00
|
|
|
// Editor value: Preferably number, else string.
|
2010-11-05 13:35:31 +01:00
|
|
|
QVariant Register::editValue() const
|
2010-09-22 09:41:15 +02:00
|
|
|
{
|
|
|
|
|
bool ok = true;
|
|
|
|
|
// Try to convert to number?
|
2010-11-05 13:35:31 +01:00
|
|
|
const qulonglong v = value.toULongLong(&ok, 0); // Autodetect format
|
2010-09-22 09:41:15 +02:00
|
|
|
if (ok)
|
2010-11-05 13:35:31 +01:00
|
|
|
return QVariant(v);
|
|
|
|
|
return QVariant(value);
|
2010-09-22 17:30:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Editor value: Preferably padded number, else padded string.
|
2010-11-05 13:35:31 +01:00
|
|
|
QString Register::displayValue(int base, int strlen) const
|
2010-09-22 17:30:22 +02:00
|
|
|
{
|
2010-11-05 13:35:31 +01:00
|
|
|
const QVariant editV = editValue();
|
2010-09-22 17:30:22 +02:00
|
|
|
if (editV.type() == QVariant::ULongLong)
|
2010-11-05 13:35:31 +01:00
|
|
|
return QString::fromAscii("%1").arg(editV.toULongLong(), strlen, base);
|
2010-09-22 17:30:22 +02:00
|
|
|
const QString stringValue = editV.toString();
|
2010-11-05 13:35:31 +01:00
|
|
|
if (stringValue.size() < strlen)
|
2011-12-21 14:02:52 +01:00
|
|
|
return QString(strlen - stringValue.size(), QLatin1Char(' ')) + QLatin1String(value);
|
2010-09-22 17:30:22 +02:00
|
|
|
return stringValue;
|
2010-09-22 09:41:15 +02:00
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
QVariant RegisterHandler::data(const QModelIndex &index, int role) const
|
|
|
|
|
{
|
2011-08-31 16:45:44 +02:00
|
|
|
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();
|
|
|
|
|
|
|
|
|
|
|
2011-08-31 16:45:44 +02:00
|
|
|
const Register ® = 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) {
|
2011-12-21 14:02:52 +01:00
|
|
|
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]");
|
2011-08-31 16:45:44 +02:00
|
|
|
return QVariant(bitWidth);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
//
|
|
|
|
|
// Toplevel
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
switch (role) {
|
|
|
|
|
case Qt::DisplayRole:
|
|
|
|
|
switch (index.column()) {
|
|
|
|
|
case 0: {
|
|
|
|
|
const QString padding = QLatin1String(" ");
|
2011-12-21 14:02:52 +01:00
|
|
|
return QVariant(padding + QLatin1String(reg.name) + padding);
|
2011-08-31 16:45:44 +02:00
|
|
|
//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) {
|
2009-04-08 12:11:30 +02:00
|
|
|
switch (section) {
|
2011-08-31 16:45:44 +02:00
|
|
|
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();
|
|
|
|
|
}
|
|
|
|
|
|
2009-08-18 08:34:48 +02:00
|
|
|
Qt::ItemFlags RegisterHandler::flags(const QModelIndex &idx) const
|
|
|
|
|
{
|
|
|
|
|
if (!idx.isValid())
|
2010-09-22 17:30:22 +02:00
|
|
|
return Qt::ItemFlags();
|
2009-08-18 08:34:48 +02:00
|
|
|
|
2010-09-22 17:30:22 +02:00
|
|
|
const Qt::ItemFlags notEditable = Qt::ItemIsSelectable|Qt::ItemIsEnabled;
|
|
|
|
|
// Can edit registers if they are hex numbers and not arrays.
|
2010-11-05 13:35:31 +01:00
|
|
|
if (idx.column() == 1
|
2011-12-21 14:02:52 +01:00
|
|
|
&& IntegerWatchLineEdit::isUnsignedHexNumber(QLatin1String(m_registers.at(idx.row()).value)))
|
2010-09-22 17:30:22 +02:00
|
|
|
return notEditable | Qt::ItemIsEditable;
|
|
|
|
|
return notEditable;
|
2009-08-18 08:34:48 +02:00
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void RegisterHandler::removeAll()
|
|
|
|
|
{
|
|
|
|
|
m_registers.clear();
|
|
|
|
|
reset();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool RegisterHandler::isEmpty() const
|
|
|
|
|
{
|
|
|
|
|
return m_registers.isEmpty();
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 14:07:09 +02:00
|
|
|
// 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;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-10 11:37:57 +01:00
|
|
|
void RegisterHandler::setRegisters(const Registers ®isters)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
m_registers = registers;
|
2011-04-06 14:07:09 +02:00
|
|
|
const int size = m_registers.size();
|
|
|
|
|
for (int r = 0; r < size; r++)
|
|
|
|
|
m_registers[r].changed = false;
|
2010-09-22 09:41:15 +02:00
|
|
|
calculateWidth();
|
2008-12-02 12:01:29 +01:00
|
|
|
reset();
|
|
|
|
|
}
|
|
|
|
|
|
2010-09-01 13:56:51 +02:00
|
|
|
void RegisterHandler::setAndMarkRegisters(const Registers ®isters)
|
|
|
|
|
{
|
2011-04-06 14:07:09 +02:00
|
|
|
if (!compareRegisterSet(m_registers, registers)) {
|
|
|
|
|
setRegisters(registers);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const int size = m_registers.size();
|
2011-08-31 16:45:44 +02:00
|
|
|
for (int r = 0; r != size; ++r) {
|
|
|
|
|
const QModelIndex regIndex = index(r, 1, QModelIndex());
|
2011-04-06 14:07:09 +02:00
|
|
|
if (m_registers.at(r).value != registers.at(r).value) {
|
|
|
|
|
// Indicate red if values change, keep changed.
|
2011-08-31 16:45:44 +02:00
|
|
|
m_registers[r].changed = m_registers.at(r).changed
|
|
|
|
|
|| !m_registers.at(r).value.isEmpty();
|
2011-04-06 14:07:09 +02:00
|
|
|
m_registers[r].value = registers.at(r).value;
|
|
|
|
|
emit dataChanged(regIndex, regIndex);
|
|
|
|
|
}
|
2011-08-31 16:45:44 +02:00
|
|
|
emit registerSet(regIndex); // Notify attached memory views.
|
2011-04-06 14:07:09 +02:00
|
|
|
}
|
2010-09-01 13:56:51 +02:00
|
|
|
}
|
|
|
|
|
|
2010-02-10 11:37:57 +01:00
|
|
|
Registers RegisterHandler::registers() const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
return m_registers;
|
|
|
|
|
}
|
2009-08-13 16:16:19 +02:00
|
|
|
|
2010-09-22 09:41:15 +02:00
|
|
|
void RegisterHandler::calculateWidth()
|
|
|
|
|
{
|
|
|
|
|
m_strlen = (m_base == 2 ? 64 : m_base == 8 ? 32 : m_base == 10 ? 26 : 16);
|
|
|
|
|
}
|
|
|
|
|
|
2009-08-13 16:16:19 +02:00
|
|
|
void RegisterHandler::setNumberBase(int base)
|
|
|
|
|
{
|
2010-09-22 09:41:15 +02:00
|
|
|
if (m_base != base) {
|
|
|
|
|
m_base = base;
|
|
|
|
|
calculateWidth();
|
|
|
|
|
emit reset();
|
|
|
|
|
}
|
2009-08-13 16:16:19 +02:00
|
|
|
}
|
2010-11-05 13:35:31 +01:00
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|