Qt/Debugger: Implement "Code" widget
This commit is contained in:
parent
b289be43eb
commit
0a5f0efe18
|
@ -56,6 +56,7 @@ set(SRCS
|
||||||
Config/Mapping/GCPadEmu.cpp
|
Config/Mapping/GCPadEmu.cpp
|
||||||
Config/Mapping/GCPadWiiUConfigDialog.cpp
|
Config/Mapping/GCPadWiiUConfigDialog.cpp
|
||||||
Config/Mapping/Hotkey3D.cpp
|
Config/Mapping/Hotkey3D.cpp
|
||||||
|
Config/Mapping/HotkeyDebugging.cpp
|
||||||
Config/Mapping/HotkeyGeneral.cpp
|
Config/Mapping/HotkeyGeneral.cpp
|
||||||
Config/Mapping/HotkeyGraphics.cpp
|
Config/Mapping/HotkeyGraphics.cpp
|
||||||
Config/Mapping/HotkeyStates.cpp
|
Config/Mapping/HotkeyStates.cpp
|
||||||
|
@ -75,6 +76,8 @@ set(SRCS
|
||||||
Config/PropertiesDialog.cpp
|
Config/PropertiesDialog.cpp
|
||||||
Config/SettingsWindow.cpp
|
Config/SettingsWindow.cpp
|
||||||
Debugger/BreakpointWidget.cpp
|
Debugger/BreakpointWidget.cpp
|
||||||
|
Debugger/CodeViewWidget.cpp
|
||||||
|
Debugger/CodeWidget.cpp
|
||||||
Debugger/NewBreakpointDialog.cpp
|
Debugger/NewBreakpointDialog.cpp
|
||||||
Debugger/RegisterColumn.cpp
|
Debugger/RegisterColumn.cpp
|
||||||
Debugger/RegisterWidget.cpp
|
Debugger/RegisterWidget.cpp
|
||||||
|
|
|
@ -0,0 +1,47 @@
|
||||||
|
// Copyright 2018 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "DolphinQt2/Config/Mapping/HotkeyDebugging.h"
|
||||||
|
|
||||||
|
#include <QGroupBox>
|
||||||
|
#include <QHBoxLayout>
|
||||||
|
#include <QVBoxLayout>
|
||||||
|
|
||||||
|
#include "Core/HotkeyManager.h"
|
||||||
|
|
||||||
|
HotkeyDebugging::HotkeyDebugging(MappingWindow* window) : MappingWidget(window)
|
||||||
|
{
|
||||||
|
CreateMainLayout();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HotkeyDebugging::CreateMainLayout()
|
||||||
|
{
|
||||||
|
m_main_layout = new QHBoxLayout();
|
||||||
|
|
||||||
|
m_main_layout->addWidget(
|
||||||
|
CreateGroupBox(tr("Stepping"), HotkeyManagerEmu::GetHotkeyGroup(HKGP_STEPPING)));
|
||||||
|
|
||||||
|
auto* vbox = new QVBoxLayout();
|
||||||
|
vbox->addWidget(CreateGroupBox(tr("Program Counter"), HotkeyManagerEmu::GetHotkeyGroup(HKGP_PC)));
|
||||||
|
vbox->addWidget(
|
||||||
|
CreateGroupBox(tr("Breakpoint"), HotkeyManagerEmu::GetHotkeyGroup(HKGP_BREAKPOINT)));
|
||||||
|
m_main_layout->addItem(vbox);
|
||||||
|
|
||||||
|
setLayout(m_main_layout);
|
||||||
|
}
|
||||||
|
|
||||||
|
InputConfig* HotkeyDebugging::GetConfig()
|
||||||
|
{
|
||||||
|
return HotkeyManagerEmu::GetConfig();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HotkeyDebugging::LoadSettings()
|
||||||
|
{
|
||||||
|
HotkeyManagerEmu::LoadConfig();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HotkeyDebugging::SaveSettings()
|
||||||
|
{
|
||||||
|
HotkeyManagerEmu::GetConfig()->SaveConfig();
|
||||||
|
}
|
|
@ -0,0 +1,25 @@
|
||||||
|
// Copyright 2018 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "DolphinQt2/Config/Mapping/MappingWidget.h"
|
||||||
|
|
||||||
|
class QHBoxLayout;
|
||||||
|
|
||||||
|
class HotkeyDebugging final : public MappingWidget
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit HotkeyDebugging(MappingWindow* window);
|
||||||
|
|
||||||
|
InputConfig* GetConfig() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void LoadSettings() override;
|
||||||
|
void SaveSettings() override;
|
||||||
|
void CreateMainLayout();
|
||||||
|
|
||||||
|
// Main
|
||||||
|
QHBoxLayout* m_main_layout;
|
||||||
|
};
|
|
@ -22,6 +22,7 @@
|
||||||
#include "DolphinQt2/Config/Mapping/GCMicrophone.h"
|
#include "DolphinQt2/Config/Mapping/GCMicrophone.h"
|
||||||
#include "DolphinQt2/Config/Mapping/GCPadEmu.h"
|
#include "DolphinQt2/Config/Mapping/GCPadEmu.h"
|
||||||
#include "DolphinQt2/Config/Mapping/Hotkey3D.h"
|
#include "DolphinQt2/Config/Mapping/Hotkey3D.h"
|
||||||
|
#include "DolphinQt2/Config/Mapping/HotkeyDebugging.h"
|
||||||
#include "DolphinQt2/Config/Mapping/HotkeyGeneral.h"
|
#include "DolphinQt2/Config/Mapping/HotkeyGeneral.h"
|
||||||
#include "DolphinQt2/Config/Mapping/HotkeyGraphics.h"
|
#include "DolphinQt2/Config/Mapping/HotkeyGraphics.h"
|
||||||
#include "DolphinQt2/Config/Mapping/HotkeyStates.h"
|
#include "DolphinQt2/Config/Mapping/HotkeyStates.h"
|
||||||
|
@ -278,6 +279,7 @@ void MappingWindow::SetMappingType(MappingWindow::Type type)
|
||||||
widget = new HotkeyGeneral(this);
|
widget = new HotkeyGeneral(this);
|
||||||
AddWidget(tr("General"), widget);
|
AddWidget(tr("General"), widget);
|
||||||
AddWidget(tr("TAS Tools"), new HotkeyTAS(this));
|
AddWidget(tr("TAS Tools"), new HotkeyTAS(this));
|
||||||
|
AddWidget(tr("Debugging"), new HotkeyDebugging(this));
|
||||||
AddWidget(tr("Wii and Wii Remote"), new HotkeyWii(this));
|
AddWidget(tr("Wii and Wii Remote"), new HotkeyWii(this));
|
||||||
AddWidget(tr("Graphics"), new HotkeyGraphics(this));
|
AddWidget(tr("Graphics"), new HotkeyGraphics(this));
|
||||||
AddWidget(tr("3D"), new Hotkey3D(this));
|
AddWidget(tr("3D"), new Hotkey3D(this));
|
||||||
|
|
|
@ -25,6 +25,7 @@ public:
|
||||||
bool do_break = true);
|
bool do_break = true);
|
||||||
void AddRangedMBP(u32 from, u32 to, bool do_read = true, bool do_write = true, bool do_log = true,
|
void AddRangedMBP(u32 from, u32 to, bool do_read = true, bool do_write = true, bool do_log = true,
|
||||||
bool do_break = true);
|
bool do_break = true);
|
||||||
|
void Update();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void closeEvent(QCloseEvent*) override;
|
void closeEvent(QCloseEvent*) override;
|
||||||
|
@ -38,8 +39,6 @@ private:
|
||||||
void OnLoad();
|
void OnLoad();
|
||||||
void OnSave();
|
void OnSave();
|
||||||
|
|
||||||
void Update();
|
|
||||||
|
|
||||||
QToolBar* m_toolbar;
|
QToolBar* m_toolbar;
|
||||||
QTableWidget* m_table;
|
QTableWidget* m_table;
|
||||||
QAction* m_load;
|
QAction* m_load;
|
||||||
|
|
|
@ -0,0 +1,544 @@
|
||||||
|
// Copyright 2018 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "DolphinQt2/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/Host.h"
|
||||||
|
#include "Core/PowerPC/PPCAnalyst.h"
|
||||||
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
#include "DolphinQt2/Debugger/CodeWidget.h"
|
||||||
|
#include "DolphinQt2/QtUtils/ActionHelper.h"
|
||||||
|
#include "DolphinQt2/Settings.h"
|
||||||
|
|
||||||
|
constexpr size_t VALID_BRANCH_LENGTH = 10;
|
||||||
|
|
||||||
|
CodeViewWidget::CodeViewWidget()
|
||||||
|
{
|
||||||
|
setColumnCount(5);
|
||||||
|
setShowGrid(false);
|
||||||
|
setContextMenuPolicy(Qt::CustomContextMenu);
|
||||||
|
setSelectionMode(QAbstractItemView::SingleSelection);
|
||||||
|
verticalScrollBar()->setHidden(true);
|
||||||
|
|
||||||
|
for (int i = 0; i < columnCount(); i++)
|
||||||
|
{
|
||||||
|
horizontalHeader()->setSectionResizeMode(i, i == 0 ? QHeaderView::Fixed :
|
||||||
|
QHeaderView::ResizeToContents);
|
||||||
|
}
|
||||||
|
|
||||||
|
verticalHeader()->hide();
|
||||||
|
horizontalHeader()->hide();
|
||||||
|
horizontalHeader()->setStretchLastSection(true);
|
||||||
|
horizontalHeader()->resizeSection(0, 32);
|
||||||
|
|
||||||
|
setFont(Settings::Instance().GetDebugFont());
|
||||||
|
|
||||||
|
Update();
|
||||||
|
|
||||||
|
connect(this, &CodeViewWidget::customContextMenuRequested, this, &CodeViewWidget::OnContextMenu);
|
||||||
|
connect(&Settings::Instance(), &Settings::DebugFontChanged, this, &QWidget::setFont);
|
||||||
|
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this, [this] {
|
||||||
|
m_address = PC;
|
||||||
|
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::Update()
|
||||||
|
{
|
||||||
|
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);
|
||||||
|
|
||||||
|
for (int i = 0; i < rows; i++)
|
||||||
|
setRowHeight(i, 24);
|
||||||
|
|
||||||
|
u32 pc = PowerPC::ppcState.pc;
|
||||||
|
|
||||||
|
if (PowerPC::debug_interface.IsBreakpoint(pc))
|
||||||
|
Core::SetState(Core::State::Paused);
|
||||||
|
|
||||||
|
for (int i = 0; i < rowCount(); i++)
|
||||||
|
{
|
||||||
|
u32 addr = m_address - ((rowCount() / 2) * 4) + i * 4;
|
||||||
|
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);
|
||||||
|
|
||||||
|
auto* ins_item = new QTableWidgetItem(QString::fromStdString(ins));
|
||||||
|
auto* param_item = new QTableWidgetItem(QString::fromStdString(param));
|
||||||
|
auto* description_item = new QTableWidgetItem(QString::fromStdString(desc));
|
||||||
|
|
||||||
|
// 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(Qt::darkGreen);
|
||||||
|
|
||||||
|
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 (color != 0xFFFFFF)
|
||||||
|
item->setBackground(QColor(color).darker(200));
|
||||||
|
|
||||||
|
if (addr == pc && item != bp_item)
|
||||||
|
{
|
||||||
|
item->setBackground(Qt::darkGreen);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (PowerPC::debug_interface.IsBreakpoint(addr))
|
||||||
|
{
|
||||||
|
bp_item->setBackground(Qt::red);
|
||||||
|
}
|
||||||
|
|
||||||
|
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())
|
||||||
|
{
|
||||||
|
addr_item->setSelected(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
g_symbolDB.FillInCallers();
|
||||||
|
|
||||||
|
repaint();
|
||||||
|
m_updating = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 CodeViewWidget::GetAddress() const
|
||||||
|
{
|
||||||
|
return m_address;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::SetAddress(u32 address)
|
||||||
|
{
|
||||||
|
if (m_address == address)
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_address = address;
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::ReplaceAddress(u32 address, bool blr)
|
||||||
|
{
|
||||||
|
auto found = std::find_if(m_repl_list.begin(), m_repl_list.end(),
|
||||||
|
[address](ReplStruct r) { return r.address == address; });
|
||||||
|
|
||||||
|
if (found != m_repl_list.end())
|
||||||
|
{
|
||||||
|
PowerPC::debug_interface.WriteExtraMemory(0, (*found).old_value, address);
|
||||||
|
m_repl_list.erase(found);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ReplStruct repl;
|
||||||
|
|
||||||
|
repl.address = address;
|
||||||
|
repl.old_value = PowerPC::debug_interface.ReadInstruction(address);
|
||||||
|
|
||||||
|
m_repl_list.push_back(repl);
|
||||||
|
|
||||||
|
PowerPC::debug_interface.Patch(address, blr ? 0x60000000 : 0x4e800020);
|
||||||
|
}
|
||||||
|
|
||||||
|
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 =
|
||||||
|
AddAction(menu, tr("Follow &branch"), this, &CodeViewWidget::OnFollowBranch);
|
||||||
|
|
||||||
|
menu->addSeparator();
|
||||||
|
|
||||||
|
AddAction(menu, tr("&Copy address"), this, &CodeViewWidget::OnCopyAddress);
|
||||||
|
auto* copy_address_action =
|
||||||
|
AddAction(menu, tr("Copy &function"), this, &CodeViewWidget::OnCopyFunction);
|
||||||
|
auto* copy_line_action =
|
||||||
|
AddAction(menu, tr("Copy code &line"), this, &CodeViewWidget::OnCopyCode);
|
||||||
|
auto* copy_hex_action = AddAction(menu, tr("Copy &hex"), this, &CodeViewWidget::OnCopyHex);
|
||||||
|
menu->addSeparator();
|
||||||
|
|
||||||
|
auto* symbol_rename_action =
|
||||||
|
AddAction(menu, tr("&Rename symbol"), this, &CodeViewWidget::OnRenameSymbol);
|
||||||
|
auto* symbol_size_action =
|
||||||
|
AddAction(menu, tr("Set symbol &size"), this, &CodeViewWidget::OnSetSymbolSize);
|
||||||
|
auto* symbol_end_action =
|
||||||
|
AddAction(menu, tr("Set symbol &end address"), this, &CodeViewWidget::OnSetSymbolEndAddress);
|
||||||
|
menu->addSeparator();
|
||||||
|
|
||||||
|
AddAction(menu, tr("Run &To Here"), this, &CodeViewWidget::OnRunToHere);
|
||||||
|
auto* function_action =
|
||||||
|
AddAction(menu, tr("&Add function"), this, &CodeViewWidget::OnAddFunction);
|
||||||
|
auto* ppc_action = AddAction(menu, tr("PPC vs x86"), this, &CodeViewWidget::OnPPCComparison);
|
||||||
|
auto* insert_blr_action = AddAction(menu, tr("&Insert blr"), this, &CodeViewWidget::OnInsertBLR);
|
||||||
|
auto* insert_nop_action = AddAction(menu, tr("Insert &nop"), this, &CodeViewWidget::OnInsertNOP);
|
||||||
|
auto* replace_action =
|
||||||
|
AddAction(menu, tr("Re&place instruction"), this, &CodeViewWidget::OnReplaceInstruction);
|
||||||
|
|
||||||
|
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);
|
||||||
|
|
||||||
|
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::OnCopyCode()
|
||||||
|
{
|
||||||
|
const u32 addr = GetContextAddress();
|
||||||
|
|
||||||
|
QApplication::clipboard()->setText(
|
||||||
|
QString::fromStdString(PowerPC::debug_interface.Disassemble(addr)));
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::OnCopyFunction()
|
||||||
|
{
|
||||||
|
const u32 address = GetContextAddress();
|
||||||
|
|
||||||
|
Symbol* symbol = g_symbolDB.GetSymbolFromAddr(address);
|
||||||
|
if (!symbol)
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::string text = symbol->name + "\r\n";
|
||||||
|
// we got a function
|
||||||
|
u32 start = symbol->address;
|
||||||
|
u32 end = start + symbol->size;
|
||||||
|
for (u32 addr = start; addr != end; addr += 4)
|
||||||
|
{
|
||||||
|
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, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::OnInsertNOP()
|
||||||
|
{
|
||||||
|
const u32 addr = GetContextAddress();
|
||||||
|
|
||||||
|
ReplaceAddress(addr, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::OnFollowBranch()
|
||||||
|
{
|
||||||
|
const u32 addr = GetContextAddress();
|
||||||
|
|
||||||
|
u32 branch_addr = GetBranchFromAddress(addr);
|
||||||
|
|
||||||
|
if (!branch_addr)
|
||||||
|
return;
|
||||||
|
|
||||||
|
SetAddress(branch_addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::OnRenameSymbol()
|
||||||
|
{
|
||||||
|
const u32 addr = GetContextAddress();
|
||||||
|
|
||||||
|
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::OnSetSymbolSize()
|
||||||
|
{
|
||||||
|
const u32 addr = GetContextAddress();
|
||||||
|
|
||||||
|
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();
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
bool good;
|
||||||
|
QString name = QInputDialog::getText(
|
||||||
|
this, tr("Change instruction"), tr("New instruction:"), QLineEdit::Normal,
|
||||||
|
QStringLiteral("%1").arg(read_result.hex, 8, 16, QLatin1Char('0')), &good);
|
||||||
|
|
||||||
|
u32 code = name.toUInt(&good, 16);
|
||||||
|
|
||||||
|
if (good)
|
||||||
|
{
|
||||||
|
PowerPC::debug_interface.Patch(addr, code);
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::resizeEvent(QResizeEvent*)
|
||||||
|
{
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeViewWidget::keyPressEvent(QKeyEvent* event)
|
||||||
|
{
|
||||||
|
switch (event->key())
|
||||||
|
{
|
||||||
|
case Qt::Key_Up:
|
||||||
|
m_address -= 3 * sizeof(u32);
|
||||||
|
Update();
|
||||||
|
return;
|
||||||
|
case Qt::Key_Down:
|
||||||
|
m_address += 3 * 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)
|
||||||
|
{
|
||||||
|
int delta = event->delta() > 0 ? -1 : 1;
|
||||||
|
|
||||||
|
m_address += delta * 3 * 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);
|
||||||
|
|
||||||
|
Update();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
|
@ -0,0 +1,72 @@
|
||||||
|
// Copyright 2018 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <QTableWidget>
|
||||||
|
|
||||||
|
#include "Common/CommonTypes.h"
|
||||||
|
|
||||||
|
class QKeyEvent;
|
||||||
|
class QMouseEvent;
|
||||||
|
class QResizeEvent;
|
||||||
|
|
||||||
|
class CodeViewWidget : public QTableWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit CodeViewWidget();
|
||||||
|
|
||||||
|
u32 GetAddress() const;
|
||||||
|
u32 GetContextAddress() const;
|
||||||
|
void SetAddress(u32 address);
|
||||||
|
|
||||||
|
void Update();
|
||||||
|
|
||||||
|
void ToggleBreakpoint();
|
||||||
|
void AddBreakpoint();
|
||||||
|
signals:
|
||||||
|
void RequestPPCComparison(u32 addr);
|
||||||
|
void SymbolsChanged();
|
||||||
|
void BreakpointsChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void ReplaceAddress(u32 address, bool blr);
|
||||||
|
|
||||||
|
void resizeEvent(QResizeEvent*) override;
|
||||||
|
void keyPressEvent(QKeyEvent* event) override;
|
||||||
|
void mousePressEvent(QMouseEvent* event) override;
|
||||||
|
void wheelEvent(QWheelEvent* event) override;
|
||||||
|
|
||||||
|
void OnContextMenu();
|
||||||
|
|
||||||
|
void OnFollowBranch();
|
||||||
|
void OnCopyAddress();
|
||||||
|
void OnCopyFunction();
|
||||||
|
void OnCopyCode();
|
||||||
|
void OnCopyHex();
|
||||||
|
void OnRenameSymbol();
|
||||||
|
void OnSetSymbolSize();
|
||||||
|
void OnSetSymbolEndAddress();
|
||||||
|
void OnRunToHere();
|
||||||
|
void OnAddFunction();
|
||||||
|
void OnPPCComparison();
|
||||||
|
void OnInsertBLR();
|
||||||
|
void OnInsertNOP();
|
||||||
|
void OnReplaceInstruction();
|
||||||
|
|
||||||
|
struct ReplStruct
|
||||||
|
{
|
||||||
|
u32 address;
|
||||||
|
u32 old_value;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<ReplStruct> m_repl_list;
|
||||||
|
bool m_updating = false;
|
||||||
|
|
||||||
|
u32 m_address = 0;
|
||||||
|
u32 m_context_address = 0;
|
||||||
|
};
|
|
@ -0,0 +1,481 @@
|
||||||
|
// Copyright 2018 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "DolphinQt2/Debugger/CodeWidget.h"
|
||||||
|
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
|
#include <QGridLayout>
|
||||||
|
#include <QGroupBox>
|
||||||
|
#include <QLineEdit>
|
||||||
|
#include <QListWidget>
|
||||||
|
#include <QSettings>
|
||||||
|
#include <QSplitter>
|
||||||
|
#include <QTableWidget>
|
||||||
|
#include <QWidget>
|
||||||
|
|
||||||
|
#include "Common/Event.h"
|
||||||
|
#include "Common/StringUtil.h"
|
||||||
|
#include "Core/Core.h"
|
||||||
|
#include "Core/Debugger/Debugger_SymbolMap.h"
|
||||||
|
#include "Core/HW/CPU.h"
|
||||||
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
#include "DolphinQt2/Debugger/CodeViewWidget.h"
|
||||||
|
#include "DolphinQt2/Settings.h"
|
||||||
|
|
||||||
|
CodeWidget::CodeWidget(QWidget* parent) : QDockWidget(parent)
|
||||||
|
{
|
||||||
|
setWindowTitle(tr("Code"));
|
||||||
|
setAllowedAreas(Qt::AllDockWidgetAreas);
|
||||||
|
|
||||||
|
QSettings settings;
|
||||||
|
|
||||||
|
restoreGeometry(settings.value(QStringLiteral("codewidget/geometry")).toByteArray());
|
||||||
|
setFloating(settings.value(QStringLiteral("codewidget/floating")).toBool());
|
||||||
|
|
||||||
|
connect(&Settings::Instance(), &Settings::CodeVisibilityChanged,
|
||||||
|
[this](bool visible) { setHidden(!visible); });
|
||||||
|
|
||||||
|
connect(&Settings::Instance(), &Settings::DebugModeToggled,
|
||||||
|
[this](bool enabled) { setHidden(!enabled || !Settings::Instance().IsCodeVisible()); });
|
||||||
|
|
||||||
|
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this, &CodeWidget::Update);
|
||||||
|
|
||||||
|
setHidden(!Settings::Instance().IsCodeVisible() || !Settings::Instance().IsDebugModeEnabled());
|
||||||
|
|
||||||
|
CreateWidgets();
|
||||||
|
ConnectWidgets();
|
||||||
|
|
||||||
|
m_code_splitter->restoreState(
|
||||||
|
settings.value(QStringLiteral("codewidget/codesplitter")).toByteArray());
|
||||||
|
m_box_splitter->restoreState(
|
||||||
|
settings.value(QStringLiteral("codewidget/boxsplitter")).toByteArray());
|
||||||
|
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeWidget::~CodeWidget()
|
||||||
|
{
|
||||||
|
QSettings settings;
|
||||||
|
|
||||||
|
settings.setValue(QStringLiteral("codewidget/geometry"), saveGeometry());
|
||||||
|
settings.setValue(QStringLiteral("codewidget/floating"), isFloating());
|
||||||
|
settings.setValue(QStringLiteral("codewidget/codesplitter"), m_code_splitter->saveState());
|
||||||
|
settings.setValue(QStringLiteral("codewidget/boxplitter"), m_box_splitter->saveState());
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::closeEvent(QCloseEvent*)
|
||||||
|
{
|
||||||
|
Settings::Instance().SetCodeVisible(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::CreateWidgets()
|
||||||
|
{
|
||||||
|
auto* layout = new QGridLayout;
|
||||||
|
|
||||||
|
m_search_address = new QLineEdit;
|
||||||
|
m_search_symbols = new QLineEdit;
|
||||||
|
m_code_view = new CodeViewWidget;
|
||||||
|
|
||||||
|
m_search_address->setPlaceholderText(tr("Search Address"));
|
||||||
|
m_search_symbols->setPlaceholderText(tr("Filter Symbols"));
|
||||||
|
|
||||||
|
// Callstack
|
||||||
|
auto* callstack_box = new QGroupBox(tr("Callstack"));
|
||||||
|
auto* callstack_layout = new QVBoxLayout;
|
||||||
|
m_callstack_list = new QListWidget;
|
||||||
|
|
||||||
|
callstack_box->setLayout(callstack_layout);
|
||||||
|
callstack_layout->addWidget(m_callstack_list);
|
||||||
|
|
||||||
|
// Symbols
|
||||||
|
auto* symbols_box = new QGroupBox(tr("Symbols"));
|
||||||
|
auto* symbols_layout = new QVBoxLayout;
|
||||||
|
m_symbols_list = new QListWidget;
|
||||||
|
|
||||||
|
symbols_box->setLayout(symbols_layout);
|
||||||
|
symbols_layout->addWidget(m_symbols_list);
|
||||||
|
|
||||||
|
// Function calls
|
||||||
|
auto* function_calls_box = new QGroupBox(tr("Function calls"));
|
||||||
|
auto* function_calls_layout = new QVBoxLayout;
|
||||||
|
m_function_calls_list = new QListWidget;
|
||||||
|
|
||||||
|
function_calls_box->setLayout(function_calls_layout);
|
||||||
|
function_calls_layout->addWidget(m_function_calls_list);
|
||||||
|
|
||||||
|
// Function callers
|
||||||
|
auto* function_callers_box = new QGroupBox(tr("Function callers"));
|
||||||
|
auto* function_callers_layout = new QVBoxLayout;
|
||||||
|
m_function_callers_list = new QListWidget;
|
||||||
|
|
||||||
|
function_callers_box->setLayout(function_callers_layout);
|
||||||
|
function_callers_layout->addWidget(m_function_callers_list);
|
||||||
|
|
||||||
|
m_box_splitter = new QSplitter(Qt::Vertical);
|
||||||
|
|
||||||
|
m_box_splitter->addWidget(callstack_box);
|
||||||
|
m_box_splitter->addWidget(symbols_box);
|
||||||
|
m_box_splitter->addWidget(function_calls_box);
|
||||||
|
m_box_splitter->addWidget(function_callers_box);
|
||||||
|
|
||||||
|
m_code_splitter = new QSplitter(Qt::Horizontal);
|
||||||
|
|
||||||
|
m_code_splitter->addWidget(m_box_splitter);
|
||||||
|
m_code_splitter->addWidget(m_code_view);
|
||||||
|
|
||||||
|
layout->addWidget(m_search_address, 0, 0);
|
||||||
|
layout->addWidget(m_search_symbols, 0, 1);
|
||||||
|
layout->addWidget(m_code_splitter, 1, 0, -1, -1);
|
||||||
|
|
||||||
|
QWidget* widget = new QWidget(this);
|
||||||
|
widget->setLayout(layout);
|
||||||
|
setWidget(widget);
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::ConnectWidgets()
|
||||||
|
{
|
||||||
|
connect(m_search_address, &QLineEdit::textChanged, this, &CodeWidget::OnSearchAddress);
|
||||||
|
connect(m_search_symbols, &QLineEdit::textChanged, this, &CodeWidget::OnSearchSymbols);
|
||||||
|
|
||||||
|
connect(m_symbols_list, &QListWidget::itemSelectionChanged, this, &CodeWidget::OnSelectSymbol);
|
||||||
|
connect(m_callstack_list, &QListWidget::itemSelectionChanged, this,
|
||||||
|
&CodeWidget::OnSelectCallstack);
|
||||||
|
connect(m_function_calls_list, &QListWidget::itemSelectionChanged, this,
|
||||||
|
&CodeWidget::OnSelectFunctionCalls);
|
||||||
|
connect(m_function_callers_list, &QListWidget::itemSelectionChanged, this,
|
||||||
|
&CodeWidget::OnSelectFunctionCallers);
|
||||||
|
|
||||||
|
connect(m_code_view, &CodeViewWidget::SymbolsChanged, this, &CodeWidget::UpdateSymbols);
|
||||||
|
connect(m_code_view, &CodeViewWidget::BreakpointsChanged, this,
|
||||||
|
[this] { emit BreakpointsChanged(); });
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::OnSearchAddress()
|
||||||
|
{
|
||||||
|
bool good = true;
|
||||||
|
u32 address = m_search_address->text().toUInt(&good, 16);
|
||||||
|
|
||||||
|
QPalette palette;
|
||||||
|
QFont font;
|
||||||
|
|
||||||
|
if (!good && !m_search_address->text().isEmpty())
|
||||||
|
{
|
||||||
|
font.setBold(true);
|
||||||
|
palette.setColor(QPalette::Text, Qt::red);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_search_address->setPalette(palette);
|
||||||
|
m_search_address->setFont(font);
|
||||||
|
|
||||||
|
if (good)
|
||||||
|
m_code_view->SetAddress(address);
|
||||||
|
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::OnSearchSymbols()
|
||||||
|
{
|
||||||
|
m_symbol_filter = m_search_symbols->text();
|
||||||
|
UpdateSymbols();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::OnSelectSymbol()
|
||||||
|
{
|
||||||
|
const auto items = m_symbols_list->selectedItems();
|
||||||
|
if (items.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
Symbol* symbol = g_symbolDB.GetSymbolFromAddr(items[0]->data(Qt::UserRole).toUInt());
|
||||||
|
|
||||||
|
m_code_view->SetAddress(items[0]->data(Qt::UserRole).toUInt());
|
||||||
|
UpdateCallstack();
|
||||||
|
UpdateFunctionCalls(symbol);
|
||||||
|
UpdateFunctionCallers(symbol);
|
||||||
|
|
||||||
|
m_code_view->setFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::OnSelectCallstack()
|
||||||
|
{
|
||||||
|
const auto items = m_callstack_list->selectedItems();
|
||||||
|
if (items.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_code_view->SetAddress(items[0]->data(Qt::UserRole).toUInt());
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::OnSelectFunctionCalls()
|
||||||
|
{
|
||||||
|
const auto items = m_function_calls_list->selectedItems();
|
||||||
|
if (items.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_code_view->SetAddress(items[0]->data(Qt::UserRole).toUInt());
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::OnSelectFunctionCallers()
|
||||||
|
{
|
||||||
|
const auto items = m_function_callers_list->selectedItems();
|
||||||
|
if (items.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
m_code_view->SetAddress(items[0]->data(Qt::UserRole).toUInt());
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::Update()
|
||||||
|
{
|
||||||
|
Symbol* symbol = g_symbolDB.GetSymbolFromAddr(m_code_view->GetAddress());
|
||||||
|
|
||||||
|
UpdateCallstack();
|
||||||
|
UpdateSymbols();
|
||||||
|
|
||||||
|
if (!symbol)
|
||||||
|
return;
|
||||||
|
|
||||||
|
UpdateFunctionCalls(symbol);
|
||||||
|
UpdateFunctionCallers(symbol);
|
||||||
|
|
||||||
|
m_code_view->Update();
|
||||||
|
m_code_view->setFocus();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::UpdateCallstack()
|
||||||
|
{
|
||||||
|
m_callstack_list->clear();
|
||||||
|
|
||||||
|
std::vector<Dolphin_Debugger::CallstackEntry> stack;
|
||||||
|
|
||||||
|
bool success = Dolphin_Debugger::GetCallstack(stack);
|
||||||
|
|
||||||
|
if (!success)
|
||||||
|
{
|
||||||
|
m_callstack_list->addItem(tr("Invalid callstack"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const auto& frame : stack)
|
||||||
|
{
|
||||||
|
auto* item =
|
||||||
|
new QListWidgetItem(QString::fromStdString(frame.Name.substr(0, frame.Name.length() - 1)));
|
||||||
|
item->setData(Qt::UserRole, frame.vAddress);
|
||||||
|
|
||||||
|
m_callstack_list->addItem(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::UpdateSymbols()
|
||||||
|
{
|
||||||
|
QString selection = m_symbols_list->selectedItems().isEmpty() ?
|
||||||
|
QStringLiteral("") :
|
||||||
|
m_symbols_list->selectedItems()[0]->text();
|
||||||
|
m_symbols_list->clear();
|
||||||
|
|
||||||
|
for (const auto& symbol : g_symbolDB.Symbols())
|
||||||
|
{
|
||||||
|
QString name = QString::fromStdString(symbol.second.name);
|
||||||
|
|
||||||
|
auto* item = new QListWidgetItem(name);
|
||||||
|
if (name == selection)
|
||||||
|
item->setSelected(true);
|
||||||
|
|
||||||
|
// Disable non-function symbols as you can't do anything with them.
|
||||||
|
if (symbol.second.type != Symbol::Type::Function)
|
||||||
|
item->setFlags(Qt::NoItemFlags);
|
||||||
|
|
||||||
|
item->setData(Qt::UserRole, symbol.second.address);
|
||||||
|
|
||||||
|
if (name.indexOf(m_symbol_filter) != -1)
|
||||||
|
m_symbols_list->addItem(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
m_symbols_list->sortItems();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::UpdateFunctionCalls(Symbol* symbol)
|
||||||
|
{
|
||||||
|
m_function_calls_list->clear();
|
||||||
|
|
||||||
|
for (const auto& call : symbol->calls)
|
||||||
|
{
|
||||||
|
u32 addr = call.function;
|
||||||
|
Symbol* call_symbol = g_symbolDB.GetSymbolFromAddr(addr);
|
||||||
|
|
||||||
|
if (call_symbol)
|
||||||
|
{
|
||||||
|
auto* item = new QListWidgetItem(QString::fromStdString(
|
||||||
|
StringFromFormat("> %s (%08x)", call_symbol->name.c_str(), addr).c_str()));
|
||||||
|
item->setData(Qt::UserRole, addr);
|
||||||
|
|
||||||
|
m_function_calls_list->addItem(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::UpdateFunctionCallers(Symbol* symbol)
|
||||||
|
{
|
||||||
|
m_function_callers_list->clear();
|
||||||
|
|
||||||
|
for (const auto& caller : symbol->callers)
|
||||||
|
{
|
||||||
|
u32 addr = caller.callAddress;
|
||||||
|
Symbol* caller_symbol = g_symbolDB.GetSymbolFromAddr(addr);
|
||||||
|
|
||||||
|
if (caller_symbol)
|
||||||
|
{
|
||||||
|
auto* item = new QListWidgetItem(QString::fromStdString(
|
||||||
|
StringFromFormat("< %s (%08x)", caller_symbol->name.c_str(), addr).c_str()));
|
||||||
|
item->setData(Qt::UserRole, addr);
|
||||||
|
|
||||||
|
m_function_callers_list->addItem(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::Step()
|
||||||
|
{
|
||||||
|
if (!CPU::IsStepping())
|
||||||
|
return;
|
||||||
|
|
||||||
|
Common::Event sync_event;
|
||||||
|
|
||||||
|
PowerPC::CoreMode old_mode = PowerPC::GetMode();
|
||||||
|
PowerPC::SetMode(PowerPC::CoreMode::Interpreter);
|
||||||
|
PowerPC::breakpoints.ClearAllTemporary();
|
||||||
|
CPU::StepOpcode(&sync_event);
|
||||||
|
sync_event.WaitFor(std::chrono::milliseconds(20));
|
||||||
|
PowerPC::SetMode(old_mode);
|
||||||
|
Core::DisplayMessage(tr("Step successful!").toStdString(), 2000);
|
||||||
|
|
||||||
|
Core::SetState(Core::State::Paused);
|
||||||
|
m_code_view->SetAddress(PC);
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::StepOver()
|
||||||
|
{
|
||||||
|
if (!CPU::IsStepping())
|
||||||
|
return;
|
||||||
|
|
||||||
|
UGeckoInstruction inst = PowerPC::HostRead_Instruction(PC);
|
||||||
|
if (inst.LK)
|
||||||
|
{
|
||||||
|
PowerPC::breakpoints.ClearAllTemporary();
|
||||||
|
PowerPC::breakpoints.Add(PC + 4, true);
|
||||||
|
CPU::EnableStepping(false);
|
||||||
|
Core::DisplayMessage(tr("Step over in progress...").toStdString(), 2000);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Step();
|
||||||
|
}
|
||||||
|
|
||||||
|
Core::SetState(Core::State::Paused);
|
||||||
|
m_code_view->SetAddress(PC);
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns true on a rfi, blr or on a bclr that evaluates to true.
|
||||||
|
static bool WillInstructionReturn(UGeckoInstruction inst)
|
||||||
|
{
|
||||||
|
// Is a rfi instruction
|
||||||
|
if (inst.hex == 0x4C000064u)
|
||||||
|
return true;
|
||||||
|
bool counter = (inst.BO_2 >> 2 & 1) != 0 || (CTR != 0) != ((inst.BO_2 >> 1 & 1) != 0);
|
||||||
|
bool condition = inst.BO_2 >> 4 != 0 || GetCRBit(inst.BI_2) == (inst.BO_2 >> 3 & 1);
|
||||||
|
bool isBclr = inst.OPCD_7 == 0b010011 && (inst.hex >> 1 & 0b10000) != 0;
|
||||||
|
return isBclr && counter && condition && !inst.LK_3;
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::StepOut()
|
||||||
|
{
|
||||||
|
if (!CPU::IsStepping())
|
||||||
|
return;
|
||||||
|
|
||||||
|
Core::SetState(Core::State::Running);
|
||||||
|
CPU::PauseAndLock(true, false);
|
||||||
|
PowerPC::breakpoints.ClearAllTemporary();
|
||||||
|
|
||||||
|
// Keep stepping until the next return instruction or timeout after five seconds
|
||||||
|
using clock = std::chrono::steady_clock;
|
||||||
|
clock::time_point timeout = clock::now() + std::chrono::seconds(5);
|
||||||
|
PowerPC::CoreMode old_mode = PowerPC::GetMode();
|
||||||
|
PowerPC::SetMode(PowerPC::CoreMode::Interpreter);
|
||||||
|
|
||||||
|
// Loop until either the current instruction is a return instruction with no Link flag
|
||||||
|
// or a breakpoint is detected so it can step at the breakpoint. If the PC is currently
|
||||||
|
// on a breakpoint, skip it.
|
||||||
|
UGeckoInstruction inst = PowerPC::HostRead_Instruction(PC);
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (WillInstructionReturn(inst))
|
||||||
|
{
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (inst.LK)
|
||||||
|
{
|
||||||
|
// Step over branches
|
||||||
|
u32 next_pc = PC + 4;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
} while (PC != next_pc && clock::now() < timeout &&
|
||||||
|
!PowerPC::breakpoints.IsAddressBreakPoint(PC));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
PowerPC::SingleStep();
|
||||||
|
}
|
||||||
|
|
||||||
|
inst = PowerPC::HostRead_Instruction(PC);
|
||||||
|
} while (clock::now() < timeout && !PowerPC::breakpoints.IsAddressBreakPoint(PC));
|
||||||
|
|
||||||
|
PowerPC::SetMode(old_mode);
|
||||||
|
CPU::PauseAndLock(false, false);
|
||||||
|
|
||||||
|
if (PowerPC::breakpoints.IsAddressBreakPoint(PC))
|
||||||
|
Core::DisplayMessage(tr("Breakpoint encountered! Step out aborted.").toStdString(), 2000);
|
||||||
|
else if (clock::now() >= timeout)
|
||||||
|
Core::DisplayMessage(tr("Step out timed out!").toStdString(), 2000);
|
||||||
|
else
|
||||||
|
Core::DisplayMessage(tr("Step out successful!").toStdString(), 2000);
|
||||||
|
|
||||||
|
Core::SetState(Core::State::Paused);
|
||||||
|
m_code_view->SetAddress(PC);
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::Skip()
|
||||||
|
{
|
||||||
|
PC += 4;
|
||||||
|
ShowPC();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::ShowPC()
|
||||||
|
{
|
||||||
|
m_code_view->SetAddress(PC);
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::SetPC()
|
||||||
|
{
|
||||||
|
PC = m_code_view->GetAddress();
|
||||||
|
Update();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::ToggleBreakpoint()
|
||||||
|
{
|
||||||
|
m_code_view->ToggleBreakpoint();
|
||||||
|
}
|
||||||
|
|
||||||
|
void CodeWidget::AddBreakpoint()
|
||||||
|
{
|
||||||
|
m_code_view->AddBreakpoint();
|
||||||
|
}
|
|
@ -0,0 +1,68 @@
|
||||||
|
// Copyright 2018 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QDockWidget>
|
||||||
|
#include <QString>
|
||||||
|
|
||||||
|
class CodeViewWidget;
|
||||||
|
class QCloseEvent;
|
||||||
|
class QLineEdit;
|
||||||
|
class QSplitter;
|
||||||
|
class QListWidget;
|
||||||
|
class QTableWidget;
|
||||||
|
struct Symbol;
|
||||||
|
|
||||||
|
class CodeWidget : public QDockWidget
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
explicit CodeWidget(QWidget* parent = nullptr);
|
||||||
|
~CodeWidget();
|
||||||
|
|
||||||
|
void Step();
|
||||||
|
void StepOver();
|
||||||
|
void StepOut();
|
||||||
|
void Skip();
|
||||||
|
void ShowPC();
|
||||||
|
void SetPC();
|
||||||
|
|
||||||
|
void ToggleBreakpoint();
|
||||||
|
void AddBreakpoint();
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void BreakpointsChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void CreateWidgets();
|
||||||
|
void ConnectWidgets();
|
||||||
|
void Update();
|
||||||
|
void UpdateCallstack();
|
||||||
|
void UpdateSymbols();
|
||||||
|
void UpdateFunctionCalls(Symbol* symbol);
|
||||||
|
void UpdateFunctionCallers(Symbol* symbol);
|
||||||
|
|
||||||
|
void OnSearchAddress();
|
||||||
|
void OnSearchSymbols();
|
||||||
|
void OnSelectSymbol();
|
||||||
|
void OnSelectCallstack();
|
||||||
|
void OnSelectFunctionCallers();
|
||||||
|
void OnSelectFunctionCalls();
|
||||||
|
|
||||||
|
void closeEvent(QCloseEvent*) override;
|
||||||
|
|
||||||
|
QLineEdit* m_search_address;
|
||||||
|
QLineEdit* m_search_symbols;
|
||||||
|
|
||||||
|
QListWidget* m_callstack_list;
|
||||||
|
QListWidget* m_symbols_list;
|
||||||
|
QListWidget* m_function_calls_list;
|
||||||
|
QListWidget* m_function_callers_list;
|
||||||
|
CodeViewWidget* m_code_view;
|
||||||
|
QSplitter* m_box_splitter;
|
||||||
|
QSplitter* m_code_splitter;
|
||||||
|
|
||||||
|
QString m_symbol_filter;
|
||||||
|
};
|
|
@ -91,6 +91,8 @@
|
||||||
<QtMoc Include="TAS\StickWidget.h" />
|
<QtMoc Include="TAS\StickWidget.h" />
|
||||||
<QtMoc Include="TAS\IRWidget.h" />
|
<QtMoc Include="TAS\IRWidget.h" />
|
||||||
<QtMoc Include="Debugger\BreakpointWidget.h" />
|
<QtMoc Include="Debugger\BreakpointWidget.h" />
|
||||||
|
<QtMoc Include="Debugger\CodeWidget.h" />
|
||||||
|
<QtMoc Include="Debugger\CodeViewWidget.h" />
|
||||||
<QtMoc Include="Debugger\NewBreakpointDialog.h" />
|
<QtMoc Include="Debugger\NewBreakpointDialog.h" />
|
||||||
<QtMoc Include="Debugger\RegisterWidget.h" />
|
<QtMoc Include="Debugger\RegisterWidget.h" />
|
||||||
<QtMoc Include="Debugger\WatchWidget.h" />
|
<QtMoc Include="Debugger\WatchWidget.h" />
|
||||||
|
@ -107,7 +109,7 @@
|
||||||
<QtMoc Include="Settings\AudioPane.h" />
|
<QtMoc Include="Settings\AudioPane.h" />
|
||||||
<QtMoc Include="Settings\WiiPane.h" />
|
<QtMoc Include="Settings\WiiPane.h" />
|
||||||
<QtMoc Include="Settings\USBDeviceAddToWhitelistDialog.h" />
|
<QtMoc Include="Settings\USBDeviceAddToWhitelistDialog.h" />
|
||||||
<QtMoc Include="MainWindow.h" />
|
<QtMoc Include="MainWindow.h" />
|
||||||
<QtMoc Include="MenuBar.h" />
|
<QtMoc Include="MenuBar.h" />
|
||||||
<QtMoc Include="NetPlay\GameListDialog.h" />
|
<QtMoc Include="NetPlay\GameListDialog.h" />
|
||||||
<QtMoc Include="NetPlay\MD5Dialog.h" />
|
<QtMoc Include="NetPlay\MD5Dialog.h" />
|
||||||
|
@ -133,6 +135,8 @@
|
||||||
<ClCompile Include="$(QtMocOutPrefix)AdvancedWidget.cpp" />
|
<ClCompile Include="$(QtMocOutPrefix)AdvancedWidget.cpp" />
|
||||||
<ClCompile Include="$(QtMocOutPrefix)BreakpointWidget.cpp" />
|
<ClCompile Include="$(QtMocOutPrefix)BreakpointWidget.cpp" />
|
||||||
<ClCompile Include="$(QtMocOutPrefix)CheatWarningWidget.cpp" />
|
<ClCompile Include="$(QtMocOutPrefix)CheatWarningWidget.cpp" />
|
||||||
|
<ClCompile Include="$(QtMocOutPrefix)CodeViewWidget.cpp" />
|
||||||
|
<ClCompile Include="$(QtMocOutPrefix)CodeWidget.cpp" />
|
||||||
<ClCompile Include="$(QtMocOutPrefix)ControllersWindow.cpp" />
|
<ClCompile Include="$(QtMocOutPrefix)ControllersWindow.cpp" />
|
||||||
<ClCompile Include="$(QtMocOutPrefix)EnhancementsWidget.cpp" />
|
<ClCompile Include="$(QtMocOutPrefix)EnhancementsWidget.cpp" />
|
||||||
<ClCompile Include="$(QtMocOutPrefix)FIFOPlayerWindow.cpp" />
|
<ClCompile Include="$(QtMocOutPrefix)FIFOPlayerWindow.cpp" />
|
||||||
|
@ -209,6 +213,7 @@
|
||||||
<ClCompile Include="Config\Mapping\GCPadEmu.cpp" />
|
<ClCompile Include="Config\Mapping\GCPadEmu.cpp" />
|
||||||
<ClCompile Include="Config\Mapping\GCPadWiiUConfigDialog.cpp" />
|
<ClCompile Include="Config\Mapping\GCPadWiiUConfigDialog.cpp" />
|
||||||
<ClCompile Include="Config\Mapping\Hotkey3D.cpp" />
|
<ClCompile Include="Config\Mapping\Hotkey3D.cpp" />
|
||||||
|
<ClCompile Include="Config\Mapping\HotkeyDebugging.cpp" />
|
||||||
<ClCompile Include="Config\Mapping\HotkeyGeneral.cpp" />
|
<ClCompile Include="Config\Mapping\HotkeyGeneral.cpp" />
|
||||||
<ClCompile Include="Config\Mapping\HotkeyGraphics.cpp" />
|
<ClCompile Include="Config\Mapping\HotkeyGraphics.cpp" />
|
||||||
<ClCompile Include="Config\Mapping\HotkeyStates.cpp" />
|
<ClCompile Include="Config\Mapping\HotkeyStates.cpp" />
|
||||||
|
@ -229,6 +234,8 @@
|
||||||
<ClCompile Include="Config\LogWidget.cpp" />
|
<ClCompile Include="Config\LogWidget.cpp" />
|
||||||
<ClCompile Include="Config\PropertiesDialog.cpp" />
|
<ClCompile Include="Config\PropertiesDialog.cpp" />
|
||||||
<ClCompile Include="Config\SettingsWindow.cpp" />
|
<ClCompile Include="Config\SettingsWindow.cpp" />
|
||||||
|
<ClCompile Include="Debugger\CodeViewWidget.cpp" />
|
||||||
|
<ClCompile Include="Debugger\CodeWidget.cpp" />
|
||||||
<ClCompile Include="FIFOPlayerWindow.cpp" />
|
<ClCompile Include="FIFOPlayerWindow.cpp" />
|
||||||
<ClCompile Include="TAS\GCTASInputWindow.cpp" />
|
<ClCompile Include="TAS\GCTASInputWindow.cpp" />
|
||||||
<ClCompile Include="TAS\WiiTASInputWindow.cpp" />
|
<ClCompile Include="TAS\WiiTASInputWindow.cpp" />
|
||||||
|
@ -275,7 +282,7 @@
|
||||||
<ClCompile Include="Settings\InterfacePane.cpp" />
|
<ClCompile Include="Settings\InterfacePane.cpp" />
|
||||||
<ClCompile Include="Settings\PathPane.cpp" />
|
<ClCompile Include="Settings\PathPane.cpp" />
|
||||||
<ClCompile Include="Settings\WiiPane.cpp" />
|
<ClCompile Include="Settings\WiiPane.cpp" />
|
||||||
<ClCompile Include="Settings\USBDeviceAddToWhitelistDialog.cpp" />
|
<ClCompile Include="Settings\USBDeviceAddToWhitelistDialog.cpp" />
|
||||||
<ClCompile Include="ToolBar.cpp" />
|
<ClCompile Include="ToolBar.cpp" />
|
||||||
<ClCompile Include="Translation.cpp" />
|
<ClCompile Include="Translation.cpp" />
|
||||||
<ClCompile Include="WiiUpdate.cpp" />
|
<ClCompile Include="WiiUpdate.cpp" />
|
||||||
|
@ -291,6 +298,8 @@
|
||||||
<ClInclude Include="Config\Mapping\HotkeyGraphics.h" />
|
<ClInclude Include="Config\Mapping\HotkeyGraphics.h" />
|
||||||
<ClInclude Include="Config\Mapping\HotkeyStates.h" />
|
<ClInclude Include="Config\Mapping\HotkeyStates.h" />
|
||||||
<ClInclude Include="Config\Mapping\HotkeyTAS.h" />
|
<ClInclude Include="Config\Mapping\HotkeyTAS.h" />
|
||||||
|
<ClInclude Include="Debugger\CodeViewWidget.h" />
|
||||||
|
<ClInclude Include="Debugger\CodeWidget.h" />
|
||||||
<ClInclude Include="TAS\Shared.h" />
|
<ClInclude Include="TAS\Shared.h" />
|
||||||
<ClInclude Include="Config\Mapping\HotkeyWii.h" />
|
<ClInclude Include="Config\Mapping\HotkeyWii.h" />
|
||||||
<ClInclude Include="Config\Mapping\MappingBool.h" />
|
<ClInclude Include="Config\Mapping\MappingBool.h" />
|
||||||
|
|
|
@ -9,7 +9,10 @@
|
||||||
|
|
||||||
#include "Common/Common.h"
|
#include "Common/Common.h"
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
|
#include "Core/Core.h"
|
||||||
|
#include "Core/Debugger/PPCDebugInterface.h"
|
||||||
#include "Core/Host.h"
|
#include "Core/Host.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
#include "DolphinQt2/Settings.h"
|
#include "DolphinQt2/Settings.h"
|
||||||
#include "VideoCommon/RenderBase.h"
|
#include "VideoCommon/RenderBase.h"
|
||||||
|
|
||||||
|
@ -90,10 +93,16 @@ bool Host_RendererIsFullscreen()
|
||||||
{
|
{
|
||||||
return Host::GetInstance()->GetRenderFullscreen();
|
return Host::GetInstance()->GetRenderFullscreen();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Host_YieldToUI()
|
void Host_YieldToUI()
|
||||||
{
|
{
|
||||||
qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
|
qApp->processEvents(QEventLoop::ExcludeUserInputEvents);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Host_UpdateDisasmDialog()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
void Host_UpdateProgressDialog(const char* caption, int position, int total)
|
void Host_UpdateProgressDialog(const char* caption, int position, int total)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -114,9 +123,6 @@ bool Host_UINeedsControllerState()
|
||||||
void Host_NotifyMapLoaded()
|
void Host_NotifyMapLoaded()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
void Host_UpdateDisasmDialog()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
void Host_ShowVideoConfig(void* parent, const std::string& backend_name)
|
void Host_ShowVideoConfig(void* parent, const std::string& backend_name)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
|
@ -194,7 +194,31 @@ void HotkeyScheduler::Run()
|
||||||
IsHotkey(HK_TRIGGER_SYNC_BUTTON, true));
|
IsHotkey(HK_TRIGGER_SYNC_BUTTON, true));
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO Debugging shortcuts (Separate PR)
|
if (IsHotkey(HK_STEP))
|
||||||
|
emit Step();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_STEP_OVER))
|
||||||
|
emit StepOver();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_STEP_OUT))
|
||||||
|
emit StepOut();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_SKIP))
|
||||||
|
emit Skip();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_SHOW_PC))
|
||||||
|
emit ShowPC();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_SET_PC))
|
||||||
|
emit Skip();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_BP_TOGGLE))
|
||||||
|
emit ToggleBreakpoint();
|
||||||
|
|
||||||
|
if (IsHotkey(HK_BP_ADD))
|
||||||
|
emit AddBreakpoint();
|
||||||
|
|
||||||
|
// TODO: HK_MBP_ADD
|
||||||
|
|
||||||
if (SConfig::GetInstance().bWii)
|
if (SConfig::GetInstance().bWii)
|
||||||
{
|
{
|
||||||
|
|
|
@ -33,6 +33,17 @@ signals:
|
||||||
void ToggleReadOnlyMode();
|
void ToggleReadOnlyMode();
|
||||||
void ConnectWiiRemote(int id);
|
void ConnectWiiRemote(int id);
|
||||||
|
|
||||||
|
void Step();
|
||||||
|
void StepOver();
|
||||||
|
void StepOut();
|
||||||
|
void Skip();
|
||||||
|
|
||||||
|
void ShowPC();
|
||||||
|
void SetPC();
|
||||||
|
|
||||||
|
void ToggleBreakpoint();
|
||||||
|
void AddBreakpoint();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void Run();
|
void Run();
|
||||||
|
|
||||||
|
|
|
@ -50,6 +50,7 @@
|
||||||
#include "DolphinQt2/Config/Mapping/MappingWindow.h"
|
#include "DolphinQt2/Config/Mapping/MappingWindow.h"
|
||||||
#include "DolphinQt2/Config/SettingsWindow.h"
|
#include "DolphinQt2/Config/SettingsWindow.h"
|
||||||
#include "DolphinQt2/Debugger/BreakpointWidget.h"
|
#include "DolphinQt2/Debugger/BreakpointWidget.h"
|
||||||
|
#include "DolphinQt2/Debugger/CodeWidget.h"
|
||||||
#include "DolphinQt2/Debugger/RegisterWidget.h"
|
#include "DolphinQt2/Debugger/RegisterWidget.h"
|
||||||
#include "DolphinQt2/Debugger/WatchWidget.h"
|
#include "DolphinQt2/Debugger/WatchWidget.h"
|
||||||
#include "DolphinQt2/FIFOPlayerWindow.h"
|
#include "DolphinQt2/FIFOPlayerWindow.h"
|
||||||
|
@ -93,6 +94,7 @@ MainWindow::MainWindow(std::unique_ptr<BootParameters> boot_parameters) : QMainW
|
||||||
ConnectRenderWidget();
|
ConnectRenderWidget();
|
||||||
ConnectStack();
|
ConnectStack();
|
||||||
ConnectMenuBar();
|
ConnectMenuBar();
|
||||||
|
ConnectHotkeys();
|
||||||
|
|
||||||
InitCoreCallbacks();
|
InitCoreCallbacks();
|
||||||
|
|
||||||
|
@ -121,8 +123,6 @@ void MainWindow::InitControllers()
|
||||||
Wiimote::Initialize(Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES);
|
Wiimote::Initialize(Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES);
|
||||||
m_hotkey_scheduler = new HotkeyScheduler();
|
m_hotkey_scheduler = new HotkeyScheduler();
|
||||||
m_hotkey_scheduler->Start();
|
m_hotkey_scheduler->Start();
|
||||||
|
|
||||||
ConnectHotkeys();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::ShutdownControllers()
|
void MainWindow::ShutdownControllers()
|
||||||
|
@ -195,11 +195,14 @@ void MainWindow::CreateComponents()
|
||||||
m_register_widget = new RegisterWidget(this);
|
m_register_widget = new RegisterWidget(this);
|
||||||
m_watch_widget = new WatchWidget(this);
|
m_watch_widget = new WatchWidget(this);
|
||||||
m_breakpoint_widget = new BreakpointWidget(this);
|
m_breakpoint_widget = new BreakpointWidget(this);
|
||||||
|
m_code_widget = new CodeWidget(this);
|
||||||
|
|
||||||
connect(m_watch_widget, &WatchWidget::RequestMemoryBreakpoint,
|
connect(m_watch_widget, &WatchWidget::RequestMemoryBreakpoint,
|
||||||
[this](u32 addr) { m_breakpoint_widget->AddAddressMBP(addr); });
|
[this](u32 addr) { m_breakpoint_widget->AddAddressMBP(addr); });
|
||||||
connect(m_register_widget, &RegisterWidget::RequestMemoryBreakpoint,
|
connect(m_register_widget, &RegisterWidget::RequestMemoryBreakpoint,
|
||||||
[this](u32 addr) { m_breakpoint_widget->AddAddressMBP(addr); });
|
[this](u32 addr) { m_breakpoint_widget->AddAddressMBP(addr); });
|
||||||
|
connect(m_code_widget, &CodeWidget::BreakpointsChanged, m_breakpoint_widget,
|
||||||
|
&BreakpointWidget::Update);
|
||||||
|
|
||||||
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
#if defined(HAVE_XRANDR) && HAVE_XRANDR
|
||||||
m_graphics_window = new GraphicsWindow(
|
m_graphics_window = new GraphicsWindow(
|
||||||
|
@ -312,11 +315,27 @@ void MainWindow::ConnectHotkeys()
|
||||||
Movie::SetReadOnly(read_only);
|
Movie::SetReadOnly(read_only);
|
||||||
emit ReadOnlyModeChanged(read_only);
|
emit ReadOnlyModeChanged(read_only);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::Step, m_code_widget, &CodeWidget::Step);
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::StepOver, m_code_widget, &CodeWidget::StepOver);
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::StepOut, m_code_widget, &CodeWidget::StepOut);
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::Skip, m_code_widget, &CodeWidget::Skip);
|
||||||
|
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::ShowPC, m_code_widget, &CodeWidget::ShowPC);
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::SetPC, m_code_widget, &CodeWidget::SetPC);
|
||||||
|
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::ToggleBreakpoint, m_code_widget,
|
||||||
|
&CodeWidget::ToggleBreakpoint);
|
||||||
|
connect(m_hotkey_scheduler, &HotkeyScheduler::AddBreakpoint, m_code_widget,
|
||||||
|
&CodeWidget::AddBreakpoint);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::ConnectToolBar()
|
void MainWindow::ConnectToolBar()
|
||||||
{
|
{
|
||||||
addToolBar(m_tool_bar);
|
addToolBar(m_tool_bar);
|
||||||
|
|
||||||
|
connect(m_tool_bar, &ToolBar::OpenPressed, this, &MainWindow::Open);
|
||||||
|
|
||||||
connect(m_tool_bar, &ToolBar::OpenPressed, this, &MainWindow::Open);
|
connect(m_tool_bar, &ToolBar::OpenPressed, this, &MainWindow::Open);
|
||||||
connect(m_tool_bar, &ToolBar::PlayPressed, this, [this]() { Play(); });
|
connect(m_tool_bar, &ToolBar::PlayPressed, this, [this]() { Play(); });
|
||||||
connect(m_tool_bar, &ToolBar::PausePressed, this, &MainWindow::Pause);
|
connect(m_tool_bar, &ToolBar::PausePressed, this, &MainWindow::Pause);
|
||||||
|
@ -326,6 +345,13 @@ void MainWindow::ConnectToolBar()
|
||||||
connect(m_tool_bar, &ToolBar::SettingsPressed, this, &MainWindow::ShowSettingsWindow);
|
connect(m_tool_bar, &ToolBar::SettingsPressed, this, &MainWindow::ShowSettingsWindow);
|
||||||
connect(m_tool_bar, &ToolBar::ControllersPressed, this, &MainWindow::ShowControllersWindow);
|
connect(m_tool_bar, &ToolBar::ControllersPressed, this, &MainWindow::ShowControllersWindow);
|
||||||
connect(m_tool_bar, &ToolBar::GraphicsPressed, this, &MainWindow::ShowGraphicsWindow);
|
connect(m_tool_bar, &ToolBar::GraphicsPressed, this, &MainWindow::ShowGraphicsWindow);
|
||||||
|
|
||||||
|
connect(m_tool_bar, &ToolBar::StepPressed, m_code_widget, &CodeWidget::Step);
|
||||||
|
connect(m_tool_bar, &ToolBar::StepOverPressed, m_code_widget, &CodeWidget::StepOver);
|
||||||
|
connect(m_tool_bar, &ToolBar::StepOutPressed, m_code_widget, &CodeWidget::StepOut);
|
||||||
|
connect(m_tool_bar, &ToolBar::SkipPressed, m_code_widget, &CodeWidget::Skip);
|
||||||
|
connect(m_tool_bar, &ToolBar::ShowPCPressed, m_code_widget, &CodeWidget::ShowPC);
|
||||||
|
connect(m_tool_bar, &ToolBar::SetPCPressed, m_code_widget, &CodeWidget::SetPC);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::ConnectGameList()
|
void MainWindow::ConnectGameList()
|
||||||
|
@ -352,11 +378,13 @@ void MainWindow::ConnectStack()
|
||||||
setTabPosition(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea, QTabWidget::North);
|
setTabPosition(Qt::LeftDockWidgetArea | Qt::RightDockWidgetArea, QTabWidget::North);
|
||||||
addDockWidget(Qt::RightDockWidgetArea, m_log_widget);
|
addDockWidget(Qt::RightDockWidgetArea, m_log_widget);
|
||||||
addDockWidget(Qt::RightDockWidgetArea, m_log_config_widget);
|
addDockWidget(Qt::RightDockWidgetArea, m_log_config_widget);
|
||||||
|
addDockWidget(Qt::RightDockWidgetArea, m_code_widget);
|
||||||
addDockWidget(Qt::RightDockWidgetArea, m_register_widget);
|
addDockWidget(Qt::RightDockWidgetArea, m_register_widget);
|
||||||
addDockWidget(Qt::RightDockWidgetArea, m_watch_widget);
|
addDockWidget(Qt::RightDockWidgetArea, m_watch_widget);
|
||||||
addDockWidget(Qt::RightDockWidgetArea, m_breakpoint_widget);
|
addDockWidget(Qt::RightDockWidgetArea, m_breakpoint_widget);
|
||||||
|
|
||||||
tabifyDockWidget(m_log_widget, m_log_config_widget);
|
tabifyDockWidget(m_log_widget, m_log_config_widget);
|
||||||
|
tabifyDockWidget(m_log_widget, m_code_widget);
|
||||||
tabifyDockWidget(m_log_widget, m_register_widget);
|
tabifyDockWidget(m_log_widget, m_register_widget);
|
||||||
tabifyDockWidget(m_log_widget, m_watch_widget);
|
tabifyDockWidget(m_log_widget, m_watch_widget);
|
||||||
tabifyDockWidget(m_log_widget, m_breakpoint_widget);
|
tabifyDockWidget(m_log_widget, m_breakpoint_widget);
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
class BreakpointWidget;
|
class BreakpointWidget;
|
||||||
struct BootParameters;
|
struct BootParameters;
|
||||||
|
class CodeWidget;
|
||||||
class FIFOPlayerWindow;
|
class FIFOPlayerWindow;
|
||||||
class HotkeyScheduler;
|
class HotkeyScheduler;
|
||||||
class LogConfigWidget;
|
class LogConfigWidget;
|
||||||
|
@ -155,6 +156,7 @@ private:
|
||||||
std::array<WiiTASInputWindow*, num_wii_controllers> m_wii_tas_input_windows{};
|
std::array<WiiTASInputWindow*, num_wii_controllers> m_wii_tas_input_windows{};
|
||||||
|
|
||||||
BreakpointWidget* m_breakpoint_widget;
|
BreakpointWidget* m_breakpoint_widget;
|
||||||
|
CodeWidget* m_code_widget;
|
||||||
LogWidget* m_log_widget;
|
LogWidget* m_log_widget;
|
||||||
LogConfigWidget* m_log_config_widget;
|
LogConfigWidget* m_log_config_widget;
|
||||||
FIFOPlayerWindow* m_fifo_window;
|
FIFOPlayerWindow* m_fifo_window;
|
||||||
|
|
|
@ -9,6 +9,8 @@
|
||||||
#include <QAction>
|
#include <QAction>
|
||||||
#include <QDesktopServices>
|
#include <QDesktopServices>
|
||||||
#include <QFileDialog>
|
#include <QFileDialog>
|
||||||
|
#include <QFontDialog>
|
||||||
|
#include <QInputDialog>
|
||||||
#include <QMap>
|
#include <QMap>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QUrl>
|
#include <QUrl>
|
||||||
|
@ -17,15 +19,22 @@
|
||||||
#include "Common/FileUtil.h"
|
#include "Common/FileUtil.h"
|
||||||
#include "Common/StringUtil.h"
|
#include "Common/StringUtil.h"
|
||||||
|
|
||||||
|
#include "Core/Boot/Boot.h"
|
||||||
#include "Core/CommonTitles.h"
|
#include "Core/CommonTitles.h"
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
#include "Core/Core.h"
|
#include "Core/Core.h"
|
||||||
|
#include "Core/Debugger/RSO.h"
|
||||||
|
#include "Core/HLE/HLE.h"
|
||||||
#include "Core/HW/WiiSaveCrypted.h"
|
#include "Core/HW/WiiSaveCrypted.h"
|
||||||
#include "Core/HW/Wiimote.h"
|
#include "Core/HW/Wiimote.h"
|
||||||
|
#include "Core/Host.h"
|
||||||
#include "Core/IOS/ES/ES.h"
|
#include "Core/IOS/ES/ES.h"
|
||||||
#include "Core/IOS/IOS.h"
|
#include "Core/IOS/IOS.h"
|
||||||
#include "Core/IOS/USB/Bluetooth/BTEmu.h"
|
#include "Core/IOS/USB/Bluetooth/BTEmu.h"
|
||||||
#include "Core/Movie.h"
|
#include "Core/Movie.h"
|
||||||
|
#include "Core/PowerPC/PPCAnalyst.h"
|
||||||
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
||||||
|
#include "Core/PowerPC/SignatureDB/SignatureDB.h"
|
||||||
#include "Core/State.h"
|
#include "Core/State.h"
|
||||||
#include "Core/TitleDatabase.h"
|
#include "Core/TitleDatabase.h"
|
||||||
#include "Core/WiiUtils.h"
|
#include "Core/WiiUtils.h"
|
||||||
|
@ -46,6 +55,7 @@ MenuBar::MenuBar(QWidget* parent) : QMenuBar(parent)
|
||||||
AddOptionsMenu();
|
AddOptionsMenu();
|
||||||
AddToolsMenu();
|
AddToolsMenu();
|
||||||
AddViewMenu();
|
AddViewMenu();
|
||||||
|
AddSymbolsMenu();
|
||||||
AddHelpMenu();
|
AddHelpMenu();
|
||||||
|
|
||||||
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
|
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
|
||||||
|
@ -83,6 +93,9 @@ void MenuBar::OnEmulationStateChanged(Core::State state)
|
||||||
m_recording_stop->setEnabled(false);
|
m_recording_stop->setEnabled(false);
|
||||||
m_recording_play->setEnabled(!running);
|
m_recording_play->setEnabled(!running);
|
||||||
|
|
||||||
|
// Symbols
|
||||||
|
m_symbols->setEnabled(running);
|
||||||
|
|
||||||
UpdateStateSlotMenu();
|
UpdateStateSlotMenu();
|
||||||
UpdateToolsMenu(running);
|
UpdateToolsMenu(running);
|
||||||
|
|
||||||
|
@ -91,9 +104,21 @@ void MenuBar::OnEmulationStateChanged(Core::State state)
|
||||||
|
|
||||||
void MenuBar::OnDebugModeToggled(bool enabled)
|
void MenuBar::OnDebugModeToggled(bool enabled)
|
||||||
{
|
{
|
||||||
|
// Options
|
||||||
|
m_boot_to_pause->setVisible(enabled);
|
||||||
|
m_automatic_start->setVisible(enabled);
|
||||||
|
m_change_font->setVisible(enabled);
|
||||||
|
|
||||||
|
// View
|
||||||
|
m_show_code->setVisible(enabled);
|
||||||
m_show_registers->setVisible(enabled);
|
m_show_registers->setVisible(enabled);
|
||||||
m_show_watch->setVisible(enabled);
|
m_show_watch->setVisible(enabled);
|
||||||
m_show_breakpoints->setVisible(enabled);
|
m_show_breakpoints->setVisible(enabled);
|
||||||
|
|
||||||
|
if (enabled)
|
||||||
|
addMenu(m_symbols);
|
||||||
|
else
|
||||||
|
removeAction(m_symbols->menuAction());
|
||||||
}
|
}
|
||||||
|
|
||||||
void MenuBar::AddFileMenu()
|
void MenuBar::AddFileMenu()
|
||||||
|
@ -289,6 +314,14 @@ void MenuBar::AddViewMenu()
|
||||||
|
|
||||||
view_menu->addSeparator();
|
view_menu->addSeparator();
|
||||||
|
|
||||||
|
m_show_code = view_menu->addAction(tr("&Code"));
|
||||||
|
m_show_code->setCheckable(true);
|
||||||
|
m_show_code->setChecked(Settings::Instance().IsCodeVisible());
|
||||||
|
|
||||||
|
connect(m_show_code, &QAction::toggled, &Settings::Instance(), &Settings::SetCodeVisible);
|
||||||
|
connect(&Settings::Instance(), &Settings::CodeVisibilityChanged, m_show_code,
|
||||||
|
&QAction::setChecked);
|
||||||
|
|
||||||
m_show_registers = view_menu->addAction(tr("&Registers"));
|
m_show_registers = view_menu->addAction(tr("&Registers"));
|
||||||
m_show_registers->setCheckable(true);
|
m_show_registers->setCheckable(true);
|
||||||
m_show_registers->setChecked(Settings::Instance().IsRegistersVisible());
|
m_show_registers->setChecked(Settings::Instance().IsRegistersVisible());
|
||||||
|
@ -334,6 +367,25 @@ void MenuBar::AddOptionsMenu()
|
||||||
AddAction(options_menu, tr("&Audio Settings"), this, &MenuBar::ConfigureAudio);
|
AddAction(options_menu, tr("&Audio Settings"), this, &MenuBar::ConfigureAudio);
|
||||||
AddAction(options_menu, tr("&Controller Settings"), this, &MenuBar::ConfigureControllers);
|
AddAction(options_menu, tr("&Controller Settings"), this, &MenuBar::ConfigureControllers);
|
||||||
AddAction(options_menu, tr("&Hotkey Settings"), this, &MenuBar::ConfigureHotkeys);
|
AddAction(options_menu, tr("&Hotkey Settings"), this, &MenuBar::ConfigureHotkeys);
|
||||||
|
|
||||||
|
options_menu->addSeparator();
|
||||||
|
|
||||||
|
// Debugging mode only
|
||||||
|
m_boot_to_pause = options_menu->addAction(tr("Boot To Pause"));
|
||||||
|
m_boot_to_pause->setCheckable(true);
|
||||||
|
m_boot_to_pause->setChecked(SConfig::GetInstance().bBootToPause);
|
||||||
|
|
||||||
|
connect(m_boot_to_pause, &QAction::toggled, this,
|
||||||
|
[this](bool enable) { SConfig::GetInstance().bBootToPause = enable; });
|
||||||
|
|
||||||
|
m_automatic_start = options_menu->addAction(tr("&Automatic Start"));
|
||||||
|
m_automatic_start->setCheckable(true);
|
||||||
|
m_automatic_start->setChecked(SConfig::GetInstance().bAutomaticStart);
|
||||||
|
|
||||||
|
connect(m_automatic_start, &QAction::toggled, this,
|
||||||
|
[this](bool enable) { SConfig::GetInstance().bAutomaticStart = enable; });
|
||||||
|
|
||||||
|
m_change_font = AddAction(options_menu, tr("Font..."), this, &MenuBar::ChangeDebugFont);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MenuBar::AddHelpMenu()
|
void MenuBar::AddHelpMenu()
|
||||||
|
@ -537,6 +589,35 @@ void MenuBar::AddMovieMenu()
|
||||||
[](bool value) { SConfig::GetInstance().m_DumpAudio = value; });
|
[](bool value) { SConfig::GetInstance().m_DumpAudio = value; });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MenuBar::AddSymbolsMenu()
|
||||||
|
{
|
||||||
|
m_symbols = addMenu(tr("Symbols"));
|
||||||
|
|
||||||
|
AddAction(m_symbols, tr("&Clear Symbols"), this, &MenuBar::ClearSymbols);
|
||||||
|
|
||||||
|
auto* generate = m_symbols->addMenu(tr("Generate Symbols From"));
|
||||||
|
AddAction(generate, tr("Address"), this, &MenuBar::GenerateSymbolsFromAddress);
|
||||||
|
AddAction(generate, tr("Signature Database"), this, &MenuBar::GenerateSymbolsFromSignatureDB);
|
||||||
|
AddAction(generate, tr("RSO Modules"), this, &MenuBar::GenerateSymbolsFromRSO);
|
||||||
|
m_symbols->addSeparator();
|
||||||
|
|
||||||
|
AddAction(m_symbols, tr("&Load Symbol Map"), this, &MenuBar::LoadSymbolMap);
|
||||||
|
AddAction(m_symbols, tr("&Save Symbol Map"), this, &MenuBar::SaveSymbolMap);
|
||||||
|
m_symbols->addSeparator();
|
||||||
|
|
||||||
|
AddAction(m_symbols, tr("&Load &Other Map File..."), this, &MenuBar::LoadOtherSymbolMap);
|
||||||
|
AddAction(m_symbols, tr("Save Symbol Map &As..."), this, &MenuBar::SaveSymbolMapAs);
|
||||||
|
m_symbols->addSeparator();
|
||||||
|
|
||||||
|
AddAction(m_symbols, tr("Save Code"), this, &MenuBar::SaveCode);
|
||||||
|
m_symbols->addSeparator();
|
||||||
|
|
||||||
|
AddAction(m_symbols, tr("&Create Signature File..."), this, &MenuBar::CreateSignatureFile);
|
||||||
|
m_symbols->addSeparator();
|
||||||
|
|
||||||
|
AddAction(m_symbols, tr("&Patch HLE Functions"), this, &MenuBar::PatchHLEFunctions);
|
||||||
|
}
|
||||||
|
|
||||||
void MenuBar::UpdateToolsMenu(bool emulation_started)
|
void MenuBar::UpdateToolsMenu(bool emulation_started)
|
||||||
{
|
{
|
||||||
m_boot_sysmenu->setEnabled(!emulation_started);
|
m_boot_sysmenu->setEnabled(!emulation_started);
|
||||||
|
@ -716,3 +797,181 @@ void MenuBar::OnReadOnlyModeChanged(bool read_only)
|
||||||
{
|
{
|
||||||
m_recording_read_only->setChecked(read_only);
|
m_recording_read_only->setChecked(read_only);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MenuBar::ChangeDebugFont()
|
||||||
|
{
|
||||||
|
bool okay;
|
||||||
|
QFont font = QFontDialog::getFont(&okay, Settings::Instance().GetDebugFont(), this,
|
||||||
|
tr("Pick a debug font"));
|
||||||
|
|
||||||
|
if (okay)
|
||||||
|
Settings::Instance().SetDebugFont(font);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::ClearSymbols()
|
||||||
|
{
|
||||||
|
auto result = QMessageBox::warning(this, tr("Confirmation"),
|
||||||
|
tr("Do you want to clear the list of symbol names?"),
|
||||||
|
QMessageBox::Yes | QMessageBox::Cancel);
|
||||||
|
|
||||||
|
if (result == QMessageBox::Cancel)
|
||||||
|
return;
|
||||||
|
|
||||||
|
g_symbolDB.Clear();
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::GenerateSymbolsFromAddress()
|
||||||
|
{
|
||||||
|
PPCAnalyst::FindFunctions(0x80000000, 0x81800000, &g_symbolDB);
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::GenerateSymbolsFromSignatureDB()
|
||||||
|
{
|
||||||
|
PPCAnalyst::FindFunctions(0x80000000, 0x81800000, &g_symbolDB);
|
||||||
|
SignatureDB db(SignatureDB::HandlerType::DSY);
|
||||||
|
if (db.Load(File::GetSysDirectory() + TOTALDB))
|
||||||
|
{
|
||||||
|
db.Apply(&g_symbolDB);
|
||||||
|
QMessageBox::information(
|
||||||
|
this, tr("Information"),
|
||||||
|
tr("Generated symbol names from '%1'").arg(QString::fromStdString(TOTALDB)));
|
||||||
|
db.List();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QMessageBox::critical(
|
||||||
|
this, tr("Error"),
|
||||||
|
tr("'%1' not found, no symbol names generated").arg(QString::fromStdString(TOTALDB)));
|
||||||
|
}
|
||||||
|
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::GenerateSymbolsFromRSO()
|
||||||
|
{
|
||||||
|
QString text = QInputDialog::getText(this, tr("Input"), tr("Enter the RSO module address:"));
|
||||||
|
bool good;
|
||||||
|
uint address = text.toUInt(&good, 16);
|
||||||
|
|
||||||
|
if (!good)
|
||||||
|
{
|
||||||
|
QMessageBox::warning(this, tr("Error"), tr("Invalid RSO module address: %1").arg(text));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
RSOChainView rso_chain;
|
||||||
|
if (rso_chain.Load(static_cast<u32>(address)))
|
||||||
|
{
|
||||||
|
rso_chain.Apply(&g_symbolDB);
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QMessageBox::warning(this, tr("Error"), tr("Failed to load RSO module at %1").arg(text));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::LoadSymbolMap()
|
||||||
|
{
|
||||||
|
std::string existing_map_file, writable_map_file;
|
||||||
|
bool map_exists = CBoot::FindMapFile(&existing_map_file, &writable_map_file);
|
||||||
|
|
||||||
|
if (!map_exists)
|
||||||
|
{
|
||||||
|
g_symbolDB.Clear();
|
||||||
|
PPCAnalyst::FindFunctions(0x81300000, 0x81800000, &g_symbolDB);
|
||||||
|
SignatureDB db(SignatureDB::HandlerType::DSY);
|
||||||
|
if (db.Load(File::GetSysDirectory() + TOTALDB))
|
||||||
|
db.Apply(&g_symbolDB);
|
||||||
|
|
||||||
|
QMessageBox::warning(this, tr("Warning"),
|
||||||
|
tr("'%1' not found, scanning for common functions instead")
|
||||||
|
.arg(QString::fromStdString(writable_map_file)));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
g_symbolDB.LoadMap(existing_map_file);
|
||||||
|
QMessageBox::information(
|
||||||
|
this, tr("Information"),
|
||||||
|
tr("Loaded symbols from '%1'").arg(QString::fromStdString(existing_map_file.c_str())));
|
||||||
|
}
|
||||||
|
|
||||||
|
HLE::PatchFunctions();
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::SaveSymbolMap()
|
||||||
|
{
|
||||||
|
std::string existing_map_file, writable_map_file;
|
||||||
|
CBoot::FindMapFile(&existing_map_file, &writable_map_file);
|
||||||
|
|
||||||
|
g_symbolDB.SaveSymbolMap(writable_map_file);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::LoadOtherSymbolMap()
|
||||||
|
{
|
||||||
|
QString file = QFileDialog::getOpenFileName(this, tr("Load map file"),
|
||||||
|
QString::fromStdString(File::GetUserPath(D_MAPS_IDX)),
|
||||||
|
tr("Dolphin Map File (*.map)"));
|
||||||
|
|
||||||
|
if (file.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
g_symbolDB.LoadMap(file.toStdString());
|
||||||
|
HLE::PatchFunctions();
|
||||||
|
Host_NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::SaveSymbolMapAs()
|
||||||
|
{
|
||||||
|
const std::string& title_id_str = SConfig::GetInstance().m_debugger_game_id;
|
||||||
|
QString file = QFileDialog::getSaveFileName(
|
||||||
|
this, tr("Save map file"),
|
||||||
|
QString::fromStdString(File::GetUserPath(D_MAPS_IDX) + "/" + title_id_str + ".map"),
|
||||||
|
tr("Dolphin Map File (*.map)"));
|
||||||
|
|
||||||
|
if (file.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
g_symbolDB.SaveSymbolMap(file.toStdString());
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::SaveCode()
|
||||||
|
{
|
||||||
|
std::string existing_map_file, writable_map_file;
|
||||||
|
CBoot::FindMapFile(&existing_map_file, &writable_map_file);
|
||||||
|
|
||||||
|
const std::string path =
|
||||||
|
writable_map_file.substr(0, writable_map_file.find_last_of(".")) + "_code.map";
|
||||||
|
|
||||||
|
g_symbolDB.SaveCodeMap(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::CreateSignatureFile()
|
||||||
|
{
|
||||||
|
QString text = QInputDialog::getText(
|
||||||
|
this, tr("Input"), tr("Only export symbols with prefix:\n(Blank for all symbols)"));
|
||||||
|
|
||||||
|
if (text.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::string prefix = text.toStdString();
|
||||||
|
|
||||||
|
QString file = QFileDialog::getSaveFileName(this, tr("Save signature file"));
|
||||||
|
|
||||||
|
if (file.isEmpty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::string save_path = file.toStdString();
|
||||||
|
SignatureDB db(save_path);
|
||||||
|
db.Populate(&g_symbolDB, prefix);
|
||||||
|
db.Save(save_path);
|
||||||
|
db.List();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MenuBar::PatchHLEFunctions()
|
||||||
|
{
|
||||||
|
HLE::PatchFunctions();
|
||||||
|
}
|
||||||
|
|
|
@ -113,12 +113,27 @@ private:
|
||||||
void AddToolsMenu();
|
void AddToolsMenu();
|
||||||
void AddHelpMenu();
|
void AddHelpMenu();
|
||||||
void AddMovieMenu();
|
void AddMovieMenu();
|
||||||
|
void AddSymbolsMenu();
|
||||||
|
|
||||||
void InstallWAD();
|
void InstallWAD();
|
||||||
void ImportWiiSave();
|
void ImportWiiSave();
|
||||||
void ExportWiiSaves();
|
void ExportWiiSaves();
|
||||||
void CheckNAND();
|
void CheckNAND();
|
||||||
void NANDExtractCertificates();
|
void NANDExtractCertificates();
|
||||||
|
void ChangeDebugFont();
|
||||||
|
|
||||||
|
// Debugging UI
|
||||||
|
void ClearSymbols();
|
||||||
|
void GenerateSymbolsFromAddress();
|
||||||
|
void GenerateSymbolsFromSignatureDB();
|
||||||
|
void GenerateSymbolsFromRSO();
|
||||||
|
void LoadSymbolMap();
|
||||||
|
void LoadOtherSymbolMap();
|
||||||
|
void SaveSymbolMap();
|
||||||
|
void SaveSymbolMapAs();
|
||||||
|
void SaveCode();
|
||||||
|
void CreateSignatureFile();
|
||||||
|
void PatchHLEFunctions();
|
||||||
|
|
||||||
void OnSelectionChanged(QSharedPointer<GameFile> game_file);
|
void OnSelectionChanged(QSharedPointer<GameFile> game_file);
|
||||||
void OnRecordingStatusChanged(bool recording);
|
void OnRecordingStatusChanged(bool recording);
|
||||||
|
@ -164,8 +179,17 @@ private:
|
||||||
QAction* m_recording_stop;
|
QAction* m_recording_stop;
|
||||||
QAction* m_recording_read_only;
|
QAction* m_recording_read_only;
|
||||||
|
|
||||||
|
// Options
|
||||||
|
QAction* m_boot_to_pause;
|
||||||
|
QAction* m_automatic_start;
|
||||||
|
QAction* m_change_font;
|
||||||
|
|
||||||
// View
|
// View
|
||||||
|
QAction* m_show_code;
|
||||||
QAction* m_show_registers;
|
QAction* m_show_registers;
|
||||||
QAction* m_show_watch;
|
QAction* m_show_watch;
|
||||||
QAction* m_show_breakpoints;
|
QAction* m_show_breakpoints;
|
||||||
|
|
||||||
|
// Symbols
|
||||||
|
QMenu* m_symbols;
|
||||||
};
|
};
|
||||||
|
|
|
@ -279,3 +279,36 @@ void Settings::SetControllerStateNeeded(bool needed)
|
||||||
{
|
{
|
||||||
m_controller_state_needed = needed;
|
m_controller_state_needed = needed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Settings::SetCodeVisible(bool enabled)
|
||||||
|
{
|
||||||
|
if (IsCodeVisible() != enabled)
|
||||||
|
{
|
||||||
|
QSettings().setValue(QStringLiteral("debugger/showcode"), enabled);
|
||||||
|
|
||||||
|
emit CodeVisibilityChanged(enabled);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Settings::IsCodeVisible() const
|
||||||
|
{
|
||||||
|
return QSettings().value(QStringLiteral("debugger/showcode")).toBool();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Settings::SetDebugFont(QFont font)
|
||||||
|
{
|
||||||
|
if (GetDebugFont() != font)
|
||||||
|
{
|
||||||
|
QSettings().setValue(QStringLiteral("debugger/font"), font);
|
||||||
|
|
||||||
|
emit DebugFontChanged(font);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QFont Settings::GetDebugFont() const
|
||||||
|
{
|
||||||
|
QFont default_font = QFont(QStringLiteral("Monospace"));
|
||||||
|
default_font.setStyleHint(QFont::TypeWriter);
|
||||||
|
|
||||||
|
return QSettings().value(QStringLiteral("debugger/font"), default_font).value<QFont>();
|
||||||
|
}
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
|
#include <QFont>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QVector>
|
#include <QVector>
|
||||||
|
|
||||||
|
@ -24,6 +25,7 @@ enum class Language;
|
||||||
|
|
||||||
class GameListModel;
|
class GameListModel;
|
||||||
class InputConfig;
|
class InputConfig;
|
||||||
|
class QFont;
|
||||||
|
|
||||||
// UI settings to be stored in the config directory.
|
// UI settings to be stored in the config directory.
|
||||||
class Settings final : public QObject
|
class Settings final : public QObject
|
||||||
|
@ -91,6 +93,10 @@ public:
|
||||||
bool IsWatchVisible() const;
|
bool IsWatchVisible() const;
|
||||||
void SetBreakpointsVisible(bool enabled);
|
void SetBreakpointsVisible(bool enabled);
|
||||||
bool IsBreakpointsVisible() const;
|
bool IsBreakpointsVisible() const;
|
||||||
|
void SetCodeVisible(bool enabled);
|
||||||
|
bool IsCodeVisible() const;
|
||||||
|
QFont GetDebugFont() const;
|
||||||
|
void SetDebugFont(QFont font);
|
||||||
|
|
||||||
// Other
|
// Other
|
||||||
GameListModel* GetGameListModel() const;
|
GameListModel* GetGameListModel() const;
|
||||||
|
@ -110,7 +116,9 @@ signals:
|
||||||
void EnableCheatsChanged(bool enabled);
|
void EnableCheatsChanged(bool enabled);
|
||||||
void WatchVisibilityChanged(bool visible);
|
void WatchVisibilityChanged(bool visible);
|
||||||
void BreakpointsVisibilityChanged(bool visible);
|
void BreakpointsVisibilityChanged(bool visible);
|
||||||
|
void CodeVisibilityChanged(bool visible);
|
||||||
void DebugModeToggled(bool enabled);
|
void DebugModeToggled(bool enabled);
|
||||||
|
void DebugFontChanged(QFont font);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
bool m_controller_state_needed = false;
|
bool m_controller_state_needed = false;
|
||||||
|
|
|
@ -28,7 +28,11 @@ ToolBar::ToolBar(QWidget* parent) : QToolBar(parent)
|
||||||
|
|
||||||
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
|
connect(&Settings::Instance(), &Settings::EmulationStateChanged, this,
|
||||||
[this](Core::State state) { OnEmulationStateChanged(state); });
|
[this](Core::State state) { OnEmulationStateChanged(state); });
|
||||||
|
|
||||||
|
connect(&Settings::Instance(), &Settings::DebugModeToggled, this, &ToolBar::OnDebugModeToggled);
|
||||||
|
|
||||||
OnEmulationStateChanged(Core::GetState());
|
OnEmulationStateChanged(Core::GetState());
|
||||||
|
OnDebugModeToggled(Settings::Instance().IsDebugModeEnabled());
|
||||||
}
|
}
|
||||||
|
|
||||||
void ToolBar::OnEmulationStateChanged(Core::State state)
|
void ToolBar::OnEmulationStateChanged(Core::State state)
|
||||||
|
@ -45,8 +49,25 @@ void ToolBar::OnEmulationStateChanged(Core::State state)
|
||||||
m_pause_action->setVisible(playing);
|
m_pause_action->setVisible(playing);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ToolBar::OnDebugModeToggled(bool enabled)
|
||||||
|
{
|
||||||
|
m_step_action->setVisible(enabled);
|
||||||
|
m_step_over_action->setVisible(enabled);
|
||||||
|
m_step_out_action->setVisible(enabled);
|
||||||
|
m_skip_action->setVisible(enabled);
|
||||||
|
m_show_pc_action->setVisible(enabled);
|
||||||
|
m_set_pc_action->setVisible(enabled);
|
||||||
|
}
|
||||||
|
|
||||||
void ToolBar::MakeActions()
|
void ToolBar::MakeActions()
|
||||||
{
|
{
|
||||||
|
m_step_action = AddAction(this, tr("Step"), this, &ToolBar::StepPressed);
|
||||||
|
m_step_over_action = AddAction(this, tr("Step Over"), this, &ToolBar::StepOverPressed);
|
||||||
|
m_step_out_action = AddAction(this, tr("Step Out"), this, &ToolBar::StepOutPressed);
|
||||||
|
m_skip_action = AddAction(this, tr("Skip"), this, &ToolBar::SkipPressed);
|
||||||
|
m_show_pc_action = AddAction(this, tr("Show PC"), this, &ToolBar::ShowPCPressed);
|
||||||
|
m_set_pc_action = AddAction(this, tr("Set PC"), this, &ToolBar::SetPCPressed);
|
||||||
|
|
||||||
m_open_action = AddAction(this, tr("Open"), this, &ToolBar::OpenPressed);
|
m_open_action = AddAction(this, tr("Open"), this, &ToolBar::OpenPressed);
|
||||||
m_play_action = AddAction(this, tr("Play"), this, &ToolBar::PlayPressed);
|
m_play_action = AddAction(this, tr("Play"), this, &ToolBar::PlayPressed);
|
||||||
m_pause_action = AddAction(this, tr("Pause"), this, &ToolBar::PausePressed);
|
m_pause_action = AddAction(this, tr("Pause"), this, &ToolBar::PausePressed);
|
||||||
|
@ -63,9 +84,11 @@ void ToolBar::MakeActions()
|
||||||
|
|
||||||
// Ensure every button has about the same width
|
// Ensure every button has about the same width
|
||||||
std::vector<QWidget*> items;
|
std::vector<QWidget*> items;
|
||||||
for (const auto& action : {m_open_action, m_play_action, m_pause_action, m_stop_action,
|
for (const auto& action :
|
||||||
m_stop_action, m_fullscreen_action, m_screenshot_action,
|
{m_open_action, m_play_action, m_pause_action, m_stop_action, m_stop_action,
|
||||||
m_config_action, m_graphics_action, m_controllers_action})
|
m_fullscreen_action, m_screenshot_action, m_config_action, m_graphics_action,
|
||||||
|
m_controllers_action, m_step_action, m_step_over_action, m_step_out_action, m_skip_action,
|
||||||
|
m_show_pc_action, m_set_pc_action})
|
||||||
{
|
{
|
||||||
items.emplace_back(widgetForAction(action));
|
items.emplace_back(widgetForAction(action));
|
||||||
}
|
}
|
||||||
|
|
|
@ -32,8 +32,16 @@ signals:
|
||||||
void ControllersPressed();
|
void ControllersPressed();
|
||||||
void GraphicsPressed();
|
void GraphicsPressed();
|
||||||
|
|
||||||
|
void StepPressed();
|
||||||
|
void StepOverPressed();
|
||||||
|
void StepOutPressed();
|
||||||
|
void SkipPressed();
|
||||||
|
void ShowPCPressed();
|
||||||
|
void SetPCPressed();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void OnEmulationStateChanged(Core::State state);
|
void OnEmulationStateChanged(Core::State state);
|
||||||
|
void OnDebugModeToggled(bool enabled);
|
||||||
|
|
||||||
void MakeActions();
|
void MakeActions();
|
||||||
void UpdateIcons();
|
void UpdateIcons();
|
||||||
|
@ -47,4 +55,11 @@ private:
|
||||||
QAction* m_config_action;
|
QAction* m_config_action;
|
||||||
QAction* m_controllers_action;
|
QAction* m_controllers_action;
|
||||||
QAction* m_graphics_action;
|
QAction* m_graphics_action;
|
||||||
|
|
||||||
|
QAction* m_step_action;
|
||||||
|
QAction* m_step_over_action;
|
||||||
|
QAction* m_step_out_action;
|
||||||
|
QAction* m_skip_action;
|
||||||
|
QAction* m_show_pc_action;
|
||||||
|
QAction* m_set_pc_action;
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue