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

300 lines
10 KiB
C++
Raw Normal View History

/****************************************************************************
2008-12-02 12:01:29 +01:00
**
** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies).
** Contact: http://www.qt-project.org/legal
2008-12-02 12:01:29 +01:00
**
** This file is part of Qt Creator.
2008-12-02 12:01:29 +01: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
** 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
2010-12-17 16:01:08 +01:00
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
****************************************************************************/
2008-12-02 15:08:31 +01:00
2008-12-02 12:01:29 +01:00
#include "moduleshandler.h"
#include <utils/elfreader.h>
2009-02-17 11:16:23 +01:00
#include <utils/qtcassert.h>
2008-12-02 12:01:29 +01:00
#include <QDebug>
#include <QSortFilterProxyModel>
2008-12-02 12:01:29 +01:00
using namespace Utils;
2008-12-02 12:01:29 +01:00
//////////////////////////////////////////////////////////////////
//
// ModulesModel
//
//////////////////////////////////////////////////////////////////
namespace Debugger {
namespace Internal {
2009-04-16 19:42:58 +02:00
class ModulesModel : public QAbstractItemModel
{
public:
explicit ModulesModel(QObject *parent)
: QAbstractItemModel(parent)
{}
int columnCount(const QModelIndex &parent) const
{ return parent.isValid() ? 0 : 5; }
int rowCount(const QModelIndex &parent) const
{ return parent.isValid() ? 0 : m_modules.size(); }
QModelIndex parent(const QModelIndex &) const { return QModelIndex(); }
QModelIndex index(int row, int column, const QModelIndex &) const
{ return createIndex(row, column); }
QVariant headerData(int section, Qt::Orientation orientation, int role) const;
QVariant data(const QModelIndex &index, int role) const;
void clearModel();
void removeModule(const QString &modulePath);
void setModules(const Modules &modules);
void updateModule(const Module &module);
int indexOfModule(const QString &modulePath) const;
Modules m_modules;
};
2008-12-02 12:01:29 +01:00
QVariant ModulesModel::headerData(int section,
Qt::Orientation orientation, int role) const
{
if (orientation == Qt::Horizontal && role == Qt::DisplayRole) {
static QString headers[] = {
ModulesHandler::tr("Module Name") + QLatin1String(" "),
ModulesHandler::tr("Module Path") + QLatin1String(" "),
ModulesHandler::tr("Symbols Read") + QLatin1String(" "),
ModulesHandler::tr("Symbols Type") + QLatin1String(" "),
ModulesHandler::tr("Start Address") + QLatin1String(" "),
ModulesHandler::tr("End Address") + QLatin1String(" ")
2008-12-02 12:01:29 +01:00
};
return headers[section];
}
return QVariant();
}
QVariant ModulesModel::data(const QModelIndex &index, int role) const
{
int row = index.row();
if (row < 0 || row >= m_modules.size())
return QVariant();
const Module &module = m_modules.at(row);
switch (index.column()) {
case 0:
if (role == Qt::DisplayRole)
return module.moduleName;
// FIXME: add icons
//if (role == Qt::DecorationRole)
// return module.symbolsRead ? icon2 : icon;
break;
case 1:
if (role == Qt::DisplayRole)
2010-05-03 19:12:52 +02:00
return module.modulePath;
if (role == Qt::ToolTipRole) {
QString msg;
if (!module.elfData.buildId.isEmpty())
msg += QString::fromLatin1("Build Id: " + module.elfData.buildId);
if (!module.elfData.debugLink.isEmpty())
msg += QString::fromLatin1("Debug Link: " + module.elfData.debugLink);
return msg;
}
2008-12-02 12:01:29 +01:00
break;
case 2:
if (role == Qt::DisplayRole)
switch (module.symbolsRead) {
case Module::UnknownReadState: return ModulesHandler::tr("Unknown");
case Module::ReadFailed: return ModulesHandler::tr("No");
case Module::ReadOk: return ModulesHandler::tr("Yes");
}
2008-12-02 12:01:29 +01:00
break;
case 3:
2010-05-03 19:12:52 +02:00
if (role == Qt::DisplayRole)
switch (module.elfData.symbolsType) {
case UnknownSymbols:
return ModulesHandler::tr("Unknown");
case NoSymbols:
return ModulesHandler::tr("None");
case PlainSymbols:
return ModulesHandler::tr("Plain");
case FastSymbols:
return ModulesHandler::tr("Fast");
case LinkedSymbols:
return ModulesHandler::tr("debuglnk");
case BuildIdSymbols:
return ModulesHandler::tr("buildid");
}
else if (role == Qt::ToolTipRole)
switch (module.elfData.symbolsType) {
case UnknownSymbols:
return ModulesHandler::tr(
"It is unknown whether this module contains debug "
"information.\nUse \"Examine Symbols\" from the "
"context menu to initiate a check.");
case NoSymbols:
return ModulesHandler::tr(
"This module neither contains nor references debug "
"information.\nStepping into the module or setting "
"breakpoints by file and line will not work.");
case PlainSymbols:
return ModulesHandler::tr(
"This module contains debug information.\nStepping "
"into the module or setting breakpoints by file and "
"is expected to work.");
case FastSymbols:
return ModulesHandler::tr(
"This module contains debug information.\nStepping "
"into the module or setting breakpoints by file and "
"is expected to work.");
case LinkedSymbols:
case BuildIdSymbols:
return ModulesHandler::tr(
"This module does not contain debug information "
"itself, but contains a reference to external "
"debug information.");
}
2010-05-03 19:12:52 +02:00
break;
case 4:
if (role == Qt::DisplayRole)
if (module.startAddress)
return QString(QLatin1String("0x")
+ QString::number(module.startAddress, 16));
break;
case 5:
if (role == Qt::DisplayRole) {
if (module.endAddress)
return QString(QLatin1String("0x")
+ QString::number(module.endAddress, 16));
//: End address of loaded module
return ModulesHandler::tr("<unknown>", "address");
}
2008-12-02 12:01:29 +01:00
break;
}
return QVariant();
}
void ModulesModel::setModules(const Modules &m)
{
m_modules = m;
reset();
}
void ModulesModel::clearModel()
{
if (!m_modules.isEmpty()) {
m_modules.clear();
reset();
}
}
int ModulesModel::indexOfModule(const QString &modulePath) const
{
// Recent modules are more likely to be unloaded first.
for (int i = m_modules.size() - 1; i >= 0; i--)
if (m_modules.at(i).modulePath == modulePath)
return i;
return -1;
}
void ModulesModel::removeModule(const QString &modulePath)
{
const int row = indexOfModule(modulePath);
QTC_ASSERT(row != -1, return);
beginRemoveRows(QModelIndex(), row, row);
m_modules.remove(row);
endRemoveRows();
}
2008-12-02 12:01:29 +01:00
void ModulesModel::updateModule(const Module &module)
{
const int row = indexOfModule(module.modulePath);
const QString path = module.modulePath;
if (path.isEmpty())
return;
try { // MinGW occasionallly throws std::bad_alloc.
ElfReader reader(path);
ElfData elfData = reader.readHeaders();
if (row == -1) {
const int n = m_modules.size();
beginInsertRows(QModelIndex(), n, n);
m_modules.push_back(module);
m_modules.back().elfData = elfData;
endInsertRows();
} else {
m_modules[row] = module;
m_modules[row].elfData = elfData;
dataChanged(index(row, 0, QModelIndex()), index(row, 4, QModelIndex()));
}
} catch(...) {
qWarning("%s: An exception occurred while reading module '%s'",
Q_FUNC_INFO, qPrintable(module.modulePath));
}
}
2008-12-02 12:01:29 +01:00
//////////////////////////////////////////////////////////////////
//
// ModulesHandler
//
//////////////////////////////////////////////////////////////////
ModulesHandler::ModulesHandler(DebuggerEngine *engine)
2008-12-02 12:01:29 +01:00
{
m_engine = engine;
m_model = new ModulesModel(this);
2008-12-02 12:01:29 +01:00
m_proxyModel = new QSortFilterProxyModel(this);
m_proxyModel->setSourceModel(m_model);
}
QAbstractItemModel *ModulesHandler::model() const
{
return m_proxyModel;
}
void ModulesHandler::removeAll()
{
m_model->clearModel();
}
void ModulesHandler::removeModule(const QString &modulePath)
{
m_model->removeModule(modulePath);
}
2008-12-02 12:01:29 +01:00
void ModulesHandler::updateModule(const Module &module)
{
m_model->updateModule(module);
}
void ModulesHandler::setModules(const Modules &modules)
2008-12-02 12:01:29 +01:00
{
m_model->setModules(modules);
2008-12-02 12:01:29 +01:00
}
Modules ModulesHandler::modules() const
2008-12-02 12:01:29 +01:00
{
return m_model->m_modules;
2008-12-02 12:01:29 +01:00
}
2009-04-16 19:42:58 +02:00
} // namespace Internal
} // namespace Debugger