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)
|
|
|
|
|
**
|
2010-12-17 17:14:20 +01:00
|
|
|
** No Commercial Usage
|
2009-08-12 10:51:25 +02:00
|
|
|
**
|
2010-12-17 17:14:20 +01:00
|
|
|
** This file contains pre-release code and may not be distributed.
|
|
|
|
|
** You may use this file in accordance with the terms and conditions
|
|
|
|
|
** contained in the Technology Preview License Agreement accompanying
|
|
|
|
|
** this package.
|
2009-08-12 10:51:25 +02:00
|
|
|
**
|
|
|
|
|
** 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.
|
|
|
|
|
**
|
2010-12-17 17:14:20 +01:00
|
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please contact
|
|
|
|
|
** Nokia at qt-info@nokia.com.
|
2009-08-12 10:51:25 +02:00
|
|
|
**
|
|
|
|
|
**************************************************************************/
|
|
|
|
|
|
2010-12-08 12:43:11 +01:00
|
|
|
#include "disassembleragent.h"
|
2010-06-16 11:08:54 +02:00
|
|
|
|
2010-11-25 11:51:09 +01:00
|
|
|
#include "breakhandler.h"
|
2010-06-16 11:08:54 +02:00
|
|
|
#include "debuggerengine.h"
|
2010-11-10 11:39:01 +01:00
|
|
|
#include "debuggercore.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>
|
|
|
|
|
#include <coreplugin/editormanager/ieditor.h>
|
|
|
|
|
#include <coreplugin/icore.h>
|
2010-12-08 12:43:11 +01:00
|
|
|
#include <coreplugin/mimedatabase.h>
|
2009-08-14 13:04:05 +02:00
|
|
|
|
2010-11-24 18:36:17 +01:00
|
|
|
#include <texteditor/basetextdocument.h>
|
2009-08-14 13:04:05 +02:00
|
|
|
#include <texteditor/basetexteditor.h>
|
|
|
|
|
#include <texteditor/basetextmark.h>
|
2010-11-24 18:36:17 +01:00
|
|
|
#include <texteditor/plaintexteditor.h>
|
2009-08-14 13:04:05 +02:00
|
|
|
#include <texteditor/texteditorconstants.h>
|
|
|
|
|
|
|
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2010-11-24 15:55:09 +01:00
|
|
|
#include <QtGui/QTextBlock>
|
2010-10-29 15:20:10 +02:00
|
|
|
#include <QtGui/QIcon>
|
2009-08-12 10:51:25 +02:00
|
|
|
|
2009-08-12 15:11:05 +02:00
|
|
|
|
2010-09-09 17:58:26 +02:00
|
|
|
using namespace Core;
|
|
|
|
|
|
2009-08-12 10:51:25 +02:00
|
|
|
namespace Debugger {
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
2010-12-14 12:29:32 +01:00
|
|
|
// DisassemblerAgent
|
2009-08-14 13:04:05 +02:00
|
|
|
//
|
|
|
|
|
///////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
class LocationMark2 : public TextEditor::ITextMark
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
LocationMark2() {}
|
|
|
|
|
|
2010-11-10 11:39:01 +01:00
|
|
|
QIcon icon() const { return debuggerCore()->locationMarkIcon(); }
|
2009-08-14 13:04:05 +02:00
|
|
|
void updateLineNumber(int /*lineNumber*/) {}
|
|
|
|
|
void updateBlock(const QTextBlock & /*block*/) {}
|
|
|
|
|
void removedFromEditor() {}
|
|
|
|
|
void documentClosing() {}
|
|
|
|
|
};
|
|
|
|
|
|
2010-11-25 11:51:09 +01:00
|
|
|
class BreakpointMarker2 : public TextEditor::ITextMark
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
BreakpointMarker2(const QIcon &icon) : m_icon(icon) {}
|
|
|
|
|
|
|
|
|
|
QIcon icon() const { return m_icon; }
|
|
|
|
|
void updateLineNumber(int) {}
|
|
|
|
|
void updateBlock(const QTextBlock &) {}
|
|
|
|
|
void removedFromEditor() {}
|
|
|
|
|
void documentClosing() {}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QIcon m_icon;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
class DisassemblerAgentPrivate
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2010-11-24 15:55:09 +01:00
|
|
|
public:
|
2010-12-14 12:29:32 +01:00
|
|
|
DisassemblerAgentPrivate();
|
|
|
|
|
~DisassemblerAgentPrivate();
|
2010-06-22 14:06:42 +02:00
|
|
|
void configureMimeType();
|
2010-02-01 16:14:57 +01:00
|
|
|
|
2010-11-24 15:55:09 +01:00
|
|
|
public:
|
2009-08-14 13:04:05 +02:00
|
|
|
QPointer<TextEditor::ITextEditor> editor;
|
2010-12-16 19:06:33 +01:00
|
|
|
Location location;
|
2010-02-01 16:14:57 +01:00
|
|
|
bool tryMixed;
|
2010-06-16 11:08:54 +02:00
|
|
|
QPointer<DebuggerEngine> engine;
|
2010-11-25 11:51:09 +01:00
|
|
|
TextEditor::ITextMark *locationMark;
|
|
|
|
|
QList<TextEditor::ITextMark *> breakpointMarks;
|
|
|
|
|
|
2010-11-24 15:55:09 +01:00
|
|
|
QHash<QString, DisassemblerLines> cache;
|
2010-06-22 14:06:42 +02:00
|
|
|
QString mimeType;
|
2009-08-14 13:04:05 +02:00
|
|
|
};
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
DisassemblerAgentPrivate::DisassemblerAgentPrivate()
|
2010-11-24 15:55:09 +01:00
|
|
|
: editor(0),
|
2010-06-22 14:06:42 +02:00
|
|
|
tryMixed(true),
|
|
|
|
|
locationMark(new LocationMark2),
|
|
|
|
|
mimeType(_("text/x-qtcreator-generic-asm"))
|
2009-08-18 11:23:01 +02:00
|
|
|
{
|
2010-06-22 14:06:42 +02:00
|
|
|
}
|
2009-08-18 11:23:01 +02:00
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
DisassemblerAgentPrivate::~DisassemblerAgentPrivate()
|
2010-11-25 11:51:09 +01:00
|
|
|
{
|
|
|
|
|
if (editor) {
|
|
|
|
|
EditorManager *editorManager = EditorManager::instance();
|
|
|
|
|
editorManager->closeEditors(QList<IEditor *>() << editor);
|
|
|
|
|
}
|
|
|
|
|
editor = 0;
|
|
|
|
|
delete locationMark;
|
|
|
|
|
}
|
2010-11-24 15:55:09 +01:00
|
|
|
|
2009-08-14 13:04:05 +02:00
|
|
|
/*!
|
2010-12-14 12:29:32 +01:00
|
|
|
\class DisassemblerAgent
|
2009-08-14 13:04:05 +02:00
|
|
|
|
|
|
|
|
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-12-14 12:29:32 +01:00
|
|
|
DisassemblerAgent::DisassemblerAgent(DebuggerEngine *engine)
|
|
|
|
|
: QObject(0), d(new DisassemblerAgentPrivate)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2010-06-16 11:08:54 +02:00
|
|
|
d->engine = engine;
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
DisassemblerAgent::~DisassemblerAgent()
|
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
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgent::cleanup()
|
2009-09-29 16:17:01 +02:00
|
|
|
{
|
|
|
|
|
d->cache.clear();
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgent::resetLocation()
|
2009-10-01 16:23:26 +02:00
|
|
|
{
|
2010-11-25 11:51:09 +01:00
|
|
|
if (!d->editor)
|
|
|
|
|
return;
|
|
|
|
|
d->editor->markableInterface()->removeMark(d->locationMark);
|
2009-10-01 16:23:26 +02:00
|
|
|
}
|
|
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
static QString frameKey(const Location &loc)
|
2009-10-02 11:45:19 +02:00
|
|
|
{
|
2010-12-16 19:06:33 +01:00
|
|
|
return _("%1:%2:%3").arg(loc.functionName())
|
|
|
|
|
.arg(loc.fileName()).arg(loc.address());
|
2009-10-02 11:45:19 +02:00
|
|
|
}
|
|
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
const Location &DisassemblerAgent::location() const
|
2010-02-01 16:14:57 +01:00
|
|
|
{
|
2010-12-16 19:06:33 +01:00
|
|
|
return d->location;
|
2010-02-01 16:14:57 +01:00
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
bool DisassemblerAgent::isMixed() const
|
2010-02-01 16:14:57 +01:00
|
|
|
{
|
|
|
|
|
return d->tryMixed
|
2010-12-16 19:06:33 +01:00
|
|
|
&& d->location.lineNumber() > 0
|
|
|
|
|
&& !d->location.functionName().isEmpty()
|
|
|
|
|
&& d->location.functionName() != _("??");
|
2010-02-01 16:14:57 +01:00
|
|
|
}
|
|
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
void DisassemblerAgent::setLocation(const Location &loc)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2010-12-16 19:06:33 +01:00
|
|
|
d->location = loc;
|
2010-02-01 16:14:57 +01:00
|
|
|
if (isMixed()) {
|
2010-11-24 15:55:09 +01:00
|
|
|
QHash<QString, DisassemblerLines>::ConstIterator it =
|
2010-12-16 19:06:33 +01:00
|
|
|
d->cache.find(frameKey(loc));
|
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'")
|
2010-12-16 19:06:33 +01:00
|
|
|
.arg(loc.functionName()).arg(loc.fileName());
|
2010-06-16 11:08:54 +02:00
|
|
|
d->engine->showMessage(msg);
|
2009-09-29 16:17:01 +02:00
|
|
|
setContents(*it);
|
2010-11-25 11:51:09 +01:00
|
|
|
updateBreakpointMarkers();
|
|
|
|
|
updateLocationMarker();
|
2009-09-29 16:17:01 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgentPrivate::configureMimeType()
|
2010-06-22 14:06:42 +02:00
|
|
|
{
|
|
|
|
|
QTC_ASSERT(editor, return);
|
|
|
|
|
|
2010-11-19 09:14:08 +01:00
|
|
|
TextEditor::BaseTextDocument *doc =
|
|
|
|
|
qobject_cast<TextEditor::BaseTextDocument *>(editor->file());
|
2010-06-22 14:06:42 +02:00
|
|
|
QTC_ASSERT(doc, return);
|
|
|
|
|
doc->setMimeType(mimeType);
|
|
|
|
|
|
2010-11-19 09:14:08 +01:00
|
|
|
TextEditor::PlainTextEditor *pe =
|
|
|
|
|
qobject_cast<TextEditor::PlainTextEditor *>(editor->widget());
|
2010-06-22 14:06:42 +02:00
|
|
|
QTC_ASSERT(pe, return);
|
|
|
|
|
|
2010-11-19 09:14:08 +01:00
|
|
|
MimeType mtype = ICore::instance()->mimeDatabase()->findByType(mimeType);
|
|
|
|
|
if (mtype)
|
2010-06-22 14:06:42 +02:00
|
|
|
pe->configure(mtype);
|
2010-11-19 09:14:08 +01:00
|
|
|
else
|
2010-06-22 14:06:42 +02:00
|
|
|
qWarning("Assembler mimetype '%s' not found.", qPrintable(mimeType));
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
QString DisassemblerAgent::mimeType() const
|
2010-06-22 14:06:42 +02:00
|
|
|
{
|
|
|
|
|
return d->mimeType;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgent::setMimeType(const QString &mt)
|
2010-06-22 14:06:42 +02:00
|
|
|
{
|
|
|
|
|
if (mt == d->mimeType)
|
|
|
|
|
return;
|
|
|
|
|
d->mimeType = mt;
|
|
|
|
|
if (d->editor)
|
|
|
|
|
d->configureMimeType();
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgent::setContents(const DisassemblerLines &contents)
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
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;
|
|
|
|
|
|
|
|
|
|
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-06-22 14:06:42 +02:00
|
|
|
QTC_ASSERT(d->editor, return);
|
2010-10-25 13:37:46 +02:00
|
|
|
d->editor->setProperty(Debugger::Constants::OPENED_BY_DEBUGGER, true);
|
|
|
|
|
d->editor->setProperty(Debugger::Constants::OPENED_WITH_DISASSEMBLY, true);
|
2010-06-22 14:06:42 +02:00
|
|
|
d->configureMimeType();
|
2010-11-24 18:36:17 +01:00
|
|
|
|
|
|
|
|
BaseTextEditor *baseTextEdit =
|
|
|
|
|
qobject_cast<BaseTextEditor *>(d->editor->widget());
|
|
|
|
|
if (baseTextEdit)
|
|
|
|
|
baseTextEdit->setRequestMarkEnabled(true);
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2010-09-14 15:15:57 +02:00
|
|
|
editorManager->activateEditor(d->editor);
|
2009-08-14 13:04:05 +02:00
|
|
|
|
2010-11-24 15:55:09 +01:00
|
|
|
QPlainTextEdit *plainTextEdit =
|
|
|
|
|
qobject_cast<QPlainTextEdit *>(d->editor->widget());
|
|
|
|
|
QTC_ASSERT(plainTextEdit, return);
|
|
|
|
|
|
|
|
|
|
QString str;
|
|
|
|
|
for (int i = 0, n = contents.size(); i != n; ++i) {
|
|
|
|
|
const DisassemblerLine &dl = contents.at(i);
|
|
|
|
|
if (dl.address) {
|
|
|
|
|
str += QString("0x");
|
|
|
|
|
str += QString::number(dl.address, 16);
|
|
|
|
|
str += " ";
|
|
|
|
|
}
|
|
|
|
|
str += dl.data;
|
|
|
|
|
str += "\n";
|
2010-06-22 14:06:42 +02:00
|
|
|
}
|
2010-11-24 15:55:09 +01:00
|
|
|
plainTextEdit->setPlainText(str);
|
|
|
|
|
plainTextEdit->setReadOnly(true);
|
2009-08-14 13:04:05 +02:00
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
d->cache.insert(frameKey(d->location), contents);
|
|
|
|
|
d->editor->setDisplayName(_("Disassembler (%1)")
|
|
|
|
|
.arg(d->location.functionName()));
|
2010-11-25 11:51:09 +01:00
|
|
|
|
|
|
|
|
updateBreakpointMarkers();
|
|
|
|
|
updateLocationMarker();
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgent::updateLocationMarker()
|
2010-11-25 11:51:09 +01:00
|
|
|
{
|
|
|
|
|
QTC_ASSERT(d->editor, return);
|
2009-08-14 13:04:05 +02:00
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
const DisassemblerLines &contents = d->cache.value(frameKey(d->location));
|
|
|
|
|
int lineNumber = contents.lineForAddress(d->location.address());
|
2009-08-14 13:04:05 +02:00
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
if (d->location.needsMarker()) {
|
2010-11-25 11:51:09 +01:00
|
|
|
d->editor->markableInterface()->removeMark(d->locationMark);
|
|
|
|
|
if (lineNumber)
|
|
|
|
|
d->editor->markableInterface()->addMark(d->locationMark, lineNumber);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QPlainTextEdit *plainTextEdit =
|
|
|
|
|
qobject_cast<QPlainTextEdit *>(d->editor->widget());
|
|
|
|
|
QTC_ASSERT(plainTextEdit, return);
|
2010-11-24 15:55:09 +01:00
|
|
|
QTextCursor tc = plainTextEdit->textCursor();
|
|
|
|
|
QTextBlock block = tc.document()->findBlockByNumber(lineNumber - 1);
|
|
|
|
|
tc.setPosition(block.position());
|
|
|
|
|
plainTextEdit->setTextCursor(tc);
|
2009-10-02 11:45:19 +02:00
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
void DisassemblerAgent::updateBreakpointMarkers()
|
2010-11-25 11:51:09 +01:00
|
|
|
{
|
|
|
|
|
if (!d->editor)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
BreakHandler *handler = breakHandler();
|
|
|
|
|
BreakpointIds ids = handler->engineBreakpointIds(d->engine);
|
|
|
|
|
if (ids.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
const DisassemblerLines &contents = d->cache.value(frameKey(d->location));
|
2010-11-25 11:51:09 +01:00
|
|
|
|
|
|
|
|
foreach (TextEditor::ITextMark *marker, d->breakpointMarks)
|
|
|
|
|
d->editor->markableInterface()->removeMark(marker);
|
|
|
|
|
d->breakpointMarks.clear();
|
|
|
|
|
foreach (BreakpointId id, ids) {
|
|
|
|
|
const quint64 address = handler->address(id);
|
|
|
|
|
if (!address)
|
|
|
|
|
continue;
|
|
|
|
|
const int lineNumber = contents.lineForAddress(address);
|
2010-11-25 14:33:13 +01:00
|
|
|
if (!lineNumber)
|
|
|
|
|
continue;
|
2010-11-25 11:51:09 +01:00
|
|
|
BreakpointMarker2 *marker = new BreakpointMarker2(handler->icon(id));
|
|
|
|
|
d->breakpointMarks.append(marker);
|
|
|
|
|
d->editor->markableInterface()->addMark(marker, lineNumber);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-14 12:29:32 +01:00
|
|
|
quint64 DisassemblerAgent::address() const
|
2009-08-14 13:04:05 +02:00
|
|
|
{
|
2010-12-16 19:06:33 +01:00
|
|
|
return d->location.address();
|
2009-08-14 13:04:05 +02:00
|
|
|
}
|
|
|
|
|
|
2010-09-21 14:26:45 +02:00
|
|
|
// Return address of an assembly line "0x0dfd bla"
|
2010-12-14 12:29:32 +01:00
|
|
|
quint64 DisassemblerAgent::addressFromDisassemblyLine(const QString &line)
|
2010-11-24 15:55:09 +01:00
|
|
|
{
|
|
|
|
|
return DisassemblerLine(line).address;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-16 19:06:33 +01:00
|
|
|
void DisassemblerAgent::setTryMixed(bool on)
|
|
|
|
|
{
|
|
|
|
|
d->tryMixed = on;
|
|
|
|
|
}
|
|
|
|
|
|
2009-08-12 10:51:25 +02:00
|
|
|
} // namespace Internal
|
|
|
|
|
} // namespace Debugger
|