mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 17:39:09 +00:00 
			
		
		
		
	git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@6222 8ced0084-cf51-0410-be5f-012b33b47a6e
		
			
				
	
	
		
			268 lines
		
	
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			7.7 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 "Debugger.h"
 | |
| 
 | |
| #include <wx/button.h>
 | |
| #include <wx/textctrl.h>
 | |
| #include <wx/listctrl.h>
 | |
| #include <wx/thread.h>
 | |
| #include <wx/listctrl.h>
 | |
| 
 | |
| #include "JitWindow.h"
 | |
| #include "HW/CPU.h"
 | |
| #include "PowerPC/PowerPC.h"
 | |
| #include "PowerPC/JitCommon/JitBase.h"
 | |
| #include "PowerPC/JitCommon/JitCache.h"
 | |
| #include "PowerPC/PPCAnalyst.h"
 | |
| #include "PowerPCDisasm.h"
 | |
| #include "Host.h"
 | |
| #include "disasm.h"
 | |
| 
 | |
| #include "Debugger/PPCDebugInterface.h"
 | |
| #include "Debugger/Debugger_SymbolMap.h"
 | |
| 
 | |
| #include "Core.h"
 | |
| #include "StringUtil.h"
 | |
| #include "LogManager.h"
 | |
| 
 | |
| // ugly that this lib included code from the main
 | |
| #include "../../DolphinWX/Src/Globals.h"
 | |
| 
 | |
| // TODO:  Fix this ugly hack
 | |
| namespace {
 | |
| CJitWindow *the_jit_window;
 | |
| }
 | |
| 
 | |
| enum
 | |
| {
 | |
| 	IDM_REFRESH_LIST = 23350,
 | |
| 	IDM_PPC_BOX,
 | |
| 	IDM_X86_BOX,
 | |
| 	IDM_NEXT,
 | |
| 	IDM_PREV,
 | |
| 	IDM_BLOCKLIST,
 | |
| };
 | |
| 
 | |
| BEGIN_EVENT_TABLE(CJitWindow, wxPanel)
 | |
| 	//EVT_TEXT(IDM_ADDRBOX, CJitWindow::OnAddrBoxChange)
 | |
| 	//EVT_LISTBOX(IDM_SYMBOLLIST, CJitWindow::OnSymbolListChange)
 | |
| 	//EVT_HOST_COMMAND(wxID_ANY, CJitWindow::OnHostMessage)
 | |
| 	EVT_BUTTON(IDM_REFRESH_LIST, CJitWindow::OnRefresh)
 | |
| END_EVENT_TABLE()
 | |
| 
 | |
| CJitWindow::CJitWindow(wxWindow* parent, wxWindowID id, const wxPoint& pos,
 | |
| 	   	const wxSize& size, long style, const wxString& name)
 | |
| : wxPanel(parent, id, pos, size, style, name)
 | |
| {
 | |
| 	the_jit_window = this;
 | |
| 	wxBoxSizer* sizerBig   = new wxBoxSizer(wxVERTICAL);
 | |
| 	wxBoxSizer* sizerSplit = new wxBoxSizer(wxHORIZONTAL);
 | |
| 	sizerSplit->Add(ppc_box = new wxTextCtrl(this, IDM_PPC_BOX, _T("(ppc)"),
 | |
| 			   	wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE), 1, wxEXPAND);
 | |
| 	sizerSplit->Add(x86_box = new wxTextCtrl(this, IDM_X86_BOX, _T("(x86)"),
 | |
| 			   	wxDefaultPosition, wxDefaultSize, wxTE_MULTILINE), 1, wxEXPAND);
 | |
| 	sizerBig->Add(block_list = new JitBlockList(this, IDM_BLOCKLIST,
 | |
| 				wxDefaultPosition, wxSize(100, 140),
 | |
| 				wxLC_REPORT | wxSUNKEN_BORDER | wxLC_ALIGN_LEFT | wxLC_SINGLE_SEL | wxLC_SORT_ASCENDING),
 | |
| 				0, wxEXPAND);
 | |
| 	sizerBig->Add(sizerSplit, 2, wxEXPAND);
 | |
| //	sizerBig->Add(memview, 5, wxEXPAND);
 | |
| //	sizerBig->Add(sizerRight, 0, wxEXPAND | wxALL, 3);
 | |
| 	sizerBig->Add(button_refresh = new wxButton(this, IDM_REFRESH_LIST, _T("&Refresh")));
 | |
| //	sizerRight->Add(addrbox = new wxTextCtrl(this, IDM_ADDRBOX, _T("")));
 | |
| //	sizerRight->Add(new wxButton(this, IDM_SETPC, _T("S&et PC")));
 | |
| 
 | |
| 	SetSizer(sizerBig);
 | |
| 
 | |
| 	sizerSplit->SetSizeHints(this);
 | |
| 	sizerSplit->Fit(this);
 | |
| 	sizerBig->SetSizeHints(this);
 | |
| 	sizerBig->Fit(this);
 | |
| }
 | |
| 
 | |
| void CJitWindow::OnRefresh(wxCommandEvent& /*event*/) {
 | |
| 	block_list->Update();
 | |
| }
 | |
| 
 | |
| void CJitWindow::ViewAddr(u32 em_address)
 | |
| {
 | |
| 	if (the_jit_window)
 | |
| 	{
 | |
| 		the_jit_window->Show(true);
 | |
| 		the_jit_window->Compare(em_address);
 | |
| 		the_jit_window->SetFocus();
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		PanicAlert("Jit window not available");
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void CJitWindow::Compare(u32 em_address)
 | |
| {
 | |
| 	u8 *xDis = new u8[1<<18];
 | |
| 	memset(xDis, 0, 1<<18);
 | |
| 
 | |
| 	disassembler x64disasm;
 | |
| 	x64disasm.set_syntax_intel();
 | |
| 
 | |
| 	int block_num = jit->GetBlockCache()->GetBlockNumberFromStartAddress(em_address);
 | |
| 	if (block_num < 0)
 | |
| 	{
 | |
| 		for (int i = 0; i < 500; i++) {
 | |
| 			block_num = jit->GetBlockCache()->GetBlockNumberFromStartAddress(em_address - 4 * i);
 | |
| 			if (block_num >= 0)
 | |
| 				break;
 | |
| 		}
 | |
| 		if (block_num >= 0) {
 | |
| 			JitBlock *block = jit->GetBlockCache()->GetBlock(block_num);
 | |
| 			if (!(block->originalAddress <= em_address &&
 | |
| 					   	block->originalSize + block->originalAddress >= em_address))
 | |
| 				block_num = -1;
 | |
| 		}
 | |
| 		// Do not merge this "if" with the above - block_num changes inside it.
 | |
| 		if (block_num < 0) {
 | |
| 			ppc_box->SetValue(wxString::FromAscii(StringFromFormat("(non-code address: %08x)",
 | |
| 						   	em_address).c_str()));
 | |
| 			x86_box->SetValue(wxString::FromAscii(StringFromFormat("(no translation)").c_str()));
 | |
| 			delete[] xDis;
 | |
| 			return;
 | |
| 		}
 | |
| 	}
 | |
| 	JitBlock *block = jit->GetBlockCache()->GetBlock(block_num);
 | |
| 
 | |
| 	// 800031f0
 | |
| 	// == Fill in x86 box
 | |
| 
 | |
| 	const u8 *code = (const u8 *)jit->GetBlockCache()->GetCompiledCodeFromBlock(block_num);
 | |
| 	u64 disasmPtr = (u64)code;
 | |
| 	int size = block->codeSize;
 | |
| 	const u8 *end = code + size;
 | |
| 	char *sptr = (char*)xDis;
 | |
| 
 | |
| 	int num_x86_instructions = 0;
 | |
| 	while ((u8*)disasmPtr < end)
 | |
| 	{
 | |
| #ifdef _M_X64
 | |
| 		disasmPtr += x64disasm.disasm64(disasmPtr, disasmPtr, (u8*)disasmPtr, sptr);
 | |
| #else
 | |
| 		disasmPtr += x64disasm.disasm32(disasmPtr, disasmPtr, (u8*)disasmPtr, sptr);
 | |
| #endif
 | |
| 		sptr += strlen(sptr);
 | |
| 		*sptr++ = 13;
 | |
| 		*sptr++ = 10;
 | |
| 		num_x86_instructions++;
 | |
| 	}
 | |
| 	x86_box->SetValue(wxString::FromAscii((char*)xDis));
 | |
| 
 | |
| 	// == Fill in ppc box
 | |
| 	u32 ppc_addr = block->originalAddress;
 | |
| 	PPCAnalyst::CodeBuffer code_buffer(32000);
 | |
| 	PPCAnalyst::BlockStats st;
 | |
| 	PPCAnalyst::BlockRegStats gpa;
 | |
| 	PPCAnalyst::BlockRegStats fpa;
 | |
| 	bool broken_block = false;
 | |
| 	u32 merged_addresses[32];
 | |
| 	const int capacity_of_merged_addresses = sizeof(merged_addresses) / sizeof(merged_addresses[0]);
 | |
| 	int size_of_merged_addresses;
 | |
| 	if (PPCAnalyst::Flatten(ppc_addr, &size, &st, &gpa, &fpa, broken_block, &code_buffer, size, merged_addresses, capacity_of_merged_addresses, size_of_merged_addresses) != 0xffffffff)
 | |
| 	{
 | |
| 		sptr = (char*)xDis;
 | |
| 		for (int i = 0; i < size; i++)
 | |
| 		{
 | |
| 			const PPCAnalyst::CodeOp &op = code_buffer.codebuffer[i];
 | |
| 			char temp[256];
 | |
| 			DisassembleGekko(op.inst.hex, op.address, temp, 256);
 | |
| 			sptr += sprintf(sptr, "%08x %s\n", op.address, temp);
 | |
| 		}
 | |
| 
 | |
| 		// Add stats to the end of the ppc box since it's generally the shortest.
 | |
| 		sptr += sprintf(sptr, "\n");
 | |
| 
 | |
| 		// Add some generic analysis
 | |
| 		if (st.isFirstBlockOfFunction)
 | |
| 			sptr += sprintf(sptr, "(first block of function)\n");
 | |
| 		if (st.isLastBlockOfFunction)
 | |
| 			sptr += sprintf(sptr, "(first block of function)\n");
 | |
| 
 | |
| 		sptr += sprintf(sptr, "%i estimated cycles\n", st.numCycles);
 | |
| 
 | |
| 		sptr += sprintf(sptr, "Num instr: PPC: %i  x86: %i  (blowup: %i%%)\n",
 | |
| 			   	size, num_x86_instructions, 100 * (num_x86_instructions / size - 1));
 | |
| 		sptr += sprintf(sptr, "Num bytes: PPC: %i  x86: %i  (blowup: %i%%)\n",
 | |
| 			   	size * 4, block->codeSize, 100 * (block->codeSize / (4 * size) - 1));
 | |
| 
 | |
| 		ppc_box->SetValue(wxString::FromAscii((char*)xDis));
 | |
| 	} else {
 | |
| 		ppc_box->SetValue(wxString::FromAscii(StringFromFormat(
 | |
| 						"(non-code address: %08x)", em_address).c_str()));
 | |
| 		x86_box->SetValue(wxString::FromAscii("---"));
 | |
| 	}
 | |
| 
 | |
| 	delete[] xDis;
 | |
| }
 | |
| 
 | |
| void CJitWindow::Update()
 | |
| {
 | |
| 
 | |
| }
 | |
| 
 | |
| void CJitWindow::OnHostMessage(wxCommandEvent& event)
 | |
| {
 | |
| 	switch (event.GetId())
 | |
| 	{
 | |
| 	    case IDM_NOTIFYMAPLOADED:
 | |
| 		    //NotifyMapLoaded();
 | |
| 		    break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // JitBlockList
 | |
| //================
 | |
| 
 | |
| enum {
 | |
| 	COLUMN_ADDRESS,
 | |
| 	COLUMN_PPCSIZE,
 | |
| 	COLUMN_X86SIZE,
 | |
| 	COLUMN_NAME,
 | |
| 	COLUMN_FLAGS,
 | |
| 	COLUMN_NUMEXEC,
 | |
| 	COLUMN_COST,  // (estimated as x86size * numexec)
 | |
| };
 | |
| 
 | |
| JitBlockList::JitBlockList(wxWindow* parent, const wxWindowID id,
 | |
| 	   	const wxPoint& pos, const wxSize& size, long style)
 | |
| 	: wxListCtrl(parent, id, pos, size, style) // | wxLC_VIRTUAL)
 | |
| {
 | |
| 	Init();
 | |
| }
 | |
| 
 | |
| void JitBlockList::Init()
 | |
| {
 | |
| 	InsertColumn(COLUMN_ADDRESS, _T("Address"));
 | |
| 	InsertColumn(COLUMN_PPCSIZE, _T("PPC Size"));
 | |
| 	InsertColumn(COLUMN_X86SIZE, _T("x86 Size"));
 | |
| 	InsertColumn(COLUMN_NAME, _T("Symbol"));
 | |
| 	InsertColumn(COLUMN_FLAGS, _T("Flags"));
 | |
| 	InsertColumn(COLUMN_NUMEXEC, _T("NumExec"));
 | |
| 	InsertColumn(COLUMN_COST, _T("Cost"));
 | |
| }
 | |
| 
 | |
| void JitBlockList::Update()
 | |
| {
 | |
| }
 |