mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-15 10:39:13 +01:00
fef1b84f0a
QStringLiterals generate a buffer so that during runtime there's very little cost to constructing a QString. However, this also means that duplicated strings cannot be optimized out into a single entry that gets referenced everywhere, taking up space in the binary. Rather than use QStringLiteral(""), we can just use QString{} (the default constructor) to signify the empty string. This gets rid of an unnecessary string buffer from being created, saving a tiny bit of space. While we're at it, we can just use the character overloads of particular functions when they're available instead of using a QString overload. The characters in this case are Latin-1 to begin with, so we can just specify the characters as QLatin1Char instances to use those overloads. These will automatically convert to QChar if needed, so this is safe.
602 lines
16 KiB
C++
602 lines
16 KiB
C++
// Copyright 2018 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include "DolphinQt/Debugger/CodeViewWidget.h"
|
|
|
|
#include <algorithm>
|
|
#include <cmath>
|
|
|
|
#include <QApplication>
|
|
#include <QClipboard>
|
|
#include <QHeaderView>
|
|
#include <QInputDialog>
|
|
#include <QKeyEvent>
|
|
#include <QMenu>
|
|
#include <QMouseEvent>
|
|
#include <QResizeEvent>
|
|
#include <QScrollBar>
|
|
#include <QTableWidgetItem>
|
|
#include <QWheelEvent>
|
|
|
|
#include "Common/StringUtil.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/Debugger/PPCDebugInterface.h"
|
|
#include "Core/PowerPC/MMU.h"
|
|
#include "Core/PowerPC/PPCAnalyst.h"
|
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
|
#include "Core/PowerPC/PowerPC.h"
|
|
#include "DolphinQt/Debugger/PatchInstructionDialog.h"
|
|
#include "DolphinQt/Resources.h"
|
|
#include "DolphinQt/Settings.h"
|
|
|
|
// "Most mouse types work in steps of 15 degrees, in which case the delta value is a multiple of
|
|
// 120; i.e., 120 units * 1/8 = 15 degrees." (http://doc.qt.io/qt-5/qwheelevent.html#angleDelta)
|
|
constexpr double SCROLL_FRACTION_DEGREES = 15.;
|
|
|
|
constexpr size_t VALID_BRANCH_LENGTH = 10;
|
|
|
|
CodeViewWidget::CodeViewWidget()
|
|
{
|
|
setColumnCount(5);
|
|
setShowGrid(false);
|
|
setContextMenuPolicy(Qt::CustomContextMenu);
|
|
setSelectionMode(QAbstractItemView::SingleSelection);
|
|
setSelectionBehavior(QAbstractItemView::SelectRows);
|
|
|
|
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
setHorizontalScrollMode(QAbstractItemView::ScrollPerPixel);
|
|
|
|
verticalHeader()->hide();
|
|
horizontalHeader()->hide();
|
|
|
|
setFont(Settings::Instance().GetDebugFont());
|
|
|
|
FontBasedSizing();
|
|
|
|
connect(this, &CodeViewWidget::customContextMenuRequested, this, &CodeViewWidget::OnContextMenu);
|
|
connect(this, &CodeViewWidget::itemSelectionChanged, this, &CodeViewWidget::OnSelectionChanged);
|
|
connect(&Settings::Instance(), &Settings::DebugFontChanged, this, &QWidget::setFont);
|
|
connect(&Settings::Instance(), &Settings::DebugFontChanged, this,
|
|
&CodeViewWidget::FontBasedSizing);
|
|
|
|
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this, [this] {
|
|
m_address = PC;
|
|
Update();
|
|
});
|
|
|
|
connect(&Settings::Instance(), &Settings::ThemeChanged, this, &CodeViewWidget::Update);
|
|
}
|
|
|
|
static u32 GetBranchFromAddress(u32 addr)
|
|
{
|
|
std::string disasm = PowerPC::debug_interface.Disassemble(addr);
|
|
size_t pos = disasm.find("->0x");
|
|
|
|
if (pos == std::string::npos)
|
|
return 0;
|
|
|
|
std::string hex = disasm.substr(pos + 2);
|
|
return std::stoul(hex, nullptr, 16);
|
|
}
|
|
|
|
void CodeViewWidget::FontBasedSizing()
|
|
{
|
|
const QFontMetrics fm(Settings::Instance().GetDebugFont());
|
|
const int rowh = fm.height() + 1;
|
|
verticalHeader()->setMaximumSectionSize(rowh);
|
|
horizontalHeader()->setMinimumSectionSize(rowh + 5);
|
|
setColumnWidth(0, rowh + 5);
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::Update()
|
|
{
|
|
if (!isVisible())
|
|
return;
|
|
|
|
if (m_updating)
|
|
return;
|
|
|
|
m_updating = true;
|
|
|
|
clearSelection();
|
|
if (rowCount() == 0)
|
|
setRowCount(1);
|
|
|
|
// Calculate (roughly) how many rows will fit in our table
|
|
int rows = std::round((height() / static_cast<float>(rowHeight(0))) - 0.25);
|
|
|
|
setRowCount(rows);
|
|
|
|
const QFontMetrics fm(Settings::Instance().GetDebugFont());
|
|
const int rowh = fm.height() + 1;
|
|
|
|
for (int i = 0; i < rows; i++)
|
|
setRowHeight(i, rowh);
|
|
|
|
u32 pc = PowerPC::ppcState.pc;
|
|
|
|
if (Core::GetState() != Core::State::Paused && PowerPC::debug_interface.IsBreakpoint(pc))
|
|
Core::SetState(Core::State::Paused);
|
|
|
|
const bool dark_theme = qApp->palette().color(QPalette::Base).valueF() < 0.5;
|
|
|
|
for (int i = 0; i < rowCount(); i++)
|
|
{
|
|
const u32 addr = m_address - ((rowCount() / 2) * 4) + i * 4;
|
|
const u32 color = PowerPC::debug_interface.GetColor(addr);
|
|
auto* bp_item = new QTableWidgetItem;
|
|
auto* addr_item = new QTableWidgetItem(QStringLiteral("%1").arg(addr, 8, 16, QLatin1Char('0')));
|
|
|
|
std::string disas = PowerPC::debug_interface.Disassemble(addr);
|
|
auto split = disas.find('\t');
|
|
|
|
std::string ins = (split == std::string::npos ? disas : disas.substr(0, split));
|
|
std::string param = (split == std::string::npos ? "" : disas.substr(split + 1));
|
|
std::string desc = PowerPC::debug_interface.GetDescription(addr);
|
|
|
|
// Adds whitespace and a minimum size to ins and param. Helps to prevent frequent resizing while
|
|
// scrolling.
|
|
const QString ins_formatted =
|
|
QStringLiteral("%1").arg(QString::fromStdString(ins), -7, QLatin1Char(' '));
|
|
const QString param_formatted =
|
|
QStringLiteral("%1").arg(QString::fromStdString(param), -19, QLatin1Char(' '));
|
|
const QString desc_formatted = QStringLiteral("%1 ").arg(QString::fromStdString(desc));
|
|
|
|
auto* ins_item = new QTableWidgetItem(ins_formatted);
|
|
auto* param_item = new QTableWidgetItem(param_formatted);
|
|
auto* description_item = new QTableWidgetItem(desc_formatted);
|
|
|
|
for (auto* item : {bp_item, addr_item, ins_item, param_item, description_item})
|
|
{
|
|
item->setFlags(Qt::ItemIsEnabled | Qt::ItemIsSelectable);
|
|
item->setData(Qt::UserRole, addr);
|
|
|
|
if (addr == pc && item != bp_item)
|
|
{
|
|
item->setBackground(QColor(Qt::green));
|
|
item->setForeground(QColor(Qt::black));
|
|
}
|
|
else if (color != 0xFFFFFF)
|
|
{
|
|
item->setBackground(dark_theme ? QColor(color).darker(240) : QColor(color));
|
|
}
|
|
}
|
|
|
|
// look for hex strings to decode branches
|
|
std::string hex_str;
|
|
size_t pos = param.find("0x");
|
|
if (pos != std::string::npos)
|
|
{
|
|
hex_str = param.substr(pos);
|
|
}
|
|
|
|
if (hex_str.length() == VALID_BRANCH_LENGTH && desc != "---")
|
|
{
|
|
description_item->setText(tr("--> %1").arg(QString::fromStdString(
|
|
PowerPC::debug_interface.GetDescription(GetBranchFromAddress(addr)))));
|
|
param_item->setForeground(Qt::magenta);
|
|
}
|
|
|
|
if (ins == "blr")
|
|
ins_item->setForeground(dark_theme ? QColor(0xa0FFa0) : Qt::darkGreen);
|
|
|
|
if (PowerPC::debug_interface.IsBreakpoint(addr))
|
|
{
|
|
bp_item->setData(
|
|
Qt::DecorationRole,
|
|
Resources::GetScaledThemeIcon("debugger_breakpoint").pixmap(QSize(rowh - 2, rowh - 2)));
|
|
}
|
|
|
|
setItem(i, 0, bp_item);
|
|
setItem(i, 1, addr_item);
|
|
setItem(i, 2, ins_item);
|
|
setItem(i, 3, param_item);
|
|
setItem(i, 4, description_item);
|
|
|
|
if (addr == GetAddress())
|
|
{
|
|
selectRow(addr_item->row());
|
|
}
|
|
}
|
|
|
|
resizeColumnsToContents();
|
|
g_symbolDB.FillInCallers();
|
|
|
|
repaint();
|
|
m_updating = false;
|
|
}
|
|
|
|
u32 CodeViewWidget::GetAddress() const
|
|
{
|
|
return m_address;
|
|
}
|
|
|
|
void CodeViewWidget::SetAddress(u32 address, SetAddressUpdate update)
|
|
{
|
|
if (m_address == address)
|
|
return;
|
|
|
|
m_address = address;
|
|
if (update == SetAddressUpdate::WithUpdate)
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::ReplaceAddress(u32 address, ReplaceWith replace)
|
|
{
|
|
PowerPC::debug_interface.UnsetPatch(address);
|
|
PowerPC::debug_interface.SetPatch(address, replace == ReplaceWith::BLR ? 0x4e800020 : 0x60000000);
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::OnContextMenu()
|
|
{
|
|
QMenu* menu = new QMenu(this);
|
|
|
|
bool running = Core::GetState() != Core::State::Uninitialized;
|
|
|
|
const u32 addr = GetContextAddress();
|
|
|
|
bool has_symbol = g_symbolDB.GetSymbolFromAddr(addr);
|
|
|
|
auto* follow_branch_action =
|
|
menu->addAction(tr("Follow &branch"), this, &CodeViewWidget::OnFollowBranch);
|
|
|
|
menu->addSeparator();
|
|
|
|
menu->addAction(tr("&Copy address"), this, &CodeViewWidget::OnCopyAddress);
|
|
auto* copy_address_action =
|
|
menu->addAction(tr("Copy &function"), this, &CodeViewWidget::OnCopyFunction);
|
|
auto* copy_line_action =
|
|
menu->addAction(tr("Copy code &line"), this, &CodeViewWidget::OnCopyCode);
|
|
auto* copy_hex_action = menu->addAction(tr("Copy &hex"), this, &CodeViewWidget::OnCopyHex);
|
|
|
|
menu->addAction(tr("Show in &memory"), this, &CodeViewWidget::OnShowInMemory);
|
|
|
|
menu->addSeparator();
|
|
|
|
auto* symbol_rename_action =
|
|
menu->addAction(tr("&Rename symbol"), this, &CodeViewWidget::OnRenameSymbol);
|
|
auto* symbol_size_action =
|
|
menu->addAction(tr("Set symbol &size"), this, &CodeViewWidget::OnSetSymbolSize);
|
|
auto* symbol_end_action =
|
|
menu->addAction(tr("Set symbol &end address"), this, &CodeViewWidget::OnSetSymbolEndAddress);
|
|
menu->addSeparator();
|
|
|
|
menu->addAction(tr("Run &To Here"), this, &CodeViewWidget::OnRunToHere);
|
|
auto* function_action =
|
|
menu->addAction(tr("&Add function"), this, &CodeViewWidget::OnAddFunction);
|
|
auto* ppc_action = menu->addAction(tr("PPC vs Host"), this, &CodeViewWidget::OnPPCComparison);
|
|
auto* insert_blr_action = menu->addAction(tr("&Insert blr"), this, &CodeViewWidget::OnInsertBLR);
|
|
auto* insert_nop_action = menu->addAction(tr("Insert &nop"), this, &CodeViewWidget::OnInsertNOP);
|
|
auto* replace_action =
|
|
menu->addAction(tr("Re&place instruction"), this, &CodeViewWidget::OnReplaceInstruction);
|
|
auto* restore_action =
|
|
menu->addAction(tr("Restore instruction"), this, &CodeViewWidget::OnRestoreInstruction);
|
|
|
|
follow_branch_action->setEnabled(running && GetBranchFromAddress(addr));
|
|
|
|
for (auto* action : {copy_address_action, copy_line_action, copy_hex_action, function_action,
|
|
ppc_action, insert_blr_action, insert_nop_action, replace_action})
|
|
action->setEnabled(running);
|
|
|
|
for (auto* action : {symbol_rename_action, symbol_size_action, symbol_end_action})
|
|
action->setEnabled(has_symbol);
|
|
|
|
restore_action->setEnabled(running && PowerPC::debug_interface.HasEnabledPatch(addr));
|
|
|
|
menu->exec(QCursor::pos());
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::OnCopyAddress()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
QApplication::clipboard()->setText(QStringLiteral("%1").arg(addr, 8, 16, QLatin1Char('0')));
|
|
}
|
|
|
|
void CodeViewWidget::OnShowInMemory()
|
|
{
|
|
emit ShowMemory(GetContextAddress());
|
|
}
|
|
|
|
void CodeViewWidget::OnCopyCode()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
QApplication::clipboard()->setText(
|
|
QString::fromStdString(PowerPC::debug_interface.Disassemble(addr)));
|
|
}
|
|
|
|
void CodeViewWidget::OnCopyFunction()
|
|
{
|
|
const u32 address = GetContextAddress();
|
|
|
|
const Common::Symbol* symbol = g_symbolDB.GetSymbolFromAddr(address);
|
|
if (!symbol)
|
|
return;
|
|
|
|
std::string text = symbol->name + "\r\n";
|
|
// we got a function
|
|
const u32 start = symbol->address;
|
|
const u32 end = start + symbol->size;
|
|
for (u32 addr = start; addr != end; addr += 4)
|
|
{
|
|
const std::string disasm = PowerPC::debug_interface.Disassemble(addr);
|
|
text += StringFromFormat("%08x: ", addr) + disasm + "\r\n";
|
|
}
|
|
|
|
QApplication::clipboard()->setText(QString::fromStdString(text));
|
|
}
|
|
|
|
void CodeViewWidget::OnCopyHex()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
const u32 instruction = PowerPC::debug_interface.ReadInstruction(addr);
|
|
|
|
QApplication::clipboard()->setText(
|
|
QStringLiteral("%1").arg(instruction, 8, 16, QLatin1Char('0')));
|
|
}
|
|
|
|
void CodeViewWidget::OnRunToHere()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
PowerPC::debug_interface.SetBreakpoint(addr);
|
|
PowerPC::debug_interface.RunToBreakpoint();
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::OnPPCComparison()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
emit RequestPPCComparison(addr);
|
|
}
|
|
|
|
void CodeViewWidget::OnAddFunction()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
g_symbolDB.AddFunction(addr);
|
|
emit SymbolsChanged();
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::OnInsertBLR()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
ReplaceAddress(addr, ReplaceWith::BLR);
|
|
}
|
|
|
|
void CodeViewWidget::OnInsertNOP()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
ReplaceAddress(addr, ReplaceWith::NOP);
|
|
}
|
|
|
|
void CodeViewWidget::OnFollowBranch()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
u32 branch_addr = GetBranchFromAddress(addr);
|
|
|
|
if (!branch_addr)
|
|
return;
|
|
|
|
SetAddress(branch_addr, SetAddressUpdate::WithUpdate);
|
|
}
|
|
|
|
void CodeViewWidget::OnRenameSymbol()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
Common::Symbol* symbol = g_symbolDB.GetSymbolFromAddr(addr);
|
|
|
|
if (!symbol)
|
|
return;
|
|
|
|
bool good;
|
|
QString name =
|
|
QInputDialog::getText(this, tr("Rename symbol"), tr("Symbol name:"), QLineEdit::Normal,
|
|
QString::fromStdString(symbol->name), &good);
|
|
|
|
if (good && !name.isEmpty())
|
|
{
|
|
symbol->Rename(name.toStdString());
|
|
emit SymbolsChanged();
|
|
Update();
|
|
}
|
|
}
|
|
|
|
void CodeViewWidget::OnSelectionChanged()
|
|
{
|
|
if (m_address == PowerPC::ppcState.pc)
|
|
{
|
|
setStyleSheet(
|
|
QStringLiteral("QTableView::item:selected {background-color: #00FF00; color: #000000;}"));
|
|
}
|
|
else if (!styleSheet().isEmpty())
|
|
{
|
|
setStyleSheet(QString{});
|
|
}
|
|
}
|
|
|
|
void CodeViewWidget::OnSetSymbolSize()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
Common::Symbol* symbol = g_symbolDB.GetSymbolFromAddr(addr);
|
|
|
|
if (!symbol)
|
|
return;
|
|
|
|
bool good;
|
|
int size =
|
|
QInputDialog::getInt(this, tr("Rename symbol"),
|
|
tr("Set symbol size (%1):").arg(QString::fromStdString(symbol->name)),
|
|
symbol->size, 1, 0xFFFF, 1, &good);
|
|
|
|
if (!good)
|
|
return;
|
|
|
|
PPCAnalyst::ReanalyzeFunction(symbol->address, *symbol, size);
|
|
emit SymbolsChanged();
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::OnSetSymbolEndAddress()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
Common::Symbol* symbol = g_symbolDB.GetSymbolFromAddr(addr);
|
|
|
|
if (!symbol)
|
|
return;
|
|
|
|
bool good;
|
|
QString name = QInputDialog::getText(
|
|
this, tr("Set symbol end address"),
|
|
tr("Symbol (%1) end address:").arg(QString::fromStdString(symbol->name)), QLineEdit::Normal,
|
|
QStringLiteral("%1").arg(addr + symbol->size, 8, 16, QLatin1Char('0')), &good);
|
|
|
|
u32 address = name.toUInt(&good, 16);
|
|
|
|
if (!good)
|
|
return;
|
|
|
|
PPCAnalyst::ReanalyzeFunction(symbol->address, *symbol, address - symbol->address);
|
|
emit SymbolsChanged();
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::OnReplaceInstruction()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
if (!PowerPC::HostIsInstructionRAMAddress(addr))
|
|
return;
|
|
|
|
const PowerPC::TryReadInstResult read_result = PowerPC::TryReadInstruction(addr);
|
|
if (!read_result.valid)
|
|
return;
|
|
|
|
PatchInstructionDialog dialog(this, addr, PowerPC::debug_interface.ReadInstruction(addr));
|
|
|
|
if (dialog.exec() == QDialog::Accepted)
|
|
{
|
|
PowerPC::debug_interface.UnsetPatch(addr);
|
|
PowerPC::debug_interface.SetPatch(addr, dialog.GetCode());
|
|
Update();
|
|
}
|
|
}
|
|
|
|
void CodeViewWidget::OnRestoreInstruction()
|
|
{
|
|
const u32 addr = GetContextAddress();
|
|
|
|
PowerPC::debug_interface.UnsetPatch(addr);
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::resizeEvent(QResizeEvent*)
|
|
{
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::keyPressEvent(QKeyEvent* event)
|
|
{
|
|
switch (event->key())
|
|
{
|
|
case Qt::Key_Up:
|
|
m_address -= sizeof(u32);
|
|
Update();
|
|
return;
|
|
case Qt::Key_Down:
|
|
m_address += sizeof(u32);
|
|
Update();
|
|
return;
|
|
case Qt::Key_PageUp:
|
|
m_address -= rowCount() * sizeof(u32);
|
|
Update();
|
|
return;
|
|
case Qt::Key_PageDown:
|
|
m_address += rowCount() * sizeof(u32);
|
|
Update();
|
|
return;
|
|
default:
|
|
QWidget::keyPressEvent(event);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CodeViewWidget::wheelEvent(QWheelEvent* event)
|
|
{
|
|
auto delta =
|
|
-static_cast<int>(std::round((event->angleDelta().y() / (SCROLL_FRACTION_DEGREES * 8))));
|
|
|
|
if (delta == 0)
|
|
return;
|
|
|
|
m_address += delta * sizeof(u32);
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::mousePressEvent(QMouseEvent* event)
|
|
{
|
|
auto* item = itemAt(event->pos());
|
|
if (item == nullptr)
|
|
return;
|
|
|
|
const u32 addr = item->data(Qt::UserRole).toUInt();
|
|
|
|
m_context_address = addr;
|
|
|
|
switch (event->button())
|
|
{
|
|
case Qt::LeftButton:
|
|
if (column(item) == 0)
|
|
ToggleBreakpoint();
|
|
else
|
|
SetAddress(addr, SetAddressUpdate::WithUpdate);
|
|
|
|
Update();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void CodeViewWidget::showEvent(QShowEvent* event)
|
|
{
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::ToggleBreakpoint()
|
|
{
|
|
if (PowerPC::debug_interface.IsBreakpoint(GetContextAddress()))
|
|
PowerPC::breakpoints.Remove(GetContextAddress());
|
|
else
|
|
PowerPC::breakpoints.Add(GetContextAddress());
|
|
|
|
emit BreakpointsChanged();
|
|
Update();
|
|
}
|
|
|
|
void CodeViewWidget::AddBreakpoint()
|
|
{
|
|
PowerPC::breakpoints.Add(GetContextAddress());
|
|
|
|
emit BreakpointsChanged();
|
|
Update();
|
|
}
|
|
|
|
u32 CodeViewWidget::GetContextAddress() const
|
|
{
|
|
return m_context_address;
|
|
}
|