forked from dolphin-emu/dolphin
		
	so the recent addition of VideoCommon/Src/Debugger.h clashed with DebuggerWX/Src/Debugger.h. The latter had atrophied anyway, so just get rid of it to resolve the conflict. git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@6526 8ced0084-cf51-0410-be5f-012b33b47a6e
		
			
				
	
	
		
			438 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			438 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
// Copyright (C) 2003 Dolphin Project.
 | 
						|
 | 
						|
// This program is free software: you can redistribute it and/or modify
 | 
						|
// it under the terms of the GNU General Public License as published by
 | 
						|
// the Free Software Foundation, version 2.0.
 | 
						|
 | 
						|
// This program is distributed in the hope that it will be useful,
 | 
						|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
						|
// GNU General Public License 2.0 for more details.
 | 
						|
 | 
						|
// A copy of the GPL 2.0 should have been included with the program.
 | 
						|
// If not, see http://www.gnu.org/licenses/
 | 
						|
 | 
						|
// Official SVN repository and contact information can be found at
 | 
						|
// http://code.google.com/p/dolphin-emu/
 | 
						|
 | 
						|
#include <wx/button.h>
 | 
						|
#include <wx/textctrl.h>
 | 
						|
#include <wx/listctrl.h>
 | 
						|
#include <wx/thread.h>
 | 
						|
#include <wx/listctrl.h>
 | 
						|
#include "MemoryWindow.h"
 | 
						|
#include "HW/CPU.h"
 | 
						|
#include "PowerPC/PowerPC.h"
 | 
						|
#include "Host.h"
 | 
						|
#include "FileUtil.h"
 | 
						|
 | 
						|
#include "Debugger/PPCDebugInterface.h"
 | 
						|
#include "PowerPC/PPCSymbolDB.h"
 | 
						|
 | 
						|
#include "Core.h"
 | 
						|
#include "ConfigManager.h"
 | 
						|
#include "LogManager.h"
 | 
						|
 | 
						|
#include "HW/Memmap.h"
 | 
						|
#include "HW/DSP.h"
 | 
						|
 | 
						|
#include "../../DolphinWX/Src/Globals.h"
 | 
						|
 | 
						|
enum
 | 
						|
{
 | 
						|
	IDM_MEM_ADDRBOX = 350,
 | 
						|
	IDM_SYMBOLLIST,
 | 
						|
	IDM_SETVALBUTTON,
 | 
						|
	IDM_DUMP_MEMORY,
 | 
						|
	IDM_DUMP_MEM2,
 | 
						|
	IDM_VALBOX,
 | 
						|
	IDM_U8,
 | 
						|
	IDM_U16,
 | 
						|
	IDM_U32,
 | 
						|
	IDM_SEARCH,
 | 
						|
	IDM_ASCII,
 | 
						|
	IDM_HEX
 | 
						|
};
 | 
						|
 | 
						|
