Files
qt-creator/src/plugins/debugger/stackhandler.cpp
hjk 9894c6eaf0 Debugger: Convert to Tr::tr
Change-Id: I5d2475c790851c68f9997ac6af72b5eaca58482d
Reviewed-by: Eike Ziller <eike.ziller@qt.io>
2022-07-08 12:32:38 +00:00

523 lines
17 KiB
C++

/****************************************************************************
**
** Copyright (C) 2016 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** 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 The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3 as published by the Free Software
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
** included in the packaging of this file. Please review the following
** information to ensure the GNU General Public License requirements will
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
**
****************************************************************************/
#include "stackhandler.h"
#include "debuggeractions.h"
#include "debuggercore.h"
#include "debuggerdialogs.h"
#include "debuggerengine.h"
#include "debuggericons.h"
#include "debuggerprotocol.h"
#include "debuggertr.h"
#include "memoryagent.h"
#include "simplifytype.h"
#include <coreplugin/icore.h>
#include <coreplugin/messagebox.h>
#include <utils/basetreeview.h>
#include <utils/fileutils.h>
#include <utils/qtcassert.h>
#include <utils/stringutils.h>
#include <QContextMenuEvent>
#include <QDebug>
#include <QDir>
#include <QFile>
#include <QFileDialog>
#include <QInputDialog>
#include <QMenu>
#include <QTextStream>
#include <QTimer>
using namespace Utils;
namespace Debugger::Internal {
/*!
\class Debugger::Internal::StackHandler
\brief The StackHandler class provides a model to represent the stack in a
QTreeView.
*/
StackHandler::StackHandler(DebuggerEngine *engine)
: m_engine(engine)
{
setObjectName("StackModel");
setHeader({Tr::tr("Level"), Tr::tr("Function"), Tr::tr("File"), Tr::tr("Line"), Tr::tr("Address") });
connect(debuggerSettings()->expandStack.action(), &QAction::triggered,
this, &StackHandler::reloadFullStack);
connect(debuggerSettings()->maximalStackDepth.action(), &QAction::triggered,
this, &StackHandler::reloadFullStack);
// For now there's always only "the" current thread.
rootItem()->appendChild(new ThreadDummyItem);
}
StackHandler::~StackHandler() = default;
QVariant SpecialStackItem::data(int column, int role) const
{
if (role == Qt::DisplayRole && column == StackLevelColumn)
return Tr::tr("...");
if (role == Qt::DisplayRole && column == StackFunctionNameColumn)
return Tr::tr("<More>");
if (role == Qt::DecorationRole && column == StackLevelColumn)
return Icons::EMPTY.icon();
return QVariant();
}
QVariant StackFrameItem::data(int column, int role) const
{
if (role == Qt::DisplayRole) {
switch (column) {
case StackLevelColumn:
return row >= 0 ? QString::number(row + 1) : QString();
case StackFunctionNameColumn:
return simplifyType(frame.function);
case StackFileNameColumn:
return frame.file.isEmpty() ? frame.module : frame.file.fileName();
case StackLineNumberColumn:
return frame.line > 0 ? QVariant(frame.line) : QVariant();
case StackAddressColumn:
if (frame.address)
return QString("0x%1").arg(frame.address, 0, 16);
return QString();
}
return QVariant();
}
if (role == Qt::DecorationRole && column == StackLevelColumn)
return handler->iconForRow(row);
if (role == Qt::ToolTipRole && debuggerSettings()->useToolTipsInStackView.value())
return frame.toToolTip();
return QVariant();
}
Qt::ItemFlags StackFrameItem::flags(int column) const
{
const bool isValid = frame.isUsable() || handler->operatesByInstruction();
return isValid && handler->isContentsValid()
? TreeItem::flags(column) : Qt::ItemFlags();
}
QIcon StackHandler::iconForRow(int row) const
{
// Return icon that indicates whether this is the active stack frame
return (m_contentsValid && row == m_currentIndex)
? Icons::LOCATION.icon() : Icons::EMPTY.icon();
}
bool StackHandler::setData(const QModelIndex &idx, const QVariant &data, int role)
{
if (role == BaseTreeView::ItemActivatedRole || role == BaseTreeView::ItemClickedRole) {
m_engine->activateFrame(idx.row());
return true;
}
if (role == BaseTreeView::ItemViewEventRole) {
ItemViewEvent ev = data.value<ItemViewEvent>();
if (ev.type() == QEvent::ContextMenu)
return contextMenuEvent(ev);
}
return false;
}
ThreadDummyItem *StackHandler::dummyThreadItem() const
{
QTC_ASSERT(rootItem()->childCount() == 1, return nullptr);
return rootItem()->childAt(0);
}
StackFrame StackHandler::currentFrame() const
{
if (m_currentIndex == -1)
return {};
QTC_ASSERT(m_currentIndex >= 0, return {});
return frameAt(m_currentIndex);
}
StackFrame StackHandler::frameAt(int index) const
{
auto threadItem = dummyThreadItem();
QTC_ASSERT(threadItem, return {});
StackFrameItem *frameItem = threadItem->childAt(index);
QTC_ASSERT(frameItem, return {});
return frameItem->frame;
}
int StackHandler::stackSize() const
{
return stackRowCount() - m_canExpand;
}
quint64 StackHandler::topAddress() const
{
QTC_ASSERT(stackRowCount() > 0, return 0);
return frameAt(0).address;
}
void StackHandler::setCurrentIndex(int level)
{
if (level == -1 || level == m_currentIndex)
return;
// Emit changed for previous frame
QModelIndex i = index(m_currentIndex, 0);
emit dataChanged(i, i);
m_currentIndex = level;
emit currentIndexChanged();
// Emit changed for new frame
i = index(m_currentIndex, 0);
emit dataChanged(i, i);
}
void StackHandler::removeAll()
{
auto threadItem = dummyThreadItem();
QTC_ASSERT(threadItem, return);
threadItem->removeChildren();
setCurrentIndex(-1);
}
bool StackHandler::operatesByInstruction() const
{
return m_engine->operatesByInstruction();
}
void StackHandler::setFrames(const StackFrames &frames, bool canExpand)
{
auto threadItem = dummyThreadItem();
QTC_ASSERT(threadItem, return);
threadItem->removeChildren();
m_contentsValid = true;
m_canExpand = canExpand;
int row = 0;
for (const StackFrame &frame : frames)
threadItem->appendChild(new StackFrameItem(this, frame, row++));
if (canExpand)
threadItem->appendChild(new SpecialStackItem(this));
if (frames.isEmpty())
m_currentIndex = -1;
else
setCurrentIndex(0);
emit stackChanged();
}
void StackHandler::setFramesAndCurrentIndex(const GdbMi &frames, bool isFull)
{
int targetFrame = -1;
StackFrames stackFrames;
const int n = frames.childCount();
for (int i = 0; i != n; ++i) {
stackFrames.append(StackFrame::parseFrame(frames.childAt(i), m_engine->runParameters()));
const StackFrame &frame = stackFrames.back();
// Initialize top frame to the first valid frame.
const bool isValid = frame.isUsable() && !frame.function.isEmpty();
if (isValid && targetFrame == -1)
targetFrame = i;
}
bool canExpand = !isFull && (n >= debuggerSettings()->maximalStackDepth.value());
debuggerSettings()->expandStack.setEnabled(canExpand);
setFrames(stackFrames, canExpand);
// We can't jump to any file if we don't have any frames.
if (stackFrames.isEmpty())
return;
// targetFrame contains the top most frame for which we have source
// information. That's typically the frame we'd like to jump to, with
// a few exceptions:
// Always jump to frame #0 when stepping by instruction.
if (m_engine->operatesByInstruction())
targetFrame = 0;
// If there is no frame with source, jump to frame #0.
if (targetFrame == -1)
targetFrame = 0;
setCurrentIndex(targetFrame);
}
void StackHandler::prependFrames(const StackFrames &frames)
{
if (frames.isEmpty())
return;
auto threadItem = dummyThreadItem();
QTC_ASSERT(threadItem, return);
const int count = frames.size();
for (int i = count - 1; i >= 0; --i)
threadItem->prependChild(new StackFrameItem(this, frames.at(i)));
if (m_currentIndex >= 0)
setCurrentIndex(m_currentIndex + count);
emit stackChanged();
}
bool StackHandler::isSpecialFrame(int index) const
{
return m_canExpand && index + 1 == stackRowCount();
}
int StackHandler::firstUsableIndex() const
{
if (!m_engine->operatesByInstruction()) {
for (int i = 0, n = stackRowCount(); i != n; ++i)
if (frameAt(i).isUsable())
return i;
}
return 0;
}
void StackHandler::scheduleResetLocation()
{
m_contentsValid = false;
}
void StackHandler::resetLocation()
{
emit layoutChanged();
}
int StackHandler::stackRowCount() const
{
// Only one "thread" for now.
auto threadItem = dummyThreadItem();
QTC_ASSERT(threadItem, return 0);
return threadItem->childCount();
}
// Input a function to be disassembled. Accept CDB syntax
// 'Module!function' for module specification
static StackFrame inputFunctionForDisassembly()
{
StackFrame frame;
QInputDialog dialog;
dialog.setInputMode(QInputDialog::TextInput);
dialog.setLabelText(Tr::tr("Function:"));
dialog.setWindowTitle(Tr::tr("Disassemble Function"));
if (dialog.exec() != QDialog::Accepted)
return frame;
const QString function = dialog.textValue();
if (function.isEmpty())
return frame;
const int bangPos = function.indexOf('!');
if (bangPos != -1) {
frame.module = function.left(bangPos);
frame.function = function.mid(bangPos + 1);
} else {
frame.function = function;
}
frame.line = 42; // trick gdb into mixed mode.
return frame;
}
template<typename Functor>
void forEachCell(Functor f, QAbstractItemModel *model, const QModelIndex &idx)
{
f(idx);
for (int i = 0, n = model->rowCount(idx); i < n; ++i)
forEachCell(f, model, model->index(i, 0, idx));
}
static QString selectedText(QWidget *widget, bool useAll)
{
QAbstractItemView *view = qobject_cast<QAbstractItemView *>(widget);
QTC_ASSERT(view, return {});
QAbstractItemModel *model = view->model();
QTC_ASSERT(model, return {});
const int ncols = model->columnCount(QModelIndex());
QVector<int> largestColumnWidths(ncols, 0);
QSet<QModelIndex> selected;
if (QItemSelectionModel *selection = view->selectionModel()) {
const QModelIndexList list = selection->selectedIndexes();
selected = QSet<QModelIndex>(list.begin(), list.end());
}
if (selected.isEmpty())
useAll = true;
// First, find the widths of the largest columns,
// so that we can print them out nicely aligned.
forEachCell([ncols, model, &largestColumnWidths, selected, useAll](const QModelIndex &idx) {
if (useAll || selected.contains(idx)) {
for (int j = 0; j < ncols; ++j) {
const QModelIndex sibling = model->sibling(idx.row(), j, idx);
const int columnWidth = model->data(sibling, Qt::DisplayRole).toString().size();
if (columnWidth > largestColumnWidths.at(j))
largestColumnWidths[j] = columnWidth;
}
}
}, model, QModelIndex());
QString str;
forEachCell([ncols, model, largestColumnWidths, &str, selected, useAll](const QModelIndex &idx) {
if (useAll || selected.contains(idx)) {
for (int j = 0; j != ncols; ++j) {
const QModelIndex sibling = model->sibling(idx.row(), j, idx);
const QString columnEntry = model->data(sibling, Qt::DisplayRole).toString();
str += columnEntry;
const int difference = largestColumnWidths.at(j) - columnEntry.size();
// Add one extra space between columns.
str += QString(qMax(difference, 0) + 1, QChar(' '));
}
str += '\n';
}
}, model, QModelIndex());
return str;
}
// Write stack frames as task file for displaying it in the build issues pane.
void StackHandler::saveTaskFile()
{
QFile file;
QFileDialog fileDialog(Core::ICore::dialogParent());
fileDialog.setAcceptMode(QFileDialog::AcceptSave);
fileDialog.selectFile(QDir::currentPath() + "/stack.tasks");
while (!file.isOpen()) {
if (fileDialog.exec() != QDialog::Accepted)
return;
const QString fileName = fileDialog.selectedFiles().constFirst();
file.setFileName(fileName);
if (!file.open(QIODevice::WriteOnly | QIODevice::Text)) {
QString msg = Tr::tr("Cannot open \"%1\": %2")
.arg(QDir::toNativeSeparators(fileName), file.errorString());
Core::AsynchronousMessageBox::warning(Tr::tr("Cannot Open Task File"), msg);
}
}
QTextStream str(&file);
forItemsAtLevel<2>([&str](StackFrameItem *item) {
const StackFrame &frame = item->frame;
if (frame.isUsable())
str << frame.file << '\t' << frame.line << "\tstack\tFrame #" << frame.level << '\n';
});
}
bool StackHandler::contextMenuEvent(const ItemViewEvent &ev)
{
auto menu = new QMenu;
const int row = ev.sourceModelIndex().row();
StackFrame frame;
if (row >= 0 && row < stackSize())
frame = frameAt(row);
const quint64 address = frame.address;
menu->addAction(debuggerSettings()->expandStack.action());
addAction(this, menu, Tr::tr("Copy Contents to Clipboard"), true, [ev] {
setClipboardAndSelection(selectedText(ev.view(), true));
});
addAction(this, menu, Tr::tr("Copy Selection to Clipboard"), true, [ev] {
setClipboardAndSelection(selectedText(ev.view(), false));
});
addAction(this, menu, Tr::tr("Save as Task File..."), true, [this] { saveTaskFile(); });
if (m_engine->hasCapability(CreateFullBacktraceCapability))
menu->addAction(debuggerSettings()->createFullBacktrace.action());
if (m_engine->hasCapability(AdditionalQmlStackCapability))
addAction(this, menu, Tr::tr("Load QML Stack"), true, [this] { m_engine->loadAdditionalQmlStack(); });
if (m_engine->hasCapability(ShowMemoryCapability))
addAction(this, menu, Tr::tr("Open Memory Editor at 0x%1").arg(address, 0, 16),
Tr::tr("Open Memory Editor"),
address,
[this, row, frame, address] {
MemoryViewSetupData data;
data.startAddress = address;
data.title = Tr::tr("Memory at Frame #%1 (%2) 0x%3")
.arg(row).arg(frame.function).arg(address, 0, 16);
data.markup.push_back(MemoryMarkup(address, 1, QColor(Qt::blue).lighter(),
Tr::tr("Frame #%1 (%2)").arg(row).arg(frame.function)));
m_engine->openMemoryView(data);
});
if (m_engine->hasCapability(DisassemblerCapability)) {
addAction(this, menu, Tr::tr("Open Disassembler at 0x%1").arg(address, 0, 16),
Tr::tr("Open Disassembler"),
address,
[this, frame] { m_engine->openDisassemblerView(frame); });
addAction(this, menu, Tr::tr("Open Disassembler at Address..."), true,
[this, address] {
AddressDialog dialog;
if (address)
dialog.setAddress(address);
if (dialog.exec() == QDialog::Accepted)
m_engine->openDisassemblerView(Location(dialog.address()));
});
addAction(this, menu, Tr::tr("Disassemble Function..."), true,
[this] {
const StackFrame frame = inputFunctionForDisassembly();
if (!frame.function.isEmpty())
m_engine->openDisassemblerView(Location(frame));
});
}
if (m_engine->hasCapability(ShowModuleSymbolsCapability)) {
addAction(this, menu, Tr::tr("Try to Load Unknown Symbols"), true,
[this] { m_engine->loadSymbolsForStack(); });
}
menu->addSeparator();
menu->addAction(debuggerSettings()->useToolTipsInStackView.action());
menu->addAction(debuggerSettings()->settingsDialog.action());
connect(menu, &QMenu::aboutToHide, menu, &QObject::deleteLater);
menu->popup(ev.globalPos());
return true;
}
void StackHandler::reloadFullStack()
{
m_engine->reloadFullStack();
}
} // Debugger::Internal