2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2014-01-07 13:27:11 +01:00
|
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
2012-10-02 09:12:39 +02:00
|
|
|
** Contact: http://www.qt-project.org/legal
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** 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
|
2014-10-01 13:21:18 +02:00
|
|
|
** conditions see http://www.qt.io/licensing. For further information
|
|
|
|
|
** use the contact form at http://www.qt.io/contact-us.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2012-10-02 09:12:39 +02:00
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
2014-10-01 13:21:18 +02:00
|
|
|
** General Public License version 2.1 or version 3 as published by the Free
|
|
|
|
|
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
|
|
|
|
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
|
|
|
|
** following information to ensure the GNU Lesser General Public License
|
|
|
|
|
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
|
|
|
|
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2012-10-02 09:12:39 +02:00
|
|
|
**
|
|
|
|
|
** In addition, as a special exception, Digia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Digia Qt LGPL Exception
|
2010-12-17 16:01:08 +01:00
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
2012-10-02 09:12:39 +02: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
|
2013-09-02 14:04:12 +02:00
|
|
|
#include <modeltest.h>
|
2011-08-31 16:45:44 +02:00
|
|
|
#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;
|
|
|
|
|
}
|
|
|
|
|
|
2013-12-05 23:29:37 -08:00
|
|
|
static const uint TopLevelId = UINT_MAX;
|
2012-09-24 15:48:17 +02:00
|
|
|
static bool isTopLevelItem(const QModelIndex &index)
|
|
|
|
|
{
|
|
|
|
|
return quintptr(index.internalId()) == quintptr(TopLevelId);
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-31 16:45:44 +02:00
|
|
|
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
|
|
|
{
|
2013-08-19 13:30:10 +02:00
|
|
|
setObjectName(QLatin1String("RegisterModel"));
|
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.
|
2012-09-24 15:48:17 +02:00
|
|
|
if (!isTopLevelItem(idx))
|
2011-08-31 16:45:44 +02:00
|
|
|
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;
|
2012-09-24 15:48:17 +02:00
|
|
|
if (!isTopLevelItem(idx))
|
2011-08-31 16:45:44 +02:00
|
|
|
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.
|
2012-09-24 15:48:17 +02:00
|
|
|
return createIndex(row, col, TopLevelId);
|
|
|
|
|
if (!isTopLevelItem(parent)) // Sub-Item has no children.
|
2011-08-31 16:45:44 +02:00
|
|
|
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();
|
2012-09-24 15:48:17 +02:00
|
|
|
if (!isTopLevelItem(idx))
|
|
|
|
|
return createIndex(idx.internalId(), 0, TopLevelId);
|
2011-08-31 16:45:44 +02:00
|
|
|
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)
|
2012-02-16 09:55:47 +02:00
|
|
|
return QString::fromLatin1("%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()
|
|
|
|
|
{
|
2012-09-20 10:31:34 +02:00
|
|
|
beginResetModel();
|
2008-12-02 12:01:29 +01:00
|
|
|
m_registers.clear();
|
2012-09-20 10:31:34 +02:00
|
|
|
endResetModel();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|
|
|
{
|
2012-09-20 10:31:34 +02:00
|
|
|
beginResetModel();
|
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();
|
2012-09-20 10:31:34 +02:00
|
|
|
endResetModel();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
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) {
|
2012-09-20 10:31:34 +02:00
|
|
|
beginResetModel();
|
2010-09-22 09:41:15 +02:00
|
|
|
m_base = base;
|
|
|
|
|
calculateWidth();
|
2012-09-20 10:31:34 +02:00
|
|
|
endResetModel();
|
2010-09-22 09:41:15 +02:00
|
|
|
}
|
2009-08-13 16:16:19 +02:00
|
|
|
}
|
2010-11-05 13:35:31 +01:00
|
|
|
|
|
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|