mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-11 16:49:12 +01:00
27d295ec7e
Changes: - MemoryWindow was cleaned up and gives more feedback on searches. Some bugs were fixed as well: - A complex bug that allowed tearing off tabs and opening multiple copies of a debug panel which lead to segfaults - Another segfault related to right-click menus on code/memory views when those tools were floating in their own window.
406 lines
12 KiB
C++
406 lines
12 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <cstddef>
|
|
#include <cstdio>
|
|
#include <string>
|
|
#include <vector>
|
|
#include <wx/button.h>
|
|
#include <wx/checkbox.h>
|
|
#include <wx/listbox.h>
|
|
#include <wx/msgdlg.h>
|
|
#include <wx/panel.h>
|
|
#include <wx/radiobox.h>
|
|
#include <wx/radiobut.h>
|
|
#include <wx/sizer.h>
|
|
#include <wx/srchctrl.h>
|
|
#include <wx/stattext.h>
|
|
#include <wx/textctrl.h>
|
|
#include <wx/utils.h>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/FileUtil.h"
|
|
#include "Common/IniFile.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "Common/SymbolDB.h"
|
|
#include "Core/ConfigManager.h"
|
|
#include "Core/Debugger/PPCDebugInterface.h"
|
|
#include "Core/HW/DSP.h"
|
|
#include "Core/HW/Memmap.h"
|
|
#include "Core/PowerPC/PowerPC.h"
|
|
#include "DolphinWX/Debugger/BreakpointWindow.h"
|
|
#include "DolphinWX/Debugger/CodeWindow.h"
|
|
#include "DolphinWX/Debugger/MemoryView.h"
|
|
#include "DolphinWX/Debugger/MemoryWindow.h"
|
|
#include "DolphinWX/Globals.h"
|
|
#include "DolphinWX/WxUtils.h"
|
|
|
|
enum
|
|
{
|
|
IDM_MEM_ADDRBOX,
|
|
IDM_SYMBOLLIST,
|
|
IDM_SETVALBUTTON,
|
|
IDM_DUMP_MEMORY,
|
|
IDM_DUMP_MEM2,
|
|
IDM_DUMP_FAKEVMEM,
|
|
IDM_VALBOX,
|
|
IDM_DATA_TYPE_RBOX,
|
|
IDM_SEARCH,
|
|
IDM_ASCII,
|
|
IDM_HEX,
|
|
IDM_MEMCHECK_OPTIONS_CHANGE
|
|
};
|
|
|
|
BEGIN_EVENT_TABLE(CMemoryWindow, wxPanel)
|
|
EVT_BUTTON(IDM_SETVALBUTTON, CMemoryWindow::SetMemoryValue)
|
|
EVT_BUTTON(IDM_DUMP_MEMORY, CMemoryWindow::OnDumpMemory)
|
|
EVT_BUTTON(IDM_DUMP_MEM2, CMemoryWindow::OnDumpMem2)
|
|
EVT_BUTTON(IDM_DUMP_FAKEVMEM, CMemoryWindow::OnDumpFakeVMEM)
|
|
EVT_RADIOBOX(IDM_DATA_TYPE_RBOX, CMemoryWindow::OnDataTypeChanged)
|
|
EVT_BUTTON(IDM_SEARCH, CMemoryWindow::OnSearch)
|
|
EVT_RADIOBUTTON(IDM_MEMCHECK_OPTIONS_CHANGE, CMemoryWindow::OnMemCheckOptionChange)
|
|
EVT_CHECKBOX(IDM_MEMCHECK_OPTIONS_CHANGE, CMemoryWindow::OnMemCheckOptionChange)
|
|
END_EVENT_TABLE()
|
|
|
|
CMemoryWindow::CMemoryWindow(wxWindow* parent, wxWindowID id, const wxPoint& pos,
|
|
const wxSize& size, long style, const wxString& name)
|
|
: wxPanel(parent, id, pos, size, style, name)
|
|
{
|
|
DebugInterface* di = &PowerPC::debug_interface;
|
|
|
|
memview = new CMemoryView(di, this);
|
|
memview->Bind(DOLPHIN_EVT_MEMORY_VIEW_DATA_TYPE_CHANGED, &CMemoryWindow::OnDataTypeChanged, this);
|
|
|
|
addrbox = new wxSearchCtrl(this, IDM_MEM_ADDRBOX);
|
|
addrbox->Bind(wxEVT_TEXT, &CMemoryWindow::OnAddrBoxChange, this);
|
|
addrbox->SetDescriptiveText(_("Search Address"));
|
|
|
|
valbox =
|
|
new wxTextCtrl(this, IDM_VALBOX, "", wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER);
|
|
valbox->Bind(wxEVT_TEXT_ENTER, &CMemoryWindow::SetMemoryValueFromValBox, this);
|
|
valbox->Bind(wxEVT_TEXT, &CMemoryWindow::OnValueChanged, this);
|
|
|
|
const int space3 = FromDIP(3);
|
|
const int space5 = FromDIP(5);
|
|
|
|
wxBoxSizer* const search_sizer = new wxBoxSizer(wxVERTICAL);
|
|
search_sizer->Add(addrbox, 0, wxEXPAND);
|
|
search_sizer->Add(valbox, 0, wxEXPAND);
|
|
search_sizer->Add(new wxButton(this, IDM_SETVALBUTTON, _("Set Value")));
|
|
|
|
wxBoxSizer* const dump_sizer = new wxBoxSizer(wxVERTICAL);
|
|
dump_sizer->Add(new wxButton(this, IDM_DUMP_MEMORY, _("Dump MRAM")), 0, wxEXPAND);
|
|
dump_sizer->Add(new wxButton(this, IDM_DUMP_MEM2, _("Dump EXRAM")), 0, wxEXPAND);
|
|
if (!SConfig::GetInstance().bMMU)
|
|
dump_sizer->Add(new wxButton(this, IDM_DUMP_FAKEVMEM, _("Dump FakeVMEM")), 0, wxEXPAND);
|
|
|
|
wxStaticBoxSizer* const sizerSearchType = new wxStaticBoxSizer(wxVERTICAL, this, _("Search"));
|
|
sizerSearchType->Add(btnSearch = new wxButton(this, IDM_SEARCH, _("Search")));
|
|
sizerSearchType->Add(m_rb_ascii = new wxRadioButton(this, IDM_ASCII, "Ascii", wxDefaultPosition,
|
|
wxDefaultSize, wxRB_GROUP));
|
|
sizerSearchType->Add(m_rb_hex = new wxRadioButton(this, IDM_HEX, _("Hex")));
|
|
m_search_result_msg =
|
|
new wxStaticText(this, wxID_ANY, wxEmptyString, wxDefaultPosition, wxDefaultSize,
|
|
wxST_NO_AUTORESIZE | wxALIGN_CENTER_HORIZONTAL);
|
|
sizerSearchType->Add(m_search_result_msg, 0, wxEXPAND);
|
|
|
|
wxArrayString data_type_options;
|
|
data_type_options.Add("U8");
|
|
data_type_options.Add("U16");
|
|
data_type_options.Add("U32");
|
|
data_type_options.Add("ASCII");
|
|
data_type_options.Add("Float32");
|
|
m_rbox_data_type = new wxRadioBox(this, IDM_DATA_TYPE_RBOX, _("Data Type"), wxDefaultPosition,
|
|
wxDefaultSize, data_type_options, 1);
|
|
|
|
wxStaticBoxSizer* const memcheck_options_sizer =
|
|
new wxStaticBoxSizer(wxVERTICAL, this, "Memory check options");
|
|
memcheck_options_sizer->Add(rdbReadWrite = new wxRadioButton(this, IDM_MEMCHECK_OPTIONS_CHANGE,
|
|
"Read and Write", wxDefaultPosition,
|
|
wxDefaultSize, wxRB_GROUP));
|
|
memcheck_options_sizer->Add(
|
|
rdbRead = new wxRadioButton(this, IDM_MEMCHECK_OPTIONS_CHANGE, "Read only"));
|
|
memcheck_options_sizer->Add(
|
|
rdbWrite = new wxRadioButton(this, IDM_MEMCHECK_OPTIONS_CHANGE, "Write only"));
|
|
memcheck_options_sizer->Add(chkLog = new wxCheckBox(this, IDM_MEMCHECK_OPTIONS_CHANGE, "Log"));
|
|
|
|
wxBoxSizer* const sizerRight = new wxBoxSizer(wxVERTICAL);
|
|
sizerRight->Add(search_sizer);
|
|
sizerRight->AddSpacer(space5);
|
|
sizerRight->Add(dump_sizer, 0, wxEXPAND);
|
|
sizerRight->Add(sizerSearchType, 0, wxEXPAND);
|
|
sizerRight->Add(m_rbox_data_type, 0, wxEXPAND);
|
|
sizerRight->Add(memcheck_options_sizer, 0, wxEXPAND);
|
|
|
|
wxBoxSizer* const sizerBig = new wxBoxSizer(wxHORIZONTAL);
|
|
sizerBig->Add(memview, 20, wxEXPAND);
|
|
sizerBig->AddSpacer(space3);
|
|
sizerBig->Add(sizerRight, 0, wxEXPAND | wxTOP | wxBOTTOM, space3);
|
|
sizerBig->AddSpacer(space3);
|
|
|
|
SetSizer(sizerBig);
|
|
m_rb_hex->SetValue(true); // Set defaults
|
|
chkLog->SetValue(true);
|
|
m_rbox_data_type->SetSelection(static_cast<int>(memview->GetDataType()));
|
|
|
|
sizerRight->Fit(this);
|
|
sizerBig->Fit(this);
|
|
}
|
|
|
|
void CMemoryWindow::JumpToAddress(u32 _Address)
|
|
{
|
|
memview->Center(_Address);
|
|
}
|
|
|
|
void CMemoryWindow::SetMemoryValueFromValBox(wxCommandEvent& event)
|
|
{
|
|
SetMemoryValue(event);
|
|
valbox->SetFocus();
|
|
}
|
|
|
|
void CMemoryWindow::SetMemoryValue(wxCommandEvent& event)
|
|
{
|
|
if (!Memory::IsInitialized())
|
|
{
|
|
WxUtils::ShowErrorDialog(_("Cannot set uninitialized memory."));
|
|
return;
|
|
}
|
|
|
|
std::string str_addr = WxStrToStr(addrbox->GetValue());
|
|
std::string str_val = WxStrToStr(valbox->GetValue());
|
|
u32 addr;
|
|
u32 val;
|
|
|
|
if (!TryParse(std::string("0x") + str_addr, &addr))
|
|
{
|
|
WxUtils::ShowErrorDialog(wxString::Format(_("Invalid address: %s"), str_addr.c_str()));
|
|
return;
|
|
}
|
|
|
|
if (!TryParse(std::string("0x") + str_val, &val))
|
|
{
|
|
WxUtils::ShowErrorDialog(wxString::Format(_("Invalid value: %s"), str_val.c_str()));
|
|
return;
|
|
}
|
|
|
|
PowerPC::HostWrite_U32(val, addr);
|
|
memview->Refresh();
|
|
}
|
|
|
|
void CMemoryWindow::OnAddrBoxChange(wxCommandEvent& event)
|
|
{
|
|
wxString txt = addrbox->GetValue();
|
|
if (txt.size())
|
|
{
|
|
u32 addr;
|
|
sscanf(WxStrToStr(txt).c_str(), "%08x", &addr);
|
|
memview->Center(addr & ~3);
|
|
}
|
|
|
|
event.Skip();
|
|
}
|
|
|
|
void CMemoryWindow::Repopulate()
|
|
{
|
|
memview->Center(PC);
|
|
}
|
|
|
|
void CMemoryWindow::OnValueChanged(wxCommandEvent&)
|
|
{
|
|
m_continue_search = false;
|
|
}
|
|
|
|
static void DumpArray(const std::string& filename, const u8* data, size_t length)
|
|
{
|
|
if (data)
|
|
{
|
|
File::IOFile f(filename, "wb");
|
|
f.WriteBytes(data, length);
|
|
}
|
|
}
|
|
|
|
// Write mram to file
|
|
void CMemoryWindow::OnDumpMemory(wxCommandEvent& event)
|
|
{
|
|
DumpArray(File::GetUserPath(F_RAMDUMP_IDX), Memory::m_pRAM, Memory::REALRAM_SIZE);
|
|
}
|
|
|
|
// Write exram (aram or mem2) to file
|
|
void CMemoryWindow::OnDumpMem2(wxCommandEvent& event)
|
|
{
|
|
if (SConfig::GetInstance().bWii)
|
|
{
|
|
DumpArray(File::GetUserPath(F_ARAMDUMP_IDX), Memory::m_pEXRAM, Memory::EXRAM_SIZE);
|
|
}
|
|
else
|
|
{
|
|
DumpArray(File::GetUserPath(F_ARAMDUMP_IDX), DSP::GetARAMPtr(), DSP::ARAM_SIZE);
|
|
}
|
|
}
|
|
|
|
// Write fake vmem to file
|
|
void CMemoryWindow::OnDumpFakeVMEM(wxCommandEvent& event)
|
|
{
|
|
DumpArray(File::GetUserPath(F_FAKEVMEMDUMP_IDX), Memory::m_pFakeVMEM, Memory::FAKEVMEM_SIZE);
|
|
}
|
|
|
|
void CMemoryWindow::OnDataTypeChanged(wxCommandEvent& ev)
|
|
{
|
|
static constexpr std::array<MemoryDataType, 5> map{{MemoryDataType::U8, MemoryDataType::U16,
|
|
MemoryDataType::U32, MemoryDataType::ASCII,
|
|
MemoryDataType::FloatingPoint}};
|
|
if (ev.GetId() == IDM_DATA_TYPE_RBOX)
|
|
{
|
|
memview->SetDataType(map.at(ev.GetSelection()));
|
|
}
|
|
else
|
|
{
|
|
// Event from the CMemoryView indicating type was changed.
|
|
auto itr = std::find(map.begin(), map.end(), static_cast<MemoryDataType>(ev.GetInt()));
|
|
int idx = -1;
|
|
if (itr != map.end())
|
|
idx = static_cast<int>(itr - map.begin());
|
|
m_rbox_data_type->SetSelection(idx);
|
|
}
|
|
}
|
|
|
|
void CMemoryWindow::OnSearch(wxCommandEvent& event)
|
|
{
|
|
wxBusyCursor hourglass_cursor;
|
|
u8* ram_ptr = nullptr;
|
|
u32 ram_size = 0;
|
|
// NOTE: We're assuming the base address is zero.
|
|
switch (memview->GetMemoryType())
|
|
{
|
|
case 0:
|
|
default:
|
|
if (Memory::m_pRAM)
|
|
{
|
|
ram_ptr = Memory::m_pRAM;
|
|
ram_size = Memory::REALRAM_SIZE;
|
|
}
|
|
break;
|
|
case 1:
|
|
{
|
|
u8* aram = DSP::GetARAMPtr();
|
|
if (aram)
|
|
{
|
|
ram_ptr = aram;
|
|
ram_size = DSP::ARAM_SIZE;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
if (!ram_ptr)
|
|
{
|
|
m_search_result_msg->SetLabel(_("Memory Not Ready"));
|
|
return;
|
|
}
|
|
|
|
std::vector<u8> search_bytes;
|
|
wxString search_val = valbox->GetValue();
|
|
|
|
if (m_rb_hex->GetValue())
|
|
{
|
|
search_val.Trim(true).Trim(false);
|
|
// If there's a trailing nybble, stick a zero in front to make it a byte
|
|
if (search_val.size() & 1)
|
|
search_val.insert(0, 1, '0');
|
|
search_bytes.reserve(search_val.size() / 2);
|
|
|
|
wxString conversion_buffer(2, ' ');
|
|
for (std::size_t i = 0; i < search_val.size(); i += 2)
|
|
{
|
|
unsigned long byte = 0;
|
|
conversion_buffer[0] = search_val[i];
|
|
conversion_buffer[1] = search_val[i + 1];
|
|
if (!conversion_buffer.ToULong(&byte, 16))
|
|
{
|
|
m_search_result_msg->SetLabel(_("Not Valid Hex"));
|
|
return;
|
|
}
|
|
search_bytes.push_back(static_cast<u8>(byte));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
const auto& bytes = search_val.ToUTF8();
|
|
search_bytes.assign(bytes.data(), bytes.data() + bytes.length());
|
|
}
|
|
search_val.Clear();
|
|
|
|
// For completeness
|
|
if (search_bytes.size() > ram_size)
|
|
{
|
|
m_search_result_msg->SetLabel(_("Value Too Large"));
|
|
return;
|
|
}
|
|
|
|
if (search_bytes.empty())
|
|
{
|
|
m_search_result_msg->SetLabel(_("No Value Given"));
|
|
return;
|
|
}
|
|
|
|
// Search starting from specified address if there is one.
|
|
u32 addr = 0; // Base address
|
|
{
|
|
wxString addr_val = addrbox->GetValue();
|
|
addr_val.Trim(true).Trim(false);
|
|
if (!addr_val.empty())
|
|
{
|
|
unsigned long addr_ul = 0;
|
|
if (addr_val.ToULong(&addr_ul, 16))
|
|
{
|
|
addr = static_cast<u32>(addr_ul);
|
|
// Don't find the result we're already looking at
|
|
if (m_continue_search && addr == m_last_search_address)
|
|
addr += 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If the current address doesn't leave enough bytes to search then we're done.
|
|
if (addr >= ram_size - search_bytes.size())
|
|
{
|
|
m_search_result_msg->SetLabel(_("Address Out of Range"));
|
|
return;
|
|
}
|
|
|
|
u8* end = &ram_ptr[ram_size - search_bytes.size() + 1];
|
|
u8* ptr = &ram_ptr[addr];
|
|
while (true)
|
|
{
|
|
ptr = std::find(ptr, end, search_bytes[0]);
|
|
if (ptr == end)
|
|
{
|
|
m_search_result_msg->SetLabel(_("No Match"));
|
|
break;
|
|
}
|
|
|
|
if (std::equal(search_bytes.begin(), search_bytes.end(), ptr))
|
|
{
|
|
m_search_result_msg->SetLabel(_("Match Found"));
|
|
u32 offset = static_cast<u32>(ptr - ram_ptr);
|
|
// NOTE: SetValue() generates a synthetic wxEVT_TEXT
|
|
addrbox->SetValue(wxString::Format("%08x", offset));
|
|
m_last_search_address = offset;
|
|
m_continue_search = true;
|
|
break;
|
|
}
|
|
|
|
++ptr;
|
|
}
|
|
}
|
|
|
|
void CMemoryWindow::OnMemCheckOptionChange(wxCommandEvent& event)
|
|
{
|
|
if (rdbReadWrite->GetValue())
|
|
memview->SetMemCheckOptions(true, true, chkLog->GetValue());
|
|
else
|
|
memview->SetMemCheckOptions(rdbRead->GetValue(), rdbWrite->GetValue(), chkLog->GetValue());
|
|
}
|