// 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 #include #include #include #include #include "JitWindow.h" #include "HW/CPU.h" #include "PowerPC/PowerPC.h" #if JITTEST #include "PowerPC/Jit64/Jit.h" #else #include "PowerPC/Jit64IL/Jit.h" #endif #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" // 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) { 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())); 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; if (PPCAnalyst::Flatten(ppc_addr, &size, &st, &gpa, &fpa, &code_buffer)) { 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); } // 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() { }