BEGIN_EVENT_TABLE(CMemoryWindow, wxPanel)
 | 
						|
	EVT_TEXT(IDM_MEM_ADDRBOX,		CMemoryWindow::OnAddrBoxChange)
 | 
						|
	EVT_LISTBOX(IDM_SYMBOLLIST,		CMemoryWindow::OnSymbolListChange)
 | 
						|
	EVT_HOST_COMMAND(wxID_ANY,		CMemoryWindow::OnHostMessage)
 | 
						|
	EVT_BUTTON(IDM_SETVALBUTTON,	CMemoryWindow::SetMemoryValue)
 | 
						|
	EVT_BUTTON(IDM_DUMP_MEMORY,		CMemoryWindow::OnDumpMemory)
 | 
						|
	EVT_BUTTON(IDM_DUMP_MEM2,		CMemoryWindow::OnDumpMem2)
 | 
						|
	EVT_CHECKBOX(IDM_U8,			CMemoryWindow::U8)
 | 
						|
	EVT_CHECKBOX(IDM_U16,			CMemoryWindow::U16)
 | 
						|
	EVT_CHECKBOX(IDM_U32,			CMemoryWindow::U32)
 | 
						|
	EVT_BUTTON(IDM_SEARCH,			CMemoryWindow::onSearch)
 | 
						|
	EVT_CHECKBOX(IDM_ASCII,			CMemoryWindow::onAscii)
 | 
						|
	EVT_CHECKBOX(IDM_HEX,			CMemoryWindow::onHex)
 | 
						|
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)
 | 
						|
{
 | 
						|
	wxBoxSizer* sizerBig   = new wxBoxSizer(wxHORIZONTAL);
 | 
						|
	wxBoxSizer* sizerRight = new wxBoxSizer(wxVERTICAL);
 | 
						|
	// Didn't see anything usefull in the left part
 | 
						|
	//wxBoxSizer* sizerLeft  = new wxBoxSizer(wxVERTICAL);
 | 
						|
 | 
						|
	DebugInterface* di = &PowerPC::debug_interface;
 | 
						|
 | 
						|
	//symbols = new wxListBox(this, IDM_SYMBOLLIST, wxDefaultPosition,
 | 
						|
	//	   	wxSize(20, 100), 0, NULL, wxLB_SORT);
 | 
						|
	//sizerLeft->Add(symbols, 1, wxEXPAND);
 | 
						|
	memview = new CMemoryView(di, this, wxID_ANY);
 | 
						|
	memview->dataType = 0;
 | 
						|
	//sizerBig->Add(sizerLeft, 1, wxEXPAND);
 | 
						|
	sizerBig->Add(memview, 20, wxEXPAND);
 | 
						|
	sizerBig->Add(sizerRight, 0, wxEXPAND | wxALL, 3);
 | 
						|
	sizerRight->Add(addrbox = new wxTextCtrl(this, IDM_MEM_ADDRBOX, _T("")));
 | 
						|
	sizerRight->Add(valbox = new wxTextCtrl(this, IDM_VALBOX, _T("")));
 | 
						|
	sizerRight->Add(new wxButton(this, IDM_SETVALBUTTON, _T("Set &Value")));
 | 
						|
 | 
						|
	sizerRight->AddSpacer(5);
 | 
						|
	sizerRight->Add(new wxButton(this, IDM_DUMP_MEMORY, _T("&Dump MRAM")));
 | 
						|
	sizerRight->Add(new wxButton(this, IDM_DUMP_MEM2, _T("&Dump EXRAM")));
 | 
						|
 | 
						|
	wxStaticBoxSizer* sizerSearchType = new wxStaticBoxSizer(wxVERTICAL, this, wxT("Search"));
 | 
						|
 | 
						|
	sizerSearchType->Add(btnSearch = new wxButton(this, IDM_SEARCH, _T("Search")));
 | 
						|
	sizerSearchType->Add(chkAscii = new wxCheckBox(this, IDM_ASCII, _T("&Ascii ")));
 | 
						|
	sizerSearchType->Add(chkHex = new wxCheckBox(this, IDM_HEX, _T("&Hex")));
 | 
						|
	sizerRight->Add(sizerSearchType);
 | 
						|
	wxStaticBoxSizer* sizerDataTypes = new wxStaticBoxSizer(wxVERTICAL, this, wxT("Data Type"));
 | 
						|
 | 
						|
	sizerDataTypes->SetMinSize(74, 40);
 | 
						|
	sizerDataTypes->Add(chk8 = new wxCheckBox(this, IDM_U8, _T("&U8")));
 | 
						|
	sizerDataTypes->Add(chk16 = new wxCheckBox(this, IDM_U16, _T("&U16")));
 | 
						|
	sizerDataTypes->Add(chk32 = new wxCheckBox(this, IDM_U32, _T("&U32")));
 | 
						|
	sizerRight->Add(sizerDataTypes);
 | 
						|
	SetSizer(sizerBig);
 | 
						|
	chkHex->SetValue(1); //Set defaults
 | 
						|
	chk8->SetValue(1);
 | 
						|
 | 
						|
	//sizerLeft->SetSizeHints(this);
 | 
						|
	//sizerLeft->Fit(this);
 | 
						|
	sizerRight->SetSizeHints(this);
 | 
						|
	sizerRight->Fit(this);
 | 
						|
	sizerBig->SetSizeHints(this);
 | 
						|
	sizerBig->Fit(this);
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::Save(IniFile& _IniFile) const
 | 
						|
{
 | 
						|
	// Prevent these bad values that can happen after a crash or hanging
 | 
						|
	if(GetPosition().x != -32000 && GetPosition().y != -32000)
 | 
						|
	{
 | 
						|
		_IniFile.Set("MemoryWindow", "x", GetPosition().x);
 | 
						|
		_IniFile.Set("MemoryWindow", "y", GetPosition().y);
 | 
						|
		_IniFile.Set("MemoryWindow", "w", GetSize().GetWidth());
 | 
						|
		_IniFile.Set("MemoryWindow", "h", GetSize().GetHeight());
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::Load(IniFile& _IniFile)
 | 
						|
{
 | 
						|
	int x, y, w, h;
 | 
						|
	_IniFile.Get("MemoryWindow", "x", &x, GetPosition().x);
 | 
						|
	_IniFile.Get("MemoryWindow", "y", &y, GetPosition().y);
 | 
						|
	_IniFile.Get("MemoryWindow", "w", &w, GetSize().GetWidth());
 | 
						|
	_IniFile.Get("MemoryWindow", "h", &h, GetSize().GetHeight());
 | 
						|
	SetSize(x, y, w, h);
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::JumpToAddress(u32 _Address)
 | 
						|
{
 | 
						|
	memview->Center(_Address);
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::SetMemoryValue(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	std::string str_addr = std::string(addrbox->GetValue().mb_str());
 | 
						|
	std::string str_val = std::string(valbox->GetValue().mb_str());
 | 
						|
	u32 addr;
 | 
						|
	u32 val;
 | 
						|
 | 
						|
	if (!TryParse(std::string("0x") + str_addr, &addr))
 | 
						|
	{
 | 
						|
		PanicAlert("Invalid Address: %s", str_addr.c_str());
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!TryParse(std::string("0x") + str_val, &val))
 | 
						|
	{
 | 
						|
		PanicAlert("Invalid Value: %s", str_val.c_str());
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	Memory::Write_U32(val, addr);
 | 
						|
	memview->Refresh();
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::OnAddrBoxChange(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	wxString txt = addrbox->GetValue();
 | 
						|
	if (txt.size())
 | 
						|
	{
 | 
						|
		u32 addr;
 | 
						|
		sscanf(txt.mb_str(), "%08x", &addr);
 | 
						|
		memview->Center(addr & ~3);
 | 
						|
	}
 | 
						|
 | 
						|
	event.Skip(1);
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::Update()
 | 
						|
{
 | 
						|
	memview->Refresh();
 | 
						|
	memview->Center(PC);
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::NotifyMapLoaded()
 | 
						|
{
 | 
						|
	symbols->Show(false); // hide it for faster filling
 | 
						|
	symbols->Clear();
 | 
						|
#if 0
 | 
						|
	#ifdef _WIN32
 | 
						|
	const FunctionDB::XFuncMap &syms = g_symbolDB.Symbols();
 | 
						|
	for (FuntionDB::XFuncMap::iterator iter = syms.begin(); iter != syms.end(); ++iter)
 | 
						|
	{
 | 
						|
	int idx = symbols->Append(iter->second.name.c_str());
 | 
						|
	symbols->SetClientData(idx, (void*)&iter->second);
 | 
						|
	}
 | 
						|
	#endif
 | 
						|
#endif
 | 
						|
	symbols->Show(true);
 | 
						|
	Update();
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::OnSymbolListChange(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	int index = symbols->GetSelection();
 | 
						|
	if (index >= 0)
 | 
						|
	{
 | 
						|
		Symbol* pSymbol = static_cast<Symbol *>(symbols->GetClientData(index));
 | 
						|
		if (pSymbol != NULL)
 | 
						|
		{
 | 
						|
			memview->Center(pSymbol->address);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::OnHostMessage(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	switch (event.GetId())
 | 
						|
	{
 | 
						|
		case IDM_NOTIFYMAPLOADED:
 | 
						|
			NotifyMapLoaded();
 | 
						|
			break;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Write mram to file
 | 
						|
void CMemoryWindow::OnDumpMemory( wxCommandEvent& event )
 | 
						|
{
 | 
						|
	FILE* f = fopen(File::GetUserPath(F_RAMDUMP_IDX), "wb");
 | 
						|
	if (f && Memory::m_pRAM)
 | 
						|
	{
 | 
						|
		fwrite(Memory::m_pRAM, Memory::REALRAM_SIZE, 1, f);
 | 
						|
		fclose(f);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// Write exram (aram or mem2) to file
 | 
						|
void CMemoryWindow::OnDumpMem2( wxCommandEvent& event )
 | 
						|
{
 | 
						|
	FILE* f = NULL;
 | 
						|
	
 | 
						|
	if (SConfig::GetInstance().m_LocalCoreStartupParameter.bWii)
 | 
						|
	{
 | 
						|
		f = fopen(File::GetUserPath(F_ARAMDUMP_IDX), "wb");
 | 
						|
		if (f && Memory::m_pEXRAM)
 | 
						|
		{
 | 
						|
			fwrite(Memory::m_pEXRAM, Memory::EXRAM_SIZE, 1, f);
 | 
						|
			fclose(f);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		u8* aram = DSP::GetARAMPtr();
 | 
						|
		f = fopen(File::GetUserPath(F_ARAMDUMP_IDX), "wb");
 | 
						|
		if (f && aram)
 | 
						|
		{
 | 
						|
			fwrite(aram, DSP::ARAM_SIZE, 1, f);
 | 
						|
			fclose(f);
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::U8(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	chk16->SetValue(0);
 | 
						|
	chk32->SetValue(0);
 | 
						|
	memview->dataType = 0;
 | 
						|
	memview->Refresh();
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::U16(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	chk8->SetValue(0);
 | 
						|
	chk32->SetValue(0);
 | 
						|
	memview->dataType = 1;
 | 
						|
	memview->Refresh();
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::U32(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	chk16->SetValue(0);
 | 
						|
	chk8->SetValue(0);
 | 
						|
	memview->dataType = 2;
 | 
						|
	memview->Refresh();
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::onSearch(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	u8* TheRAM = 0;
 | 
						|
	u32 szRAM = 0;
 | 
						|
	switch (memview->GetMemoryType())
 | 
						|
	{
 | 
						|
	case 0:
 | 
						|
	default:
 | 
						|
		if (Memory::m_pRAM)
 | 
						|
		{
 | 
						|
			TheRAM = Memory::m_pRAM;
 | 
						|
			szRAM = Memory::REALRAM_SIZE;
 | 
						|
		}
 | 
						|
		break;
 | 
						|
	case 1:
 | 
						|
		{
 | 
						|
			u8* aram = DSP::GetARAMPtr();
 | 
						|
			if (aram)
 | 
						|
			{
 | 
						|
				TheRAM = aram;
 | 
						|
				szRAM = DSP::ARAM_SIZE;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		break;
 | 
						|
	}
 | 
						|
	//Now we have memory to look in
 | 
						|
	//Are we looking for ASCII string, or hex?
 | 
						|
	//memview->cu
 | 
						|
	wxString rawData = valbox->GetValue();
 | 
						|
	std::vector<u8> Dest; //May need a better name
 | 
						|
	u32 size = 0;
 | 
						|
	int pad = rawData.size()%2; //If it's uneven
 | 
						|
	unsigned int i = 0;
 | 
						|
	long count = 0;
 | 
						|
	char copy[3] = {0};
 | 
						|
	long newsize = 0;
 | 
						|
	unsigned char *tmp2 = 0;
 | 
						|
	char* tmpstr = 0;
 | 
						|
	switch (chkHex->GetValue())
 | 
						|
	{
 | 
						|
	case 1://We are looking for hex
 | 
						|
		//If it's uneven
 | 
						|
		size = (rawData.size()/2) + pad;
 | 
						|
		Dest.resize(size+32);
 | 
						|
		newsize = rawData.size();
 | 
						|
 | 
						|
		if (pad)
 | 
						|
	   	{
 | 
						|
			tmpstr = new char[newsize + 2];
 | 
						|
			memset(tmpstr, 0, newsize + 2);
 | 
						|
			tmpstr[0] = '0';
 | 
						|
		}
 | 
						|
	   	else
 | 
						|
	   	{
 | 
						|
			tmpstr = new char[newsize + 1];
 | 
						|
			memset(tmpstr, 0, newsize + 1);
 | 
						|
		}
 | 
						|
		//sprintf(tmpstr, "%s%s", tmpstr, rawData.c_str());
 | 
						|
		//strcpy(&tmpstr[1], rawData.ToAscii());
 | 
						|
		//memcpy(&tmpstr[1], &rawData.c_str()[0], rawData.size());
 | 
						|
		sprintf(tmpstr, "%s%s", tmpstr, (const char *)rawData.mb_str());
 | 
						|
		tmp2 = &Dest.front();
 | 
						|
		count = 0;
 | 
						|
		for(i = 0; i < strlen(tmpstr); i++)
 | 
						|
		{
 | 
						|
			copy[0] = tmpstr[i];
 | 
						|
			copy[1] = tmpstr[i+1];
 | 
						|
			copy[2] = 0;
 | 
						|
			int tmpint;
 | 
						|
			sscanf(copy, "%02x", &tmpint);
 | 
						|
			tmp2[count++] = tmpint;
 | 
						|
			// Dest[count] should now be the hex of what the two chars were!
 | 
						|
			// Also should add a check to make sure it's A-F only
 | 
						|
			//sscanf(copy, "%02x", &tmp2[count++]);
 | 
						|
			i += 1;
 | 
						|
		}
 | 
						|
		delete[] tmpstr;
 | 
						|
		break;
 | 
						|
	case 0://Looking for an ascii string
 | 
						|
		size = rawData.size();
 | 
						|
		Dest.resize(size+1);
 | 
						|
		tmpstr = new char[size+1];
 | 
						|
 | 
						|
		tmp2 = &Dest.front();
 | 
						|
		sprintf(tmpstr, "%s", (const char *)rawData.mb_str());
 | 
						|
 | 
						|
		for(i = 0; i < size; i++)
 | 
						|
			tmp2[i] = tmpstr[i];
 | 
						|
 | 
						|
		delete[] tmpstr;
 | 
						|
		break;
 | 
						|
	}
 | 
						|
	if(size)
 | 
						|
	{
 | 
						|
		unsigned char* pnt = &Dest.front();
 | 
						|
		unsigned int k = 0;
 | 
						|
		//grab
 | 
						|
		wxString txt = addrbox->GetValue();
 | 
						|
		u32 addr = 0;
 | 
						|
		if (txt.size())
 | 
						|
		{
 | 
						|
			sscanf(txt.mb_str(), "%08x", &addr);
 | 
						|
		}
 | 
						|
		i = addr+4;
 | 
						|
		for( ; i < szRAM; i++)
 | 
						|
		{
 | 
						|
			for(k = 0; k < size; k++)
 | 
						|
			{
 | 
						|
				if(i + k > szRAM) break;
 | 
						|
				if(k > size) break;
 | 
						|
				if(pnt[k] != TheRAM[i+k])
 | 
						|
				{
 | 
						|
					k = 0;
 | 
						|
					break;
 | 
						|
				}
 | 
						|
			}
 | 
						|
			if(k == size)
 | 
						|
			{
 | 
						|
				//Match was found
 | 
						|
				wxMessageBox(_T("A match was found. Placing viewer at the offset."));
 | 
						|
				wxChar tmpwxstr[128] = {0};
 | 
						|
				wxSprintf(tmpwxstr, _T("%08x"), i);
 | 
						|
				wxString tmpwx(tmpwxstr);
 | 
						|
				addrbox->SetValue(tmpwx);
 | 
						|
				//memview->curAddress = i;
 | 
						|
				//memview->Refresh();
 | 
						|
				OnAddrBoxChange(event);
 | 
						|
				return;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		wxMessageBox(_T("No match was found."));
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::onAscii(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	chkHex->SetValue(0);
 | 
						|
}
 | 
						|
 | 
						|
void CMemoryWindow::onHex(wxCommandEvent& event)
 | 
						|
{
 | 
						|
	chkAscii->SetValue(0);
 | 
						|
}
 |