forked from qt-creator/qt-creator
Turns out setting the column resize mode on the view itself is counter-productive as it takes away the ability to manually re-adjust. So set the mode only temporarily to get the preferred width than switch back to manually resize mode and use the hint to set some initial size. Also use the length of the header label as absolute minimum. Change-Id: Ic17e31334b23ce6d541f9459cd22be65145046d3 Reviewed-by: Eike Ziller <eike.ziller@digia.com>
277 lines
9.6 KiB
C++
277 lines
9.6 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
|
|
** Contact: http://www.qt-project.org/legal
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** 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
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#include "stackwindow.h"
|
|
#include "stackhandler.h"
|
|
|
|
#include "debuggeractions.h"
|
|
#include "debuggercore.h"
|
|
#include "debuggerengine.h"
|
|
#include "debuggerdialogs.h"
|
|
#include "memoryagent.h"
|
|
|
|
|
|
#include <utils/savedaction.h>
|
|
|
|
#include <QDebug>
|
|
#include <QTextStream>
|
|
#include <QFile>
|
|
#include <QDir>
|
|
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include <QContextMenuEvent>
|
|
#include <QInputDialog>
|
|
#include <QFileDialog>
|
|
#include <QMessageBox>
|
|
#include <QMenu>
|
|
|
|
namespace Debugger {
|
|
namespace Internal {
|
|
|
|
static DebuggerEngine *currentEngine()
|
|
{
|
|
return debuggerCore()->currentEngine();
|
|
}
|
|
|
|
StackTreeView::StackTreeView(QWidget *parent)
|
|
: BaseTreeView(parent)
|
|
{
|
|
setWindowTitle(tr("Stack"));
|
|
|
|
connect(debuggerCore()->action(UseAddressInStackView), SIGNAL(toggled(bool)),
|
|
SLOT(showAddressColumn(bool)));
|
|
connect(debuggerCore()->action(ExpandStack), SIGNAL(triggered()),
|
|
SLOT(reloadFullStack()));
|
|
connect(debuggerCore()->action(MaximalStackDepth), SIGNAL(triggered()),
|
|
SLOT(reloadFullStack()));
|
|
showAddressColumn(false);
|
|
}
|
|
|
|
void StackTreeView::showAddressColumn(bool on)
|
|
{
|
|
setColumnHidden(4, !on);
|
|
}
|
|
|
|
void StackTreeView::rowActivated(const QModelIndex &index)
|
|
{
|
|
currentEngine()->activateFrame(index.row());
|
|
}
|
|
|
|
void StackTreeView::setModel(QAbstractItemModel *model)
|
|
{
|
|
BaseTreeView::setModel(model);
|
|
resizeColumnToContents(0);
|
|
resizeColumnToContents(3);
|
|
showAddressColumn(debuggerCore()->action(UseAddressInStackView)->isChecked());
|
|
}
|
|
|
|
// Input a function to be disassembled. Accept CDB syntax
|
|
// 'Module!function' for module specification
|
|
|
|
static inline StackFrame inputFunctionForDisassembly()
|
|
{
|
|
StackFrame frame;
|
|
QInputDialog dialog;
|
|
dialog.setInputMode(QInputDialog::TextInput);
|
|
dialog.setLabelText(StackTreeView::tr("Function:"));
|
|
dialog.setWindowTitle(StackTreeView::tr("Disassemble Function"));
|
|
dialog.setWindowFlags(dialog.windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
if (dialog.exec() != QDialog::Accepted)
|
|
return frame;
|
|
const QString function = dialog.textValue();
|
|
if (function.isEmpty())
|
|
return frame;
|
|
const int bangPos = function.indexOf(QLatin1Char('!'));
|
|
if (bangPos != -1) {
|
|
frame.from = function.left(bangPos);
|
|
frame.function = function.mid(bangPos + 1);
|
|
} else {
|
|
frame.function = function;
|
|
}
|
|
frame.line = 42; // trick gdb into mixed mode.
|
|
return frame;
|
|
}
|
|
|
|
// Write stack frames as task file for displaying it in the build issues pane.
|
|
void saveTaskFile(QWidget *parent, const StackHandler *sh)
|
|
{
|
|
QFile file;
|
|
QFileDialog fileDialog(parent);
|
|
fileDialog.setAcceptMode(QFileDialog::AcceptSave);
|
|
fileDialog.selectFile(QDir::currentPath() + QLatin1String("/stack.tasks"));
|
|
while (!file.isOpen()) {
|
|
if (fileDialog.exec() != QDialog::Accepted)
|
|
return;
|
|
const QString fileName = fileDialog.selectedFiles().front();
|
|
file.setFileName(fileName);
|
|
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
|
|
QMessageBox::warning(parent, StackTreeView::tr("Cannot Open Task File"),
|
|
StackTreeView::tr("Cannot open \"%1\": %2").arg(QDir::toNativeSeparators(fileName), file.errorString()));
|
|
}
|
|
}
|
|
|
|
QTextStream str(&file);
|
|
foreach (const StackFrame &frame, sh->frames()) {
|
|
if (frame.isUsable())
|
|
str << frame.file << '\t' << frame.line << "\tstack\tFrame #" << frame.level << '\n';
|
|
}
|
|
}
|
|
|
|
void StackTreeView::contextMenuEvent(QContextMenuEvent *ev)
|
|
{
|
|
DebuggerEngine *engine = currentEngine();
|
|
StackHandler *handler = engine->stackHandler();
|
|
const QModelIndex index = indexAt(ev->pos());
|
|
const int row = index.row();
|
|
StackFrame frame;
|
|
if (row >= 0 && row < handler->stackSize())
|
|
frame = handler->frameAt(row);
|
|
const quint64 address = frame.address;
|
|
|
|
QMenu menu;
|
|
menu.addAction(debuggerCore()->action(ExpandStack));
|
|
|
|
QAction *actCopyContents = menu.addAction(tr("Copy Contents to Clipboard"));
|
|
actCopyContents->setEnabled(model() != 0);
|
|
|
|
QAction *actSaveTaskFile = menu.addAction(tr("Save as Task File..."));
|
|
actSaveTaskFile->setEnabled(model() != 0);
|
|
|
|
if (engine->hasCapability(CreateFullBacktraceCapability))
|
|
menu.addAction(debuggerCore()->action(CreateFullBacktrace));
|
|
|
|
QAction *additionalQmlStackAction = 0;
|
|
if (engine->hasCapability(AdditionalQmlStackCapability))
|
|
additionalQmlStackAction = menu.addAction(tr("Load QML Stack"));
|
|
|
|
QAction *actShowMemory = 0;
|
|
if (engine->hasCapability(ShowMemoryCapability)) {
|
|
actShowMemory = menu.addAction(QString());
|
|
if (address == 0) {
|
|
actShowMemory->setText(tr("Open Memory Editor"));
|
|
actShowMemory->setEnabled(false);
|
|
} else {
|
|
actShowMemory->setText(tr("Open Memory Editor at 0x%1").arg(address, 0, 16));
|
|
actShowMemory->setEnabled(engine->hasCapability(ShowMemoryCapability));
|
|
}
|
|
}
|
|
|
|
QAction *actShowDisassemblerAt = 0;
|
|
QAction *actShowDisassemblerAtAddress = 0;
|
|
QAction *actShowDisassemblerAtFunction = 0;
|
|
|
|
if (engine->hasCapability(DisassemblerCapability)) {
|
|
actShowDisassemblerAt = menu.addAction(QString());
|
|
actShowDisassemblerAtAddress = menu.addAction(tr("Open Disassembler at Address..."));
|
|
actShowDisassemblerAtFunction = menu.addAction(tr("Disassemble Function..."));
|
|
if (address == 0) {
|
|
actShowDisassemblerAt->setText(tr("Open Disassembler"));
|
|
actShowDisassemblerAt->setEnabled(false);
|
|
} else {
|
|
actShowDisassemblerAt->setText(tr("Open Disassembler at 0x%1").arg(address, 0, 16));
|
|
}
|
|
}
|
|
|
|
|
|
QAction *actLoadSymbols = 0;
|
|
if (engine->hasCapability(ShowModuleSymbolsCapability))
|
|
actLoadSymbols = menu.addAction(tr("Try to Load Unknown Symbols"));
|
|
|
|
#if 0 // @TODO: not implemented
|
|
menu.addAction(debuggerCore()->action(UseToolTipsInStackView));
|
|
#endif
|
|
if (engine->hasCapability(MemoryAddressCapability))
|
|
menu.addAction(debuggerCore()->action(UseAddressInStackView));
|
|
|
|
menu.addSeparator();
|
|
menu.addAction(debuggerCore()->action(UseToolTipsInStackView));
|
|
menu.addSeparator();
|
|
|
|
addBaseContextActions(&menu);
|
|
|
|
QAction *act = menu.exec(ev->globalPos());
|
|
if (!act)
|
|
return;
|
|
|
|
if (act == actCopyContents) {
|
|
copyContentsToClipboard();
|
|
} else if (act == actShowMemory) {
|
|
const QString title = tr("Memory at Frame #%1 (%2) 0x%3").
|
|
arg(row).arg(frame.function).arg(address, 0, 16);
|
|
QList<MemoryMarkup> ml;
|
|
ml.push_back(MemoryMarkup(address, 1, QColor(Qt::blue).lighter(),
|
|
tr("Frame #%1 (%2)").arg(row).arg(frame.function)));
|
|
engine->openMemoryView(address, 0, ml, QPoint(), title);
|
|
} else if (act == actShowDisassemblerAtAddress) {
|
|
AddressDialog dialog;
|
|
if (address)
|
|
dialog.setAddress(address);
|
|
if (dialog.exec() == QDialog::Accepted)
|
|
currentEngine()->openDisassemblerView(Location(dialog.address()));
|
|
} else if (act == actShowDisassemblerAtFunction) {
|
|
const StackFrame frame = inputFunctionForDisassembly();
|
|
if (!frame.function.isEmpty())
|
|
currentEngine()->openDisassemblerView(Location(frame));
|
|
} else if (act == actShowDisassemblerAt)
|
|
engine->openDisassemblerView(frame);
|
|
else if (act == actLoadSymbols)
|
|
engine->loadSymbolsForStack();
|
|
else if (act == actSaveTaskFile)
|
|
saveTaskFile(this, handler);
|
|
else if (act == additionalQmlStackAction)
|
|
engine->loadAdditionalQmlStack();
|
|
}
|
|
|
|
void StackTreeView::copyContentsToClipboard()
|
|
{
|
|
QString str;
|
|
int n = model()->rowCount();
|
|
int m = model()->columnCount();
|
|
for (int i = 0; i != n; ++i) {
|
|
for (int j = 0; j != m; ++j) {
|
|
QModelIndex index = model()->index(i, j);
|
|
str += model()->data(index).toString();
|
|
str += QLatin1Char('\t');
|
|
}
|
|
str += QLatin1Char('\n');
|
|
}
|
|
QClipboard *clipboard = QApplication::clipboard();
|
|
clipboard->setText(str, QClipboard::Selection);
|
|
clipboard->setText(str, QClipboard::Clipboard);
|
|
}
|
|
|
|
void StackTreeView::reloadFullStack()
|
|
{
|
|
currentEngine()->reloadFullStack();
|
|
}
|
|
|
|
} // namespace Internal
|
|
} // namespace Debugger
|