2009-08-12 10:51:25 +02:00
|
|
|
/**************************************************************************
|
|
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2010-03-05 11:25:49 +01:00
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
2009-08-12 10:51:25 +02:00
|
|
|
**
|
|
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
|
|
|
|
**
|
|
|
|
|
** Commercial Usage
|
|
|
|
|
**
|
|
|
|
|
** Licensees holding valid Qt Commercial licenses may use this file in
|
|
|
|
|
** accordance with the Qt Commercial License Agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
|
|
|
** a written agreement between you and Nokia.
|
|
|
|
|
**
|
|
|
|
|
** 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.
|
|
|
|
|
**
|
|
|
|
|
** If you are unsure which license is appropriate for your use, please
|
2009-08-14 09:30:56 +02:00
|
|
|
** contact the sales department at http://qt.nokia.com/contact.
|
2009-08-12 10:51:25 +02:00
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include "debuggeragents.h"
|
2010-06-16 11:08:54 +02:00
|
|
|
|
|
|
|
|
#include "debuggerengine.h"
|
|
|
|
|
#include "debuggerplugin.h"
|
2009-09-25 15:02:16 +02:00
|
|
|
#include "debuggerstringutils.h"
|
2010-06-16 11:08:54 +02:00
|
|
|
#include "stackframe.h"
|
2009-08-12 10:51:25 +02:00
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
#include <coreplugin/coreconstants.h>
|
2009-08-12 10:51:25 +02:00
|
|
|
#include <coreplugin/editormanager/editormanager.h>
|
2009-08-14 13:04:05 +02:00
|
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
|
|
|
#include <coreplugin/icore.h>
|
|
|
|
|
|
|
|
|
|
#include <texteditor/basetexteditor.h>
|
|
|
|
|
#include <texteditor/basetextmark.h>
|
|
|
|
|
#include <texteditor/itexteditor.h>
|
|
|
|
|
#include <texteditor/texteditorconstants.h>
|
|
|
|
|
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2009-10-02 11:45:19 +02:00
|
|
|
#include <QtCore/QDebug>
|
2009-11-19 09:51:21 +01:00
|
|
|
|
|
|
|
|
#include <QtGui/QMessageBox>
|
2009-08-14 13:04:05 +02:00
|
|
|
#include <QtGui/QPlainTextEdit>
|
|
|
|
|
#include <QtGui/QTextCursor>
|
2009-08-18 11:23:01 +02:00
|
|
|
#include <QtGui/QSyntaxHighlighter>
|
2009-08-12 10:51:25 +02:00
|
|
|
|
2009-08-12 15:11:05 +02:00
|
|
|
#include <limits.h>
|
|
|
|
|
|
2009-08-12 10:51:25 +02:00
|
|
|
namespace Debugger {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// MemoryViewAgent
|
|
|
|
|
//
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2009-09-08 11:59:21 +02:00
|
|
|
/*!
|
2009-08-14 13:04:05 +02:00
|
|
|
\class MemoryViewAgent
|
|
|
|
|
|
|
|
|
|
Objects form this class are created in response to user actions in
|
|
|
|
|
the Gui for showing raw memory from the inferior. After creation
|
|
|
|
|
it handles communication between the engine and the bineditor.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
MemoryViewAgent::MemoryViewAgent(DebuggerEngine *engine, quint64 addr)
|
|
|
|
|
: QObject(engine), m_engine(engine)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
QTC_ASSERT(engine, /**/);
|
2010-02-17 17:33:42 +01:00
|
|
|
createBinEditor(addr);
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
MemoryViewAgent::MemoryViewAgent(DebuggerEngine *engine, const QString &addr)
|
|
|
|
|
: QObject(engine), m_engine(engine)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
QTC_ASSERT(engine, /**/);
|
2009-08-14 13:04:05 +02:00
|
|
|
bool ok = true;
|
2010-02-17 17:33:42 +01:00
|
|
|
createBinEditor(addr.toULongLong(&ok, 0));
|
2009-08-18 08:34:48 +02:00
|
|
|
//qDebug() << " ADDRESS: " << addr << addr.toUInt(&ok, 0);
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
MemoryViewAgent::~MemoryViewAgent()
|
|
|
|
|
{
|
2010-02-17 17:33:42 +01:00
|
|
|
foreach (QPointer<Core::IEditor> editor, m_editors) {
|
|
|
|
|
if (editor)
|
|
|
|
|
editor->deleteLater();
|
|
|
|
|
}
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2010-02-17 17:33:42 +01:00
|
|
|
void MemoryViewAgent::createBinEditor(quint64 addr)
|
2009-08-12 10:51:25 +02:00
|
|
|
{
|
|
|
|
|
Core::EditorManager *editorManager = Core::EditorManager::instance();
|
2009-08-19 14:41:51 +02:00
|
|
|
QString titlePattern = tr("Memory $");
|
2010-02-17 17:33:42 +01:00
|
|
|
Core::IEditor *editor = editorManager->openEditorWithContents(
|
2010-01-07 18:17:24 +01:00
|
|
|
Core::Constants::K_DEFAULT_BINARY_EDITOR_ID,
|
2009-08-12 10:51:25 +02:00
|
|
|
&titlePattern);
|
2010-02-17 17:33:42 +01:00
|
|
|
if (editor) {
|
|
|
|
|
connect(editor->widget(), SIGNAL(lazyDataRequested(Core::IEditor *, quint64,bool)),
|
|
|
|
|
this, SLOT(fetchLazyData(Core::IEditor *, quint64,bool)));
|
|
|
|
|
connect(editor->widget(), SIGNAL(newWindowRequested(quint64)),
|
|
|
|
|
this, SLOT(createBinEditor(quint64)));
|
|
|
|
|
m_editors << editor;
|
|
|
|
|
editorManager->activateEditor(editor);
|
|
|
|
|
QMetaObject::invokeMethod(editor->widget(), "setNewWindowRequestAllowed");
|
|
|
|
|
QMetaObject::invokeMethod(editor->widget(), "setLazyData",
|
2009-11-19 09:51:21 +01:00
|
|
|
Q_ARG(quint64, addr), Q_ARG(int, 1024 * 1024), Q_ARG(int, BinBlockSize));
|
|
|
|
|
} else {
|
2010-06-16 11:08:54 +02:00
|
|
|
DebuggerPlugin::instance()->showMessageBox(QMessageBox::Warning,
|
2009-11-19 09:51:21 +01:00
|
|
|
tr("No memory viewer available"),
|
2009-12-04 12:18:47 +01:00
|
|
|
tr("The memory contents cannot be shown as no viewer plugin "
|
2010-01-08 11:31:38 +01:00
|
|
|
"for binary data has been loaded."));
|
2009-11-19 09:51:21 +01:00
|
|
|
deleteLater();
|
|
|
|
|
}
|
2009-08-12 10:51:25 +02:00
|
|
|
}
|
|
|
|
|
|
2010-02-17 17:33:42 +01:00
|
|
|
void MemoryViewAgent::fetchLazyData(Core::IEditor *editor, quint64 block, bool sync)
|
2009-08-12 10:51:25 +02:00
|
|
|
{
|
|
|
|
|
Q_UNUSED(sync); // FIXME: needed support for incremental searching
|
2010-06-16 11:08:54 +02:00
|
|
|
m_engine->fetchMemory(this, editor, BinBlockSize * block, BinBlockSize);
|
2009-08-12 10:51:25 +02:00
|
|
|
}
|
|
|
|
|
|
2010-02-17 17:33:42 +01:00
|
|
|
void MemoryViewAgent::addLazyData(QObject *editorToken, quint64 addr,
|
|
|
|
|
const QByteArray &ba)
|
2009-08-12 10:51:25 +02:00
|
|
|
{
|
2010-02-17 17:33:42 +01:00
|
|
|
Core::IEditor *editor = qobject_cast<Core::IEditor *>(editorToken);
|
|
|
|
|
if (editor && editor->widget()) {
|
|
|
|
|
Core::EditorManager::instance()->activateEditor(editor);
|
|
|
|
|
QMetaObject::invokeMethod(editor->widget(), "addLazyData",
|
2009-09-17 08:59:35 +02:00
|
|
|
Q_ARG(quint64, addr / BinBlockSize), Q_ARG(QByteArray, ba));
|
2010-02-17 17:33:42 +01:00
|
|
|
}
|
2009-08-12 10:51:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// DisassemblerViewAgent
|
|
|
|
|
//
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
// Used for the disassembler view
|
|
|
|
|
class LocationMark2 : public TextEditor::ITextMark
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
LocationMark2() {}
|
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
QIcon icon() const { return DebuggerPlugin::instance()->locationMarkIcon(); }
|
2009-08-14 13:04:05 +02:00
|
|
|
void updateLineNumber(int /*lineNumber*/) {}
|
|
|
|
|
void updateBlock(const QTextBlock & /*block*/) {}
|
|
|
|
|
void removedFromEditor() {}
|
|
|
|
|
void documentClosing() {}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct DisassemblerViewAgentPrivate
|
|
|
|
|
{
|
2010-02-01 16:14:57 +01:00
|
|
|
DisassemblerViewAgentPrivate() { tryMixed = true; }
|
|
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
QPointer<TextEditor::ITextEditor> editor;
|
2009-09-29 16:17:01 +02:00
|
|
|
StackFrame frame;
|
2010-02-01 16:14:57 +01:00
|
|
|
bool tryMixed;
|
2010-06-16 11:08:54 +02:00
|
|
|
QPointer<DebuggerEngine> engine;
|
2009-08-14 13:04:05 +02:00
|
|
|
LocationMark2 *locationMark;
|
2009-09-29 16:17:01 +02:00
|
|
|
QHash<QString, QString> cache;
|
2009-08-14 13:04:05 +02:00
|
|
|
};
|
|
|
|
|
|
2009-08-18 11:23:01 +02:00
|
|
|
/*!
|
|
|
|
|
\class DisassemblerSyntaxHighlighter
|
|
|
|
|
|
|
|
|
|
Simple syntax highlighter to make the disassembler text less prominent.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
class DisassemblerHighlighter : public QSyntaxHighlighter
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
DisassemblerHighlighter(QPlainTextEdit *parent)
|
|
|
|
|
: QSyntaxHighlighter(parent->document())
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
void highlightBlock(const QString &text)
|
|
|
|
|
{
|
|
|
|
|
if (!text.isEmpty() && text.at(0) != ' ') {
|
|
|
|
|
QTextCharFormat format;
|
|
|
|
|
format.setForeground(QColor(128, 128, 128));
|
|
|
|
|
setFormat(0, text.size(), format);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
/*!
|
|
|
|
|
\class DisassemblerViewAgent
|
|
|
|
|
|
|
|
|
|
Objects from this class are created in response to user actions in
|
|
|
|
|
the Gui for showing disassembled memory from the inferior. After creation
|
|
|
|
|
it handles communication between the engine and the editor.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-06-16 11:08:54 +02:00
|
|
|
DisassemblerViewAgent::DisassemblerViewAgent(DebuggerEngine *engine)
|
2009-10-06 09:22:57 +02:00
|
|
|
: QObject(0), d(new DisassemblerViewAgentPrivate)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
|
|
|
|
d->editor = 0;
|
|
|
|
|
d->locationMark = new LocationMark2();
|
2010-06-16 11:08:54 +02:00
|
|
|
d->engine = engine;
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DisassemblerViewAgent::~DisassemblerViewAgent()
|
|
|
|
|
{
|
|
|
|
|
if (d->editor)
|
|
|
|
|
d->editor->deleteLater();
|
2009-09-29 16:17:01 +02:00
|
|
|
d->editor = 0;
|
2009-11-04 17:11:35 +01:00
|
|
|
delete d->locationMark;
|
|
|
|
|
d->locationMark = 0;
|
2009-08-14 13:04:05 +02:00
|
|
|
delete d;
|
2009-09-28 09:41:07 +02:00
|
|
|
d = 0;
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2009-09-29 16:17:01 +02:00
|
|
|
void DisassemblerViewAgent::cleanup()
|
|
|
|
|
{
|
|
|
|
|
d->cache.clear();
|
|
|
|
|
//if (d->editor)
|
|
|
|
|
// d->editor->deleteLater();
|
|
|
|
|
//d->editor = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-01 16:23:26 +02:00
|
|
|
void DisassemblerViewAgent::resetLocation()
|
|
|
|
|
{
|
|
|
|
|
if (d->editor)
|
|
|
|
|
d->editor->markableInterface()->removeMark(d->locationMark);
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-02 11:45:19 +02:00
|
|
|
QString frameKey(const StackFrame &frame)
|
|
|
|
|
{
|
|
|
|
|
return _("%1:%2:%3").arg(frame.function).arg(frame.file).arg(frame.from);
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-01 16:14:57 +01:00
|
|
|
const StackFrame &DisassemblerViewAgent::frame() const
|
|
|
|
|
{
|
|
|
|
|
return d->frame;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool DisassemblerViewAgent::isMixed() const
|
|
|
|
|
{
|
|
|
|
|
return d->tryMixed
|
|
|
|
|
&& d->frame.line > 0
|
|
|
|
|
&& !d->frame.function.isEmpty()
|
|
|
|
|
&& d->frame.function != _("??");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DisassemblerViewAgent::setFrame(const StackFrame &frame, bool tryMixed)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2009-09-29 16:17:01 +02:00
|
|
|
d->frame = frame;
|
2010-02-01 16:14:57 +01:00
|
|
|
d->tryMixed = tryMixed;
|
|
|
|
|
if (isMixed()) {
|
2009-10-02 11:45:19 +02:00
|
|
|
QHash<QString, QString>::ConstIterator it = d->cache.find(frameKey(frame));
|
2009-09-29 16:17:01 +02:00
|
|
|
if (it != d->cache.end()) {
|
2010-01-11 10:22:55 +01:00
|
|
|
QString msg = _("Use cache disassembler for '%1' in '%2'")
|
2009-10-02 11:45:19 +02:00
|
|
|
.arg(frame.function).arg(frame.file);
|
2010-06-16 11:08:54 +02:00
|
|
|
d->engine->showMessage(msg);
|
2009-09-29 16:17:01 +02:00
|
|
|
setContents(*it);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2010-01-29 21:33:57 +01:00
|
|
|
}
|
2010-06-16 11:08:54 +02:00
|
|
|
d->engine->fetchDisassembler(this);
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DisassemblerViewAgent::setContents(const QString &contents)
|
|
|
|
|
{
|
2009-09-28 09:41:07 +02:00
|
|
|
QTC_ASSERT(d, return);
|
2009-08-14 13:04:05 +02:00
|
|
|
using namespace Core;
|
|
|
|
|
using namespace TextEditor;
|
|
|
|
|
|
2009-10-02 11:45:19 +02:00
|
|
|
d->cache.insert(frameKey(d->frame), contents);
|
2009-08-18 11:23:01 +02:00
|
|
|
QPlainTextEdit *plainTextEdit = 0;
|
2009-08-14 13:04:05 +02:00
|
|
|
EditorManager *editorManager = EditorManager::instance();
|
|
|
|
|
if (!d->editor) {
|
|
|
|
|
QString titlePattern = "Disassembler";
|
|
|
|
|
d->editor = qobject_cast<ITextEditor *>(
|
|
|
|
|
editorManager->openEditorWithContents(
|
2010-01-07 18:17:24 +01:00
|
|
|
Core::Constants::K_DEFAULT_TEXT_EDITOR_ID,
|
2009-08-18 17:27:30 +02:00
|
|
|
&titlePattern));
|
2010-03-16 16:43:03 +01:00
|
|
|
d->editor->setProperty("OpenedByDebugger", true);
|
2010-04-09 14:55:09 +02:00
|
|
|
d->editor->setProperty("DisassemblerView", true);
|
2009-08-18 17:27:30 +02:00
|
|
|
QTC_ASSERT(d->editor, return);
|
2009-08-18 11:23:01 +02:00
|
|
|
if ((plainTextEdit = qobject_cast<QPlainTextEdit *>(d->editor->widget())))
|
|
|
|
|
(void) new DisassemblerHighlighter(plainTextEdit);
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
editorManager->activateEditor(d->editor);
|
|
|
|
|
|
2009-08-18 11:23:01 +02:00
|
|
|
plainTextEdit = qobject_cast<QPlainTextEdit *>(d->editor->widget());
|
2009-08-14 13:04:05 +02:00
|
|
|
if (plainTextEdit)
|
|
|
|
|
plainTextEdit->setPlainText(contents);
|
|
|
|
|
|
|
|
|
|
d->editor->markableInterface()->removeMark(d->locationMark);
|
2009-09-29 16:17:01 +02:00
|
|
|
d->editor->setDisplayName(_("Disassembler (%1)").arg(d->frame.function));
|
2009-08-14 13:04:05 +02:00
|
|
|
|
|
|
|
|
for (int pos = 0, line = 0; ; ++line, ++pos) {
|
2009-09-29 16:17:01 +02:00
|
|
|
if (contents.midRef(pos, d->frame.address.size()) == d->frame.address) {
|
2009-08-14 13:04:05 +02:00
|
|
|
d->editor->markableInterface()->addMark(d->locationMark, line + 1);
|
|
|
|
|
if (plainTextEdit) {
|
|
|
|
|
QTextCursor tc = plainTextEdit->textCursor();
|
|
|
|
|
tc.setPosition(pos);
|
|
|
|
|
plainTextEdit->setTextCursor(tc);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
pos = contents.indexOf('\n', pos + 1);
|
|
|
|
|
if (pos == -1)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-10-02 11:45:19 +02:00
|
|
|
bool DisassemblerViewAgent::contentsCoversAddress(const QString &contents) const
|
|
|
|
|
{
|
|
|
|
|
QTC_ASSERT(d, return false);
|
2010-01-29 21:33:57 +01:00
|
|
|
for (int pos = 0, line = 0; ; ++line, ++pos) {
|
2009-10-02 11:45:19 +02:00
|
|
|
if (contents.midRef(pos, d->frame.address.size()) == d->frame.address)
|
|
|
|
|
return true;
|
|
|
|
|
pos = contents.indexOf('\n', pos + 1);
|
|
|
|
|
if (pos == -1)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
QString DisassemblerViewAgent::address() const
|
|
|
|
|
{
|
2009-09-29 16:17:01 +02:00
|
|
|
return d->frame.address;
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2009-08-12 10:51:25 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|