mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-19 20:41:16 +01:00
284 lines
8.1 KiB
C++
284 lines
8.1 KiB
C++
// Copyright 2009 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <cstdio>
|
|
#include <wx/artprov.h>
|
|
#include <wx/aui/auibook.h>
|
|
#include <wx/aui/dockart.h>
|
|
#include <wx/aui/framemanager.h>
|
|
#include <wx/listbox.h>
|
|
#include <wx/panel.h>
|
|
#include <wx/sizer.h>
|
|
#include <wx/textctrl.h>
|
|
#include <wx/thread.h>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "Common/SymbolDB.h"
|
|
#include "Core/DSP/DSPCore.h"
|
|
#include "Core/HW/DSPLLE/DSPDebugInterface.h"
|
|
#include "Core/HW/DSPLLE/DSPSymbols.h"
|
|
#include "Core/Host.h"
|
|
#include "DolphinWX/AuiToolBar.h"
|
|
#include "DolphinWX/Debugger/CodeView.h"
|
|
#include "DolphinWX/Debugger/DSPDebugWindow.h"
|
|
#include "DolphinWX/Debugger/DSPRegisterView.h"
|
|
#include "DolphinWX/Debugger/DebuggerUIUtil.h"
|
|
#include "DolphinWX/Debugger/MemoryView.h"
|
|
#include "DolphinWX/WxUtils.h"
|
|
|
|
static DSPDebuggerLLE* m_DebuggerFrame = nullptr;
|
|
|
|
DSPDebuggerLLE::DSPDebuggerLLE(wxWindow* parent, wxWindowID id)
|
|
: wxPanel(parent, id, wxDefaultPosition, wxDefaultSize, wxTAB_TRAVERSAL, _("DSP LLE Debugger")),
|
|
m_CachedStepCounter(UINT64_MAX), m_toolbar_item_size(FromDIP(wxSize(16, 16)))
|
|
{
|
|
Bind(wxEVT_MENU, &DSPDebuggerLLE::OnChangeState, this, ID_RUNTOOL, ID_SHOWPCTOOL);
|
|
|
|
m_DebuggerFrame = this;
|
|
|
|
// notify wxAUI which frame to use
|
|
m_mgr.SetManagedWindow(this);
|
|
m_mgr.SetFlags(wxAUI_MGR_DEFAULT | wxAUI_MGR_LIVE_RESIZE);
|
|
|
|
m_Toolbar =
|
|
new DolphinAuiToolBar(this, ID_TOOLBAR, wxDefaultPosition, wxDefaultSize, wxAUI_TB_HORZ_TEXT);
|
|
m_Toolbar->AddTool(ID_RUNTOOL, _("Pause"),
|
|
wxArtProvider::GetBitmap(wxART_TICK_MARK, wxART_OTHER, m_toolbar_item_size));
|
|
// i18n: Here, "Step" is a verb. This function is used for
|
|
// going through code step by step.
|
|
m_Toolbar->AddTool(ID_STEPTOOL, _("Step"),
|
|
wxArtProvider::GetBitmap(wxART_GO_DOWN, wxART_OTHER, m_toolbar_item_size));
|
|
m_Toolbar->AddTool(
|
|
// i18n: Here, PC is an acronym for program counter, not personal computer.
|
|
ID_SHOWPCTOOL, _("Show PC"),
|
|
wxArtProvider::GetBitmap(wxART_GO_TO_PARENT, wxART_OTHER, m_toolbar_item_size));
|
|
m_Toolbar->AddSeparator();
|
|
|
|
m_addr_txtctrl = new wxTextCtrl(m_Toolbar, wxID_ANY, wxEmptyString, wxDefaultPosition,
|
|
wxDefaultSize, wxTE_PROCESS_ENTER);
|
|
m_addr_txtctrl->Bind(wxEVT_TEXT_ENTER, &DSPDebuggerLLE::OnAddrBoxChange, this);
|
|
|
|
m_Toolbar->AddControl(m_addr_txtctrl);
|
|
m_Toolbar->Realize();
|
|
|
|
m_SymbolList = new wxListBox(this, wxID_ANY, wxDefaultPosition, wxDLG_UNIT(this, wxSize(100, 80)),
|
|
0, nullptr, wxLB_SORT);
|
|
m_SymbolList->Bind(wxEVT_LISTBOX, &DSPDebuggerLLE::OnSymbolListChange, this);
|
|
|
|
m_MainNotebook = new wxAuiNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize,
|
|
wxAUI_NB_TOP | wxAUI_NB_TAB_SPLIT | wxAUI_NB_TAB_MOVE);
|
|
|
|
wxPanel* code_panel = new wxPanel(m_MainNotebook, wxID_ANY);
|
|
wxBoxSizer* code_sizer = new wxBoxSizer(wxVERTICAL);
|
|
m_CodeView = new CCodeView(&debug_interface, &DSP::Symbols::g_dsp_symbol_db, code_panel);
|
|
m_CodeView->SetPlain();
|
|
code_sizer->Add(m_CodeView, 1, wxEXPAND);
|
|
code_panel->SetSizer(code_sizer);
|
|
m_MainNotebook->AddPage(code_panel, _("Disassembly"), true);
|
|
|
|
wxPanel* mem_panel = new wxPanel(m_MainNotebook, wxID_ANY);
|
|
wxBoxSizer* mem_sizer = new wxBoxSizer(wxVERTICAL);
|
|
m_MemView = new CMemoryView(&debug_interface, mem_panel);
|
|
mem_sizer->Add(m_MemView, 1, wxEXPAND);
|
|
mem_panel->SetSizer(mem_sizer);
|
|
m_MainNotebook->AddPage(mem_panel, _("Memory"));
|
|
|
|
m_Regs = new DSPRegisterView(this);
|
|
|
|
// add the panes to the manager
|
|
m_mgr.AddPane(m_Toolbar,
|
|
wxAuiPaneInfo().ToolbarPane().Top().LeftDockable(false).RightDockable(false));
|
|
|
|
m_mgr.AddPane(m_SymbolList,
|
|
wxAuiPaneInfo().Left().CloseButton(false).Caption(_("Symbols")).Dockable(true));
|
|
|
|
m_mgr.AddPane(
|
|
m_MainNotebook,
|
|
wxAuiPaneInfo().Name("m_MainNotebook").Center().CloseButton(false).MaximizeButton(true));
|
|
|
|
m_mgr.AddPane(m_Regs,
|
|
wxAuiPaneInfo().Right().CloseButton(false).Caption(_("Registers")).Dockable(true));
|
|
|
|
m_mgr.GetArtProvider()->SetFont(wxAUI_DOCKART_CAPTION_FONT, DebuggerFont);
|
|
UpdateState();
|
|
|
|
m_mgr.Update();
|
|
}
|
|
|
|
DSPDebuggerLLE::~DSPDebuggerLLE()
|
|
{
|
|
m_mgr.UnInit();
|
|
m_DebuggerFrame = nullptr;
|
|
}
|
|
|
|
void DSPDebuggerLLE::OnChangeState(wxCommandEvent& event)
|
|
{
|
|
const DSP::State dsp_state = DSP::DSPCore_GetState();
|
|
|
|
if (dsp_state == DSP::State::Stopped)
|
|
return;
|
|
|
|
switch (event.GetId())
|
|
{
|
|
case ID_RUNTOOL:
|
|
if (dsp_state == DSP::State::Running)
|
|
DSP::DSPCore_SetState(DSP::State::Stepping);
|
|
else
|
|
DSP::DSPCore_SetState(DSP::State::Running);
|
|
break;
|
|
|
|
case ID_STEPTOOL:
|
|
if (dsp_state == DSP::State::Stepping)
|
|
{
|
|
DSP::DSPCore_Step();
|
|
Repopulate();
|
|
}
|
|
break;
|
|
|
|
case ID_SHOWPCTOOL:
|
|
FocusOnPC();
|
|
break;
|
|
}
|
|
|
|
UpdateState();
|
|
m_mgr.Update();
|
|
}
|
|
|
|
void Host_RefreshDSPDebuggerWindow()
|
|
{
|
|
// FIXME: This should use QueueEvent to post the update request to the UI thread.
|
|
// Need to check if this can safely be performed asynchronously or if it races.
|
|
// FIXME: This probably belongs in Main.cpp with the other host functions.
|
|
// NOTE: The DSP never tells us when it shuts down. It probably should.
|
|
if (m_DebuggerFrame)
|
|
m_DebuggerFrame->Repopulate();
|
|
}
|
|
|
|
void DSPDebuggerLLE::Repopulate()
|
|
{
|
|
if (!wxIsMainThread())
|
|
wxMutexGuiEnter();
|
|
UpdateSymbolMap();
|
|
UpdateDisAsmListView();
|
|
UpdateRegisterFlags();
|
|
UpdateState();
|
|
if (!wxIsMainThread())
|
|
wxMutexGuiLeave();
|
|
}
|
|
|
|
void DSPDebuggerLLE::FocusOnPC()
|
|
{
|
|
JumpToAddress(DSP::g_dsp.pc);
|
|
}
|
|
|
|
void DSPDebuggerLLE::UpdateState()
|
|
{
|
|
if (DSP::DSPCore_GetState() == DSP::State::Running)
|
|
{
|
|
m_Toolbar->SetToolLabel(ID_RUNTOOL, _("Pause"));
|
|
m_Toolbar->SetToolBitmap(
|
|
ID_RUNTOOL, wxArtProvider::GetBitmap(wxART_TICK_MARK, wxART_OTHER, m_toolbar_item_size));
|
|
m_Toolbar->EnableTool(ID_STEPTOOL, false);
|
|
}
|
|
else
|
|
{
|
|
m_Toolbar->SetToolLabel(ID_RUNTOOL, _("Run"));
|
|
m_Toolbar->SetToolBitmap(
|
|
ID_RUNTOOL, wxArtProvider::GetBitmap(wxART_GO_FORWARD, wxART_OTHER, m_toolbar_item_size));
|
|
m_Toolbar->EnableTool(ID_STEPTOOL, true);
|
|
}
|
|
m_Toolbar->Realize();
|
|
}
|
|
|
|
void DSPDebuggerLLE::UpdateDisAsmListView()
|
|
{
|
|
if (m_CachedStepCounter == DSP::g_dsp.step_counter)
|
|
return;
|
|
|
|
// show PC
|
|
FocusOnPC();
|
|
m_CachedStepCounter = DSP::g_dsp.step_counter;
|
|
m_Regs->Repopulate();
|
|
}
|
|
|
|
void DSPDebuggerLLE::UpdateSymbolMap()
|
|
{
|
|
if (DSP::g_dsp.dram == nullptr)
|
|
return;
|
|
|
|
m_SymbolList->Freeze(); // HyperIris: wx style fast filling
|
|
m_SymbolList->Clear();
|
|
for (const auto& symbol : DSP::Symbols::g_dsp_symbol_db.Symbols())
|
|
{
|
|
int idx = m_SymbolList->Append(StrToWxStr(symbol.second.name));
|
|
m_SymbolList->SetClientData(idx, (void*)&symbol.second);
|
|
}
|
|
m_SymbolList->Thaw();
|
|
}
|
|
|
|
void DSPDebuggerLLE::OnSymbolListChange(wxCommandEvent& event)
|
|
{
|
|
int index = m_SymbolList->GetSelection();
|
|
if (index >= 0)
|
|
{
|
|
Symbol* pSymbol = static_cast<Symbol*>(m_SymbolList->GetClientData(index));
|
|
if (pSymbol != nullptr)
|
|
{
|
|
if (pSymbol->type == Symbol::Type::Function)
|
|
{
|
|
JumpToAddress(pSymbol->address);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void DSPDebuggerLLE::UpdateRegisterFlags()
|
|
{
|
|
}
|
|
|
|
void DSPDebuggerLLE::OnAddrBoxChange(wxCommandEvent& event)
|
|
{
|
|
wxString txt = m_addr_txtctrl->GetValue();
|
|
|
|
auto text = StripSpaces(WxStrToStr(txt));
|
|
if (text.size())
|
|
{
|
|
u32 addr;
|
|
sscanf(text.c_str(), "%04x", &addr);
|
|
if (JumpToAddress(addr))
|
|
m_addr_txtctrl->SetBackgroundColour(*wxWHITE);
|
|
else
|
|
m_addr_txtctrl->SetBackgroundColour(*wxRED);
|
|
}
|
|
event.Skip();
|
|
}
|
|
|
|
bool DSPDebuggerLLE::JumpToAddress(u16 addr)
|
|
{
|
|
int page = m_MainNotebook->GetSelection();
|
|
if (page == 0)
|
|
{
|
|
// Center on valid instruction in IRAM/IROM
|
|
int new_line = DSP::Symbols::Addr2Line(addr);
|
|
if (new_line >= 0)
|
|
{
|
|
m_CodeView->Center(new_line);
|
|
return true;
|
|
}
|
|
}
|
|
else if (page == 1)
|
|
{
|
|
// Center on any location in any valid ROM/RAM
|
|
int seg = addr >> 12;
|
|
if (seg == 0 || seg == 1 || seg == 8 || seg == 0xf)
|
|
{
|
|
m_MemView->Center(addr);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|