2008-12-08 05:30:24 +00:00
|
|
|
// Copyright (C) 2003-2008 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>
|
2009-02-23 20:29:55 +00:00
|
|
|
|
2008-12-08 05:30:24 +00:00
|
|
|
#include "JitWindow.h"
|
|
|
|
#include "HW/CPU.h"
|
|
|
|
#include "PowerPC/PowerPC.h"
|
2009-02-23 20:29:55 +00:00
|
|
|
#if JITTEST
|
2008-12-08 05:30:24 +00:00
|
|
|
#include "PowerPC/Jit64/Jit.h"
|
2009-02-23 20:29:55 +00:00
|
|
|
#else
|
|
|
|
#include "PowerPC/Jit64IL/Jit.h"
|
|
|
|
#endif
|
|
|
|
#include "PowerPC/JitCommon/JitCache.h"
|
2008-12-18 11:09:03 +00:00
|
|
|
#include "PowerPC/PPCAnalyst.h"
|
|
|
|
#include "PowerPCDisasm.h"
|
2008-12-08 05:30:24 +00:00
|
|
|
#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"
|
|
|
|
|
|
|
|
// UGLY
|
|
|
|
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, wxFrame)
|
|
|
|
// 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 wxString& title, const wxPoint& pos, const wxSize& size, long style)
|
|
|
|
: wxFrame(parent, id, title, pos, size, style)
|
|
|
|
{
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CJitWindow::~CJitWindow()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CJitWindow::Save(IniFile& _IniFile) const
|
|
|
|
{
|
|
|
|
_IniFile.Set("JitWindow", "x", GetPosition().x);
|
|
|
|
_IniFile.Set("JitWindow", "y", GetPosition().y);
|
|
|
|
_IniFile.Set("JitWindow", "w", GetSize().GetWidth());
|
|
|
|
_IniFile.Set("JitWindow", "h", GetSize().GetHeight());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CJitWindow::Load(IniFile& _IniFile)
|
|
|
|
{
|
|
|
|
int x,y,w,h;
|
|
|
|
_IniFile.Get("JitWindow", "x", &x, GetPosition().x);
|
|
|
|
_IniFile.Get("JitWindow", "y", &y, GetPosition().y);
|
|
|
|
_IniFile.Get("JitWindow", "w", &w, GetSize().GetWidth());
|
|
|
|
_IniFile.Get("JitWindow", "h", &h, GetSize().GetHeight());
|
|
|
|
SetSize(x, y, w, h);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CJitWindow::OnRefresh(wxCommandEvent& /*event*/) {
|
|
|
|
block_list->Update();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CJitWindow::ViewAddr(u32 em_address)
|
|
|
|
{
|
2009-01-21 11:44:36 +00:00
|
|
|
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");
|
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CJitWindow::Compare(u32 em_address)
|
|
|
|
{
|
2008-12-29 02:02:35 +00:00
|
|
|
u8 *xDis = new u8[1<<18];
|
|
|
|
memset(xDis, 0, 1<<18);
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
disassembler x64disasm;
|
|
|
|
x64disasm.set_syntax_intel();
|
|
|
|
|
2008-12-20 12:46:14 +00:00
|
|
|
int block_num = jit.GetBlockCache()->GetBlockNumberFromStartAddress(em_address);
|
2008-12-08 05:30:24 +00:00
|
|
|
if (block_num < 0)
|
|
|
|
{
|
2008-12-19 22:21:29 +00:00
|
|
|
for (int i = 0; i < 500; i++) {
|
2008-12-20 12:46:14 +00:00
|
|
|
block_num = jit.GetBlockCache()->GetBlockNumberFromStartAddress(em_address - 4 * i);
|
2008-12-19 22:21:29 +00:00
|
|
|
if (block_num >= 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (block_num >= 0) {
|
2008-12-20 12:10:59 +00:00
|
|
|
JitBlock *block = jit.GetBlockCache()->GetBlock(block_num);
|
2008-12-19 22:21:29 +00:00
|
|
|
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()));
|
|
|
|
return;
|
|
|
|
}
|
2008-12-08 05:30:24 +00:00
|
|
|
}
|
2008-12-20 12:10:59 +00:00
|
|
|
JitBlock *block = jit.GetBlockCache()->GetBlock(block_num);
|
2008-12-18 11:09:03 +00:00
|
|
|
|
2008-12-19 22:21:29 +00:00
|
|
|
// 800031f0
|
2008-12-19 21:35:57 +00:00
|
|
|
// == Fill in x86 box
|
|
|
|
|
2008-12-20 12:10:59 +00:00
|
|
|
const u8 *code = (const u8 *)jit.GetBlockCache()->GetCompiledCodeFromBlock(block_num);
|
2008-12-19 21:35:57 +00:00
|
|
|
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)
|
|
|
|
{
|
2009-06-12 23:56:31 +00:00
|
|
|
#ifdef _M_X64
|
2008-12-19 21:35:57 +00:00
|
|
|
disasmPtr += x64disasm.disasm64(disasmPtr, disasmPtr, (u8*)disasmPtr, sptr);
|
2009-06-12 23:56:31 +00:00
|
|
|
#else
|
|
|
|
disasmPtr += x64disasm.disasm32(disasmPtr, disasmPtr, (u8*)disasmPtr, sptr);
|
|
|
|
#endif
|
2008-12-19 21:35:57 +00:00
|
|
|
sptr += strlen(sptr);
|
|
|
|
*sptr++ = 13;
|
|
|
|
*sptr++ = 10;
|
|
|
|
num_x86_instructions++;
|
|
|
|
}
|
|
|
|
x86_box->SetValue(wxString::FromAscii((char*)xDis));
|
|
|
|
|
2008-12-18 11:09:03 +00:00
|
|
|
// == Fill in ppc box
|
|
|
|
u32 ppc_addr = block->originalAddress;
|
|
|
|
PPCAnalyst::CodeBuffer code_buffer(32000);
|
|
|
|
PPCAnalyst::BlockStats st;
|
|
|
|
PPCAnalyst::BlockRegStats gpa;
|
|
|
|
PPCAnalyst::BlockRegStats fpa;
|
2008-12-18 13:21:02 +00:00
|
|
|
if (PPCAnalyst::Flatten(ppc_addr, &size, &st, &gpa, &fpa, &code_buffer))
|
|
|
|
{
|
2008-12-18 11:09:03 +00:00
|
|
|
char *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);
|
|
|
|
}
|
2008-12-19 21:35:57 +00:00
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
2009-02-09 20:35:30 +00:00
|
|
|
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));
|
2008-12-19 21:35:57 +00:00
|
|
|
|
2008-12-18 11:09:03 +00:00
|
|
|
ppc_box->SetValue(wxString::FromAscii((char*)xDis));
|
|
|
|
} else {
|
2008-12-20 17:22:30 +00:00
|
|
|
ppc_box->SetValue(wxString::FromAscii(StringFromFormat("(non-code address: %08x)", em_address).c_str()));
|
2009-02-09 20:35:30 +00:00
|
|
|
x86_box->SetValue(wxString::FromAscii("---"));
|
2008-12-18 11:09:03 +00:00
|
|
|
}
|
|
|
|
|
2008-12-08 05:30:24 +00:00
|
|
|
|
|
|
|
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()
|
|
|
|
{
|
|
|
|
}
|