#include "stdafx.h" #include "DebuggerUI.h" #include "Assembler.h" #include "Breakpoints.h" #include "OpInfo.h" #include "Symbols.h" #include void CCommandList::Attach(HWND hWndNew) { CListViewCtrl::Attach(hWndNew); ModifyStyle(LVS_OWNERDRAWFIXED, 0, 0); SetExtendedListViewStyle(LVS_EX_FULLROWSELECT | LVS_EX_DOUBLEBUFFER | LVS_EX_LABELTIP); CDC hDC = GetDC(); float DPIScale = hDC.GetDeviceCaps(LOGPIXELSX) / 96.0f; AddColumn(L"", COL_ARROWS); SetColumnWidth(COL_ARROWS, (int)(30 * DPIScale)); AddColumn(L"Address", COL_ADDRESS); SetColumnWidth(COL_ADDRESS, (int)(70 * DPIScale)); AddColumn(L"Command", COL_COMMAND); SetColumnWidth(COL_COMMAND, (int)(65 * DPIScale)); AddColumn(L"Parameters", COL_PARAMETERS); SetColumnWidth(COL_PARAMETERS, (int)(130 * DPIScale)); AddColumn(L"Symbol", COL_SYMBOL); SetColumnWidth(COL_SYMBOL, (int)(180 * DPIScale)); } CDebugCommandsView * CDebugCommandsView::_this = nullptr; HHOOK CDebugCommandsView::hWinMessageHook = nullptr; CDebugCommandsView::CDebugCommandsView(CDebuggerUI * debugger, SyncEvent & StepEvent) : CDebugDialog(debugger), CToolTipDialog(), m_StepEvent(StepEvent), m_Attached(false) { m_HistoryIndex = -1; m_bIgnoreAddrChange = false; m_StartAddress = 0x80000000; m_Breakpoints = m_Debugger->Breakpoints(); m_bEditing = false; m_CommandListRows = 39; m_RowHeight = 13; g_Settings->RegisterChangeCB(GameRunning_CPU_Running, this, (CSettings::SettingChangedFunc)GameCpuRunningChanged); } CDebugCommandsView::~CDebugCommandsView() { g_Settings->UnregisterChangeCB(GameRunning_CPU_Running, this, (CSettings::SettingChangedFunc)GameCpuRunningChanged); } LRESULT CDebugCommandsView::OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL & /*bHandled*/) { m_StartAddress = (uint32_t)(g_Reg ? g_Reg->m_PROGRAM_COUNTER : 0x80000000); g_Settings->RegisterChangeCB(Debugger_WaitingForStep, this, (CSettings::SettingChangedFunc)StaticWaitingForStepChanged); g_Settings->RegisterChangeCB(Debugger_SteppingOps, this, (CSettings::SettingChangedFunc)StaticSteppingOpsChanged); m_CommandList.Attach(GetDlgItem(IDC_CMD_LIST)); m_BreakpointList.Attach(GetDlgItem(IDC_BP_LIST)); m_AddressEdit.Attach(GetDlgItem(IDC_ADDR_EDIT)); m_PCEdit.Attach(GetDlgItem(IDC_PC_EDIT)); m_ViewPCButton.Attach(GetDlgItem(IDC_VIEWPC_BTN)); m_StepButton.Attach(GetDlgItem(IDC_STEP_BTN)); m_StepOverButton.Attach(GetDlgItem(IDC_STEPOVER_BTN)); m_SkipButton.Attach(GetDlgItem(IDC_SKIP_BTN)); m_GoButton.Attach(GetDlgItem(IDC_GO_BTN)); m_RegisterTabs.Attach(GetDlgItem(IDC_REG_TABS), m_Debugger); m_Scrollbar.Attach(GetDlgItem(IDC_SCRL_BAR)); m_BackButton.Attach(GetDlgItem(IDC_BACK_BTN)); m_ForwardButton.Attach(GetDlgItem(IDC_FORWARD_BTN)); m_OpEdit.Attach(GetDlgItem(IDC_OP_EDIT)); DlgResize_Init(false, true); DlgSavePos_Init(DebuggerUI_CommandsPos); DlgToolTip_Init(); // Setup address input m_AddressEdit.SetDisplayType(CEditNumber32::DisplayHex); m_AddressEdit.SetLimitText(8); // Setup PC register input m_PCEdit.SetDisplayType(CEditNumber32::DisplayHex); m_PCEdit.SetLimitText(8); m_bIgnorePCChange = true; m_PCEdit.SetValue(m_StartAddress, DisplayMode::ZeroExtend); // Setup view PC button m_ViewPCButton.EnableWindow(FALSE); m_StepButton.EnableWindow(FALSE); m_StepOverButton.EnableWindow(FALSE); m_SkipButton.EnableWindow(FALSE); m_GoButton.EnableWindow(FALSE); // Setup breakpoint list m_BreakpointList.ModifyStyle(NULL, LBS_NOTIFY); RefreshBreakpointList(); // Setup list scrollbar m_Scrollbar.SetScrollRange(0, 100, FALSE); m_Scrollbar.SetScrollPos(50, TRUE); // Setup history buttons ToggleHistoryButtons(); // Op editor m_OpEdit.SetCommandsWindow(this); m_bIgnoreAddrChange = true; m_AddressEdit.SetValue(m_StartAddress, DisplayMode::ZeroExtend); ShowAddress(m_StartAddress, TRUE); m_bIgnoreAddrChange = false; if (isStepping()) { m_ViewPCButton.EnableWindow(TRUE); m_StepButton.EnableWindow(TRUE); m_StepOverButton.EnableWindow(TRUE); m_SkipButton.EnableWindow(TRUE); m_GoButton.EnableWindow(TRUE); } _this = this; DWORD dwThreadID = ::GetCurrentThreadId(); hWinMessageHook = SetWindowsHookEx(WH_GETMESSAGE, (HOOKPROC)HookProc, nullptr, dwThreadID); LoadWindowPos(); RedrawCommandsAndRegisters(); WindowCreated(); m_Attached = true; RecompilerCheck(); return TRUE; } void CDebugCommandsView::GameCpuRunningChanged(CDebugCommandsView * DebugCommandsView) { DebugCommandsView->RecompilerCheck(); } void CDebugCommandsView::RecompilerCheck(void) { if (!g_Settings->LoadBool(GameRunning_CPU_Running)) { return; } if (!_this->IsWindow()) { return; } if (g_Settings->LoadBool(Debugger_Enabled) && !g_Settings->LoadBool(Setting_ForceInterpreterCPU) && (CPU_TYPE)g_Settings->LoadDword(Game_CpuType) != CPU_Interpreter) { // TODO: Remove this or fix? MessageBox(L"Debugger support for the recompiler core is experimental.\n\n" L"For optimal experience, enable \"Always use interpreter core\" " L"in advanced settings and restart the emulator.", L"Warning", MB_ICONWARNING | MB_OK); } } void CDebugCommandsView::OnExitSizeMove(void) { SaveWindowPos(true); } LRESULT CDebugCommandsView::OnDestroy(void) { m_Attached = false; g_Settings->UnregisterChangeCB(Debugger_SteppingOps, this, (CSettings::SettingChangedFunc)StaticSteppingOpsChanged); g_Settings->UnregisterChangeCB(Debugger_WaitingForStep, this, (CSettings::SettingChangedFunc)StaticWaitingForStepChanged); UnhookWindowsHookEx(hWinMessageHook); m_OpEdit.Detach(); m_ForwardButton.Detach(); m_BackButton.Detach(); m_Scrollbar.Detach(); m_RegisterTabs.Detach(); m_GoButton.Detach(); m_SkipButton.Detach(); m_StepButton.Detach(); m_StepOverButton.Detach(); m_ViewPCButton.Detach(); m_PCEdit.Detach(); m_AddressEdit.Detach(); m_BreakpointList.Detach(); m_CommandList.Detach(); return 0; } void CDebugCommandsView::InterceptKeyDown(WPARAM wParam, LPARAM /*lParam*/) { switch (wParam) { case VK_F1: CPUSkip(); break; case VK_F2: if (WaitingForStep()) { m_StepEvent.Trigger(); } break; case VK_F3: CPUStepOver(); break; case VK_F4: CPUResume(); break; } } void CDebugCommandsView::InterceptMouseWheel(WPARAM wParam, LPARAM /*lParam*/) { uint32_t newAddress = m_StartAddress - ((short)HIWORD(wParam) / WHEEL_DELTA) * 4; m_StartAddress = newAddress; m_AddressEdit.SetValue(m_StartAddress, DisplayMode::ZeroExtend); } LRESULT CALLBACK CDebugCommandsView::HookProc(int nCode, WPARAM wParam, LPARAM lParam) { MSG * pMsg = (MSG *)lParam; if (pMsg->message == WM_KEYDOWN) { _this->InterceptKeyDown(pMsg->wParam, pMsg->lParam); } else if (pMsg->message == WM_MOUSEWHEEL) { _this->InterceptMouseWheel(pMsg->wParam, pMsg->lParam); } if (nCode < 0) { return CallNextHookEx(hWinMessageHook, nCode, wParam, lParam); } return 0; } void CDebugCommandsView::ClearBranchArrows() { m_BranchArrows.clear(); } void CDebugCommandsView::AddBranchArrow(int startPos, int endPos) { int startMargin = 0; int endMargin = 0; int margin = 0; for (size_t j = 0; j < m_BranchArrows.size(); j++) { BRANCHARROW arrow = m_BranchArrows[j]; // Arrow's start or end position within another arrow's stride if ((startPos >= arrow.startPos && startPos <= arrow.endPos) || (endPos >= arrow.startPos && endPos <= arrow.endPos) || (arrow.startPos <= startPos && arrow.startPos >= endPos)) { if (margin <= arrow.margin) { margin = arrow.margin + 1; } } if (startPos == arrow.startPos) { startMargin = arrow.startMargin + 1; } if (startPos == arrow.endPos) { startMargin = arrow.endMargin + 1; } if (endPos == arrow.startPos) { endMargin = arrow.startMargin + 1; } if (endPos == arrow.endPos) { endMargin = arrow.endMargin + 1; } } m_BranchArrows.push_back({startPos, endPos, startMargin, endMargin, margin}); } void CDebugCommandsView::HistoryPushState() { m_History.push_back(m_StartAddress); m_HistoryIndex = (int)((INT_PTR)(m_History.size() - 1)); ToggleHistoryButtons(); } const char * CDebugCommandsView::GetDataAddressNotes(uint32_t vAddr) { switch (vAddr) { case 0xA3F00000: return "RDRAM_CONFIG_REG/RDRAM_DEVICE_TYPE_REG"; case 0xA3F00004: return "RDRAM_DEVICE_ID_REG"; case 0xA3F00008: return "RDRAM_DELAY_REG"; case 0xA3F0000C: return "RDRAM_MODE_REG"; case 0xA3F00010: return "RDRAM_REF_INTERVAL_REG"; case 0xA3F00014: return "RDRAM_REF_ROW_REG"; case 0xA3F00018: return "RDRAM_RAS_INTERVAL_REG"; case 0xA3F0001C: return "RDRAM_MIN_INTERVAL_REG"; case 0xA3F00020: return "RDRAM_ADDR_SELECT_REG"; case 0xA3F00024: return "RDRAM_DEVICE_MANUF_REG"; case 0xA4040000: return "SP_MEM_ADDR_REG"; case 0xA4040004: return "SP_DRAM_ADDR_REG"; case 0xA4040008: return "SP_RD_LEN_REG"; case 0xA404000C: return "SP_WR_LEN_REG"; case 0xA4040010: return "SP_STATUS_REG"; case 0xA4040014: return "SP_DMA_FULL_REG"; case 0xA4040018: return "SP_DMA_BUSY_REG"; case 0xA404001C: return "SP_SEMAPHORE_REG"; case 0xA4080000: return "SP_PC"; case 0xA4100000: return "DPC_START_REG"; case 0xA4100004: return "DPC_END_REG"; case 0xA4100008: return "DPC_CURRENT_REG"; case 0xA410000C: return "DPC_STATUS_REG"; case 0xA4100010: return "DPC_CLOCK_REG"; case 0xA4100014: return "DPC_BUFBUSY_REG"; case 0xA4100018: return "DPC_PIPEBUSY_REG"; case 0xA410001C: return "DPC_TMEM_REG"; case 0xA4300000: return "MI_INIT_MODE_REG/MI_MODE_REG"; case 0xA4300004: return "MI_VERSION_REG/MI_NOOP_REG"; case 0xA4300008: return "MI_INTR_REG"; case 0xA430000C: return "MI_INTR_MASK_REG"; case 0xA4400000: return "VI_STATUS_REG/VI_CONTROL_REG"; case 0xA4400004: return "VI_ORIGIN_REG/VI_DRAM_ADDR_REG"; case 0xA4400008: return "VI_WIDTH_REG/VI_H_WIDTH_REG"; case 0xA440000C: return "VI_INTR_REG/VI_V_INTR_REG"; case 0xA4400010: return "VI_CURRENT_REG/VI_V_CURRENT_LINE_REG"; case 0xA4400014: return "VI_BURST_REG/VI_TIMING_REG"; case 0xA4400018: return "VI_V_SYNC_REG"; case 0xA440001C: return "VI_H_SYNC_REG"; case 0xA4400020: return "VI_LEAP_REG/VI_H_SYNC_LEAP_REG"; case 0xA4400024: return "VI_H_START_REG/VI_H_VIDEO_REG"; case 0xA4400028: return "VI_V_START_REG/VI_V_VIDEO_REG"; case 0xA440002C: return "VI_V_BURST_REG"; case 0xA4400030: return "VI_X_SCALE_REG"; case 0xA4400034: return "VI_Y_SCALE_REG"; case 0xA4500000: return "AI_DRAM_ADDR_REG"; case 0xA4500004: return "AI_LEN_REG"; case 0xA4500008: return "AI_CONTROL_REG"; case 0xA450000C: return "AI_STATUS_REG"; case 0xA4500010: return "AI_DACRATE_REG"; case 0xA4500014: return "AI_BITRATE_REG"; case 0xA4600000: return "PI_DRAM_ADDR_REG"; case 0xA4600004: return "PI_CART_ADDR_REG"; case 0xA4600008: return "PI_RD_LEN_REG"; case 0xA460000C: return "PI_WR_LEN_REG"; case 0xA4600010: return "PI_STATUS_REG"; case 0xA4600014: return "PI_BSD_DOM1_LAT_REG"; case 0xA4600018: return "PI_BSD_DOM1_PWD_REG"; case 0xA460001C: return "PI_BSD_DOM1_PGS_REG"; case 0xA4600020: return "PI_BSD_DOM1_RLS_REG"; case 0xA4600024: return "PI_BSD_DOM2_LAT_REG"; case 0xA4600028: return "PI_BSD_DOM2_PWD_REG"; case 0xA460002C: return "PI_BSD_DOM2_PGS_REG"; case 0xA4600030: return "PI_BSD_DOM2_RLS_REG"; case 0xA4700000: return "RI_MODE_REG"; case 0xA4700004: return "RI_CONFIG_REG"; case 0xA4700008: return "RI_CURRENT_LOAD_REG"; case 0xA470000C: return "RI_SELECT_REG"; case 0xA4700010: return "RI_REFRESH_REG/RI_COUNT_REG"; case 0xA4700014: return "RI_LATENCY_REG"; case 0xA4700018: return "RI_RERROR_REG"; case 0xA470001C: return "RI_WERROR_REG"; case 0xA4800000: return "SI_DRAM_ADDR_REG"; case 0xA4800004: return "SI_PIF_ADDR_RD64B_REG"; case 0xA4800010: return "SI_PIF_ADDR_WR64B_REG"; case 0xA4800018: return "SI_STATUS_REG"; case 0xA5000500: return "ASIC_DATA"; case 0xA5000504: return "ASIC_MISC_REG"; case 0xA5000508: return "ASIC_STATUS"; case 0xA500050C: return "ASIC_CUR_TK"; case 0xA5000510: return "ASIC_BM_STATUS"; case 0xA5000514: return "ASIC_ERR_SECTOR"; case 0xA5000518: return "ASIC_SEQ_STATUS"; case 0xA500051C: return "ASIC_CUR_SECTOR"; case 0xA5000520: return "ASIC_HARD_RESET"; case 0xA5000524: return "ASIC_C1_S0"; case 0xA5000528: return "ASIC_HOST_SECBYTE"; case 0xA500052C: return "ASIC_C1_S2"; case 0xA5000530: return "ASIC_SEC_BYTE"; case 0xA5000534: return "ASIC_C1_S4"; case 0xA5000538: return "ASIC_C1_S6"; case 0xA500053C: return "ASIC_CUR_ADDR"; case 0xA5000540: return "ASIC_ID_REG"; case 0xA5000544: return "ASIC_TEST_REG"; case 0xA5000548: return "ASIC_TEST_PIN_SEL"; case 0xB0000004: return "Header: Clock rate"; case 0xB0000008: return "Header: Game entry point"; case 0xB000000C: return "Header: Release"; case 0xB0000010: return "Header: CRC1"; case 0xB0000014: return "Header: CRC2"; } return nullptr; } const char * CDebugCommandsView::GetCodeAddressNotes(uint32_t vAddr) { switch (vAddr) { case 0x80000000: return "Exception: TLB Refill"; case 0x80000080: return "Exception: XTLB Refill"; case 0x80000100: return "Exception: Cache error (See A0000100)"; case 0x80000180: return "Exception: General"; case 0xA0000100: return "Exception: Cache error"; case 0xBFC00000: return "Exception: Reset/NMI"; case 0xBFC00200: return "Exception: TLB Refill (boot)"; case 0xBFC00280: return "Exception: XTLB Refill (boot)"; case 0xBFC00300: return "Exception: Cache error (boot)"; case 0xBFC00380: return "Exception: General (boot)"; } if (g_MMU == nullptr) { return nullptr; } uint8_t * rom = g_Rom->GetRomAddress(); uint32_t gameEntryPoint = *(uint32_t *)&rom[0x08]; if (vAddr == gameEntryPoint) { return "Game entry point"; } return nullptr; } void CDebugCommandsView::ShowAddress(uint32_t address, bool top, bool bUserInput) { if (top == TRUE) { m_StartAddress = address - address % 4; if (!bUserInput) { m_bIgnoreAddrChange = true; m_AddressEdit.SetValue(address, DisplayMode::ZeroExtend); } if (!isStepping()) { // Disable buttons m_ViewPCButton.EnableWindow(FALSE); m_StepButton.EnableWindow(FALSE); m_StepOverButton.EnableWindow(FALSE); m_SkipButton.EnableWindow(FALSE); m_GoButton.EnableWindow(FALSE); m_RegisterTabs.SetColorsEnabled(false); } } else { bool bOutOfView = address < m_StartAddress || address > m_StartAddress + (m_CommandListRows - 1) * 4; if (bOutOfView) { m_StartAddress = address - address % 4; m_bIgnoreAddrChange = true; m_AddressEdit.SetValue(address, DisplayMode::ZeroExtend); } if (m_History.size() == 0 || m_History[m_HistoryIndex] != m_StartAddress) { HistoryPushState(); } m_bIgnorePCChange = true; m_PCEdit.SetValue((uint32_t)g_Reg->m_PROGRAM_COUNTER, DisplayMode::ZeroExtend); // Enable buttons m_ViewPCButton.EnableWindow(TRUE); m_StepButton.EnableWindow(TRUE); m_StepOverButton.EnableWindow(TRUE); m_SkipButton.EnableWindow(TRUE); m_GoButton.EnableWindow(TRUE); m_RegisterTabs.SetColorsEnabled(true); } m_CommandList.SetRedraw(FALSE); m_CommandList.DeleteAllItems(); ClearBranchArrows(); m_bvAnnotatedLines.clear(); for (int i = 0; i < m_CommandListRows; i++) { uint32_t opAddr = m_StartAddress + i * 4; m_CommandList.AddItem(i, CCommandList::COL_ARROWS, L" "); char addrStr[9]; sprintf(addrStr, "%08X", opAddr); m_CommandList.AddItem(i, CCommandList::COL_ADDRESS, stdstr(addrStr).ToUTF16().c_str()); COpInfo OpInfo; R4300iOpcode & OpCode = OpInfo.m_OpCode; if (!m_Debugger->DebugLoad_VAddr(opAddr, OpCode.Value)) { m_CommandList.AddItem(i, CCommandList::COL_COMMAND, L"***"); m_bvAnnotatedLines.push_back(false); continue; } R4300iInstruction Instruction(opAddr, OpCode.Value); std::string cmdArgs = Instruction.Param(); CSymbol jalSymbol; // Show subroutine symbol name for JAL target if (OpCode.op == R4300i_JAL) { uint32_t targetAddr = (m_StartAddress & 0xF0000000) | (OpCode.target << 2); if (m_Debugger->SymbolTable()->GetSymbolByAddress(targetAddr, &jalSymbol)) { cmdArgs = jalSymbol.m_Name; } } // Detect reads and writes to mapped registers, cart header data, etc. const char * annotation = nullptr; bool bLoadStoreAnnotation = false; CSymbol memSymbol; if (OpInfo.IsLoadStoreCommand()) { for (int offset = -4; offset > -24; offset -= 4) { R4300iOpcode OpCodeTest; if (!m_Debugger->DebugLoad_VAddr(opAddr + offset, OpCodeTest.Value)) { break; } if (OpCodeTest.op != R4300i_LUI) { continue; } if (OpCodeTest.rt != OpCode.rs) { continue; } uint32_t memAddr = (OpCodeTest.immediate << 16) + (short)OpCode.offset; if (m_Debugger->SymbolTable()->GetSymbolByAddress(memAddr, &memSymbol)) { annotation = memSymbol.m_Name; } else { annotation = GetDataAddressNotes(memAddr); } break; } } if (annotation == nullptr) { annotation = GetCodeAddressNotes(opAddr); } else { bLoadStoreAnnotation = true; } m_CommandList.AddItem(i, CCommandList::COL_COMMAND, stdstr(Instruction.Name()).ToUTF16().c_str()); m_CommandList.AddItem(i, CCommandList::COL_PARAMETERS, stdstr(cmdArgs).ToUTF16().c_str()); // Show routine symbol name for this address CSymbol pcSymbol; if (m_Debugger->SymbolTable()->GetSymbolByAddress(opAddr, &pcSymbol)) { m_CommandList.AddItem(i, CCommandList::COL_SYMBOL, stdstr(pcSymbol.m_Name).ToUTF16().c_str()); m_bvAnnotatedLines.push_back(false); } else if (annotation != nullptr) { const char * annotationFormat = bLoadStoreAnnotation ? "// (%s)" : "// %s"; m_CommandList.AddItem(i, CCommandList::COL_SYMBOL, stdstr_f(annotationFormat, annotation).ToUTF16().c_str()); m_bvAnnotatedLines.push_back(true); } else { m_bvAnnotatedLines.push_back(false); } // Add arrow for branch instruction if (OpInfo.IsBranch()) { int startPos = i; int endPos = startPos + (int16_t)OpCode.offset + 1; AddBranchArrow(startPos, endPos); } // Branch arrow for close J if (OpCode.op == R4300i_J) { uint32_t target = (OpCode.target << 2); int dist = target - (opAddr & 0x3FFFFFF); if (abs(dist) < 0x10000) { int startPos = i; int endPos = startPos + (dist / 4); AddBranchArrow(startPos, endPos); } } } if (!top) // Update registers when called via breakpoint/stepping { m_RegisterTabs.RefreshEdits(); } RefreshBreakpointList(); m_CommandList.SetRedraw(TRUE); } // Highlight command list items and draw branch arrows LRESULT CDebugCommandsView::OnCustomDrawList(NMHDR * pNMHDR) { NMLVCUSTOMDRAW * pLVCD = reinterpret_cast(pNMHDR); DWORD drawStage = pLVCD->nmcd.dwDrawStage; switch (drawStage) { case CDDS_PREPAINT: return (CDRF_NOTIFYITEMDRAW | CDRF_NOTIFYPOSTPAINT); case CDDS_ITEMPREPAINT: return CDRF_NOTIFYSUBITEMDRAW; case (CDDS_ITEMPREPAINT | CDDS_SUBITEM): break; case CDDS_POSTPAINT: DrawBranchArrows(pLVCD->nmcd.hdc); return CDRF_DODEFAULT; default: return CDRF_DODEFAULT; } uint32_t nItem = (uint32_t)pLVCD->nmcd.dwItemSpec; uint32_t nSubItem = pLVCD->iSubItem; uint32_t address = m_StartAddress + (nItem * 4); uint32_t pc = (g_Reg != nullptr) ? (uint32_t)g_Reg->m_PROGRAM_COUNTER : 0; R4300iOpcode pcOpcode; if (!m_Debugger->DebugLoad_VAddr(pc, pcOpcode.Value)) { pcOpcode.Value = 0; } if (nSubItem == CCommandList::COL_ARROWS) { return CDRF_DODEFAULT; } if (nSubItem == CCommandList::COL_ADDRESS) // ADDR { CBreakpoints::BPSTATE bpState = m_Breakpoints->ExecutionBPExists(address); if (bpState == CBreakpoints::BP_SET) { // Breakpoint pLVCD->clrTextBk = RGB(0x44, 0x00, 0x00); pLVCD->clrText = (address == pc && isDebugging()) ? RGB(0xFF, 0xFF, 0x00) : RGB(0xFF, 0xCC, 0xCC); } else if (bpState == CBreakpoints::BP_SET_TEMP) { // Breakpoint pLVCD->clrTextBk = RGB(0x66, 0x44, 0x00); pLVCD->clrText = (address == pc && isDebugging()) ? RGB(0xFF, 0xFF, 0x00) : RGB(0xFF, 0xEE, 0xCC); } else if (address == pc && isStepping()) { // PC pLVCD->clrTextBk = RGB(0x88, 0x88, 0x88); pLVCD->clrText = RGB(0xFF, 0xFF, 0); } else { // Default pLVCD->clrTextBk = RGB(0xEE, 0xEE, 0xEE); pLVCD->clrText = RGB(0x44, 0x44, 0x44); } return CDRF_DODEFAULT; } // (nSubItem == 1 || nSubItem == 2) // Command and arguments COpInfo OpInfo; R4300iOpcode & OpCode = OpInfo.m_OpCode; bool bAddrOkay = m_Debugger->DebugLoad_VAddr(address, OpCode.Value); struct { COLORREF bg; COLORREF fg; } colors; if (!bAddrOkay) { colors = {0xFFFFFF, 0xFF0000}; } else if (address == pc && isStepping()) { colors = {0xFFFFAA, 0x222200}; } else if (IsOpEdited(address)) { colors = {0xFFEEFF, 0xFF00FF}; } else if (OpInfo.IsStackAlloc()) { colors = {0xCCDDFF, 0x001144}; } else if (OpInfo.IsStackFree()) { colors = {0xFFDDDD, 0x440000}; } else if (OpInfo.IsNOP()) { colors = {0xFFFFFF, 0x888888}; } else if (OpInfo.IsJump()) { colors = {0xEEFFEE, 0x006600}; } else if (OpInfo.IsBranch()) { colors = {0xFFFFFF, 0x337700}; } else { colors = {0xFFFFFF, 0x0000000}; } // Gray annotations if (nSubItem == CCommandList::COL_SYMBOL) { if (m_bvAnnotatedLines[nItem]) { colors.fg = 0x666666; } } pLVCD->clrTextBk = _byteswap_ulong(colors.bg) >> 8; pLVCD->clrText = _byteswap_ulong(colors.fg) >> 8; if (!isStepping()) { return CDRF_DODEFAULT; } // Color register usage // TODO: localize to temporary register context (don't look before/after jumps and frame shifts) COLORREF clrUsedRegister = RGB(0xF5, 0xF0, 0xFF); // Light purple COLORREF clrAffectedRegister = RGB(0xFF, 0xF0, 0xFF); // Light pink int pcUsedRegA = 0, pcUsedRegB = 0, pcChangedReg = 0; int curUsedRegA = 0, curUsedRegB = 0, curChangedReg = 0; if (pcOpcode.op == R4300i_SPECIAL) { pcUsedRegA = pcOpcode.rs; pcUsedRegB = pcOpcode.rt; pcChangedReg = pcOpcode.rd; } else { pcUsedRegA = pcOpcode.rs; pcChangedReg = pcOpcode.rt; } if (OpCode.op == R4300i_SPECIAL) { curUsedRegA = OpCode.rs; curUsedRegB = OpCode.rt; curChangedReg = OpCode.rd; } else { curUsedRegA = OpCode.rs; curChangedReg = OpCode.rt; } if (address < pc) { if (curChangedReg != 0 && (pcUsedRegA == curChangedReg || pcUsedRegB == curChangedReg)) { pLVCD->clrTextBk = clrUsedRegister; } } else if (address > pc) { if (pcChangedReg != 0 && (curUsedRegA == pcChangedReg || curUsedRegB == pcChangedReg)) { pLVCD->clrTextBk = clrAffectedRegister; } } return CDRF_DODEFAULT; } LRESULT CDebugCommandsView::OnMeasureItem(UINT /*uMsg*/, WPARAM wParam, LPARAM lParam, BOOL & /*bHandled*/) { if (wParam == IDC_CMD_LIST) { CClientDC dc(m_hWnd); dc.SelectFont(GetFont()); TEXTMETRIC tm; dc.GetTextMetrics(&tm); m_RowHeight = tm.tmHeight + tm.tmExternalLeading; MEASUREITEMSTRUCT * lpMeasureItem = (MEASUREITEMSTRUCT *)lParam; lpMeasureItem->itemHeight = m_RowHeight; } return FALSE; } // Draw branch arrows void CDebugCommandsView::DrawBranchArrows(HDC listDC) { COLORREF colors[] = { RGB(240, 240, 240), // White RGB(30, 135, 255), // Blue RGB(255, 0, 200), // Pink RGB(215, 155, 0), // Yellow RGB(100, 180, 0), // Green RGB(200, 100, 255), // Purple RGB(120, 120, 120), // Gray RGB(0, 220, 160), // Cyan RGB(255, 100, 0), // Orange RGB(255, 255, 0), // Yellow }; int nColors = sizeof(colors) / sizeof(COLORREF); CRect listRect; m_CommandList.GetWindowRect(&listRect); ScreenToClient(&listRect); CRect headRect; m_CommandList.GetHeader().GetWindowRect(&headRect); ScreenToClient(&headRect); int colWidth = m_CommandList.GetColumnWidth(CCommandList::COL_ARROWS); int baseX = colWidth - 4; int baseY = headRect.bottom + 7; CRect paneRect; paneRect.top = headRect.bottom; paneRect.left = 0; paneRect.right = colWidth; paneRect.bottom = listRect.bottom; COLORREF bgColor = RGB(30, 30, 30); CBrush hBrushBg(CreateSolidBrush(bgColor)); FillRect(listDC, &paneRect, hBrushBg); for (size_t i = 0; i < m_BranchArrows.size(); i++) { int colorIdx = i % nColors; COLORREF color = colors[colorIdx]; BRANCHARROW arrow = m_BranchArrows[i]; int begX = baseX - arrow.startMargin * 3; int endX = baseX - arrow.endMargin * 3; int begY = baseY + arrow.startPos * m_RowHeight; int endY = baseY + arrow.endPos * m_RowHeight; bool bEndVisible = true; if (endY < headRect.bottom) { endY = headRect.bottom + 1; bEndVisible = false; } else if (endY > listRect.bottom) { endY = listRect.bottom - 2; bEndVisible = false; } int marginX = baseX - (4 + arrow.margin * 3); // Draw start pointer SetPixel(listDC, begX + 0, begY - 1, color); SetPixel(listDC, begX + 1, begY - 2, color); SetPixel(listDC, begX + 0, begY + 1, color); SetPixel(listDC, begX + 1, begY + 2, color); // Draw outline CPen hPenOutline(CreatePen(PS_SOLID, 3, bgColor)); SelectObject(listDC, hPenOutline); MoveToEx(listDC, begX - 1, begY, nullptr); LineTo(listDC, marginX, begY); LineTo(listDC, marginX, endY); if (bEndVisible) { LineTo(listDC, endX + 2, endY); } // Draw fill line CPen hPen(CreatePen(PS_SOLID, 1, color)); SelectObject(listDC, hPen); MoveToEx(listDC, begX - 1, begY, nullptr); LineTo(listDC, marginX, begY); LineTo(listDC, marginX, endY); if (bEndVisible) { LineTo(listDC, endX + 2, endY); } // Draw end pointer if (bEndVisible) { SetPixel(listDC, endX - 0, endY - 1, color); SetPixel(listDC, endX - 1, endY - 2, color); SetPixel(listDC, endX - 1, endY - 1, color); SetPixel(listDC, endX - 0, endY + 1, color); SetPixel(listDC, endX - 1, endY + 2, color); SetPixel(listDC, endX - 1, endY + 1, color); SetPixel(listDC, endX - 1, endY + 3, RGB(30, 30, 30)); SetPixel(listDC, endX - 1, endY - 3, RGB(30, 30, 30)); } } } void CDebugCommandsView::RefreshBreakpointList() { m_BreakpointList.ResetContent(); char rowStr[16]; CBreakpoints::breakpoints_t ReadBreakPoints = m_Breakpoints->ReadMem(); for (CBreakpoints::breakpoints_t::iterator itr = ReadBreakPoints.begin(); itr != ReadBreakPoints.end(); itr++) { sprintf(rowStr, "R %s%08X", itr->second ? "T " : "", itr->first); int index = m_BreakpointList.AddString(stdstr(rowStr).ToUTF16().c_str()); m_BreakpointList.SetItemData(index, itr->first); } CBreakpoints::breakpoints_t WriteBreakPoints = m_Breakpoints->WriteMem(); for (CBreakpoints::breakpoints_t::iterator itr = WriteBreakPoints.begin(); itr != WriteBreakPoints.end(); itr++) { sprintf(rowStr, "W %s%08X", itr->second ? "T " : "", itr->first); int index = m_BreakpointList.AddString(stdstr(rowStr).ToUTF16().c_str()); m_BreakpointList.SetItemData(index, itr->first); } CBreakpoints::breakpoints_t ExecutionBreakPoints = m_Breakpoints->Execution(); for (CBreakpoints::breakpoints_t::iterator itr = ExecutionBreakPoints.begin(); itr != ExecutionBreakPoints.end(); itr++) { sprintf(rowStr, "E %s%08X", itr->second ? "T " : "", itr->first); int index = m_BreakpointList.AddString(stdstr(rowStr).ToUTF16().c_str()); m_BreakpointList.SetItemData(index, itr->first); } } void CDebugCommandsView::RemoveSelectedBreakpoints() { int nItem = m_BreakpointList.GetCurSel(); if (nItem == LB_ERR) { return; } wchar_t itemText[32]; m_BreakpointList.GetText(nItem, itemText); uint32_t address = (uint32_t)(m_BreakpointList.GetItemData(nItem)); switch (itemText[0]) { case L'E': m_Breakpoints->RemoveExecution(address); break; case L'W': m_Breakpoints->WBPRemove(address); break; case L'R': m_Breakpoints->RBPRemove(address); break; } RefreshBreakpointList(); } void CDebugCommandsView::CPUSkip() { g_Settings->SaveBool(Debugger_SkipOp, true); if (WaitingForStep()) { m_StepEvent.Trigger(); } } void CDebugCommandsView::CPUResume() { g_Settings->SaveBool(Debugger_SteppingOps, false); if (WaitingForStep()) { m_StepEvent.Trigger(); } } void CDebugCommandsView::CPUStepOver() { COpInfo opInfo; if (!m_Debugger->DebugLoad_VAddr((uint32_t)g_Reg->m_PROGRAM_COUNTER, opInfo.m_OpCode.Value)) { return; } if (opInfo.IsJAL()) { // Put temp breakpoints on return address and resume m_Breakpoints->AddExecution((uint32_t)g_Reg->m_PROGRAM_COUNTER + 8, true); CPUResume(); } else { // Normal step if (WaitingForStep()) { m_StepEvent.Trigger(); } } } LRESULT CDebugCommandsView::OnBackButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { if (m_HistoryIndex > 0) { m_HistoryIndex--; m_AddressEdit.SetValue(m_History[m_HistoryIndex], DisplayMode::ZeroExtend); ToggleHistoryButtons(); } return FALSE; } LRESULT CDebugCommandsView::OnForwardButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { if (m_History.size() > 0 && m_HistoryIndex < (int)m_History.size() - 1) { m_HistoryIndex++; m_AddressEdit.SetValue(m_History[m_HistoryIndex], DisplayMode::ZeroExtend); ToggleHistoryButtons(); } return FALSE; } LRESULT CDebugCommandsView::OnViewPCButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { if (g_Reg != nullptr && isStepping()) { ShowAddress((uint32_t)g_Reg->m_PROGRAM_COUNTER, TRUE); } return FALSE; } LRESULT CDebugCommandsView::OnSymbolsButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_Debugger->OpenSymbolsWindow(); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuRunTo(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { // Add temp breakpoints and resume m_Breakpoints->AddExecution(m_SelectedAddress, true); return FALSE; } LRESULT CDebugCommandsView::OnGoButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { CPUResume(); m_AddressEdit.SetFocus(); return FALSE; } LRESULT CDebugCommandsView::OnStepButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { if (WaitingForStep()) { m_StepEvent.Trigger(); } return FALSE; } LRESULT CDebugCommandsView::OnStepOverButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { CPUStepOver(); return FALSE; } LRESULT CDebugCommandsView::OnSkipButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { CPUSkip(); m_AddressEdit.SetFocus(); return FALSE; } LRESULT CDebugCommandsView::OnClearBPButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_Breakpoints->BPClear(); RefreshBreakpointList(); ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnAddBPButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_AddBreakpointDlg.DoModal(m_Debugger); RefreshBreakpointList(); ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnRemoveBPButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { RemoveSelectedBreakpoints(); ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnCopyTabRegistersButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_RegisterTabs.CopyTabRegisters(); return FALSE; } LRESULT CDebugCommandsView::OnCopyAllRegistersButton(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_RegisterTabs.CopyAllRegisters(); return FALSE; } LRESULT CDebugCommandsView::OnCancel(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { EndDialog(0); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuEdit(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { BeginOpEdit(m_SelectedAddress); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuInsertNOP(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { EditOp(m_SelectedAddress, 0x00000000); ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuRestore(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { RestoreOp(m_SelectedAddress); ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuRestoreAll(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { RestoreAllOps(); ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuAddSymbol(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_AddSymbolDlg.DoModal(m_Debugger, m_SelectedAddress, SYM_CODE); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuFollowJump(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { HistoryPushState(); ShowAddress(m_FollowAddress, TRUE); HistoryPushState(); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuViewMemory(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_Debugger->Debug_ShowMemoryLocation(m_FollowAddress, true); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuToggleBP(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { if (m_Breakpoints->ExecutionBPExists(m_SelectedAddress)) { m_Breakpoints->RemoveExecution(m_SelectedAddress); } else { m_Breakpoints->AddExecution(m_SelectedAddress); } ShowAddress(m_StartAddress, TRUE); return FALSE; } LRESULT CDebugCommandsView::OnPopupmenuClearBP(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { m_Breakpoints->EBPClear(); ShowAddress(m_StartAddress, TRUE); return FALSE; } void CDebugCommandsView::BeginOpEdit(uint32_t address) { uint32_t opcode; if (!m_Debugger->DebugLoad_VAddr(address, opcode)) { return; } CRect listRect; m_CommandList.GetWindowRect(&listRect); ScreenToClient(&listRect); m_bEditing = true; //ShowAddress(address, FALSE); int nItem = (address - m_StartAddress) / 4; CRect itemRect; m_CommandList.GetSubItemRect(nItem, CCommandList::COL_COMMAND, 0, &itemRect); //itemRect.bottom += 0; itemRect.left += listRect.left + 3; itemRect.right += 100; m_OpEdit.ShowWindow(SW_SHOW); m_OpEdit.MoveWindow(&itemRect); m_OpEdit.BringWindowToTop(); m_OpEdit.SetWindowText(stdstr(R4300iInstruction(address, opcode).NameAndParam()).ToUTF16().c_str()); m_OpEdit.SetFocus(); m_OpEdit.SetSelAll(); m_CommandList.RedrawWindow(); m_OpEdit.RedrawWindow(); } void CDebugCommandsView::EndOpEdit() { m_bEditing = false; m_OpEdit.SetWindowText(L""); m_OpEdit.ShowWindow(SW_HIDE); } LRESULT CDebugCommandsView::OnAddrChanged(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL & /*bHandled*/) { if (!m_Attached) { return 0; } if (m_bIgnoreAddrChange) { m_bIgnoreAddrChange = false; return 0; } uint32_t address = m_AddressEdit.GetValue(); ShowAddress(address, TRUE, TRUE); return 0; } LRESULT CDebugCommandsView::OnPCChanged(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hWndCtl*/, BOOL & /*bHandled*/) { if (!m_Attached) { return 0; } if (m_bIgnorePCChange) { m_bIgnorePCChange = false; return 0; } if (g_Reg != nullptr && isStepping()) { g_Reg->m_PROGRAM_COUNTER = m_PCEdit.GetValue(); } return 0; } LRESULT CDebugCommandsView::OnCommandListClicked(NMHDR * /*pNMHDR*/) { EndOpEdit(); return 0; } LRESULT CDebugCommandsView::OnCommandListDblClicked(NMHDR * pNMHDR) { // Set PC breakpoint NMITEMACTIVATE * pIA = reinterpret_cast(pNMHDR); int nItem = pIA->iItem; uint32_t address = m_StartAddress + nItem * 4; if (m_Breakpoints->ExecutionBPExists(address)) { m_Breakpoints->RemoveExecution(address); } else { m_Breakpoints->AddExecution(address); } // Cancel blue highlight m_AddressEdit.SetFocus(); RefreshBreakpointList(); return 0; } LRESULT CDebugCommandsView::OnCommandListRightClicked(NMHDR * pNMHDR) { EndOpEdit(); NMITEMACTIVATE * pIA = reinterpret_cast(pNMHDR); int nItem = pIA->iItem; uint32_t address = m_StartAddress + nItem * 4; m_SelectedAddress = address; if (!m_Debugger->DebugLoad_VAddr(m_SelectedAddress, m_SelectedOpCode.Value)) { return 0; } HMENU hMenu = LoadMenu(GetModuleHandle(nullptr), MAKEINTRESOURCE(IDR_OP_POPUP)); HMENU hPopupMenu = GetSubMenu(hMenu, 0); if (m_SelectedOpInfo.IsStaticJump()) { m_FollowAddress = (m_SelectedAddress & 0xF0000000) | (m_SelectedOpCode.target * 4); } else if (m_SelectedOpInfo.IsBranch()) { m_FollowAddress = m_SelectedAddress + ((int16_t)m_SelectedOpCode.offset + 1) * 4; } else { EnableMenuItem(hPopupMenu, ID_POPUPMENU_FOLLOWJUMP, MF_DISABLED | MF_GRAYED); } if (m_SelectedOpInfo.IsLoadStoreCommand()) { m_FollowAddress = g_Reg->m_GPR[m_SelectedOpCode.base].UW[0] + (int16_t)m_SelectedOpCode.offset; } else { EnableMenuItem(hPopupMenu, ID_POPUPMENU_VIEWMEMORY, MF_DISABLED | MF_GRAYED); } if (!IsOpEdited(m_SelectedAddress)) { EnableMenuItem(hPopupMenu, ID_POPUPMENU_RESTORE, MF_DISABLED | MF_GRAYED); } if (m_EditedOps.size() == 0) { EnableMenuItem(hPopupMenu, ID_POPUPMENU_RESTOREALL, MF_DISABLED | MF_GRAYED); } if (m_SelectedOpInfo.IsNOP()) { EnableMenuItem(hPopupMenu, ID_POPUPMENU_INSERTNOP, MF_DISABLED | MF_GRAYED); } if (m_Breakpoints->Execution().size() == 0) { EnableMenuItem(hPopupMenu, ID_POPUPMENU_CLEARBPS, MF_DISABLED | MF_GRAYED); } POINT mouse; GetCursorPos(&mouse); TrackPopupMenu(hPopupMenu, TPM_LEFTALIGN, mouse.x, mouse.y, 0, m_hWnd, nullptr); DestroyMenu(hMenu); return 0; } LRESULT CDebugCommandsView::OnListBoxClicked(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL & /*bHandled*/) { if (wID == IDC_BP_LIST) { int index = m_BreakpointList.GetCaretIndex(); uint32_t address = (uint32_t)m_BreakpointList.GetItemData(index); int len = m_BreakpointList.GetTextLen(index); std::wstring rowText; rowText.resize(len); m_BreakpointList.GetText(index, (wchar_t *)rowText.data()); if (rowText[0] == L'E') { ShowAddress(address, true); } else { m_Debugger->Debug_ShowMemoryLocation(address, true); } } return FALSE; } LRESULT CDebugCommandsView::OnActivate(UINT /*uMsg*/, WPARAM wParam, LPARAM /*lParam*/, BOOL & /*bHandled*/) { if (!m_Attached) { return false; } if (LOWORD(wParam) != WA_INACTIVE) { ShowAddress(m_StartAddress, TRUE); RefreshBreakpointList(); } return FALSE; } void CDebugCommandsView::RedrawCommandsAndRegisters() { CRect listRect; m_CommandList.GetWindowRect(listRect); CRect headRect; CHeaderCtrl listHead = m_CommandList.GetHeader(); listHead.GetWindowRect(&headRect); int rowsHeight = listRect.Height() - headRect.Height(); int nRows = (rowsHeight / m_RowHeight); if (m_CommandListRows != nRows) { m_CommandListRows = nRows; ShowAddress(m_StartAddress, TRUE); } m_RegisterTabs.RedrawCurrentTab(); // Fix command list header listHead.ResizeClient(listRect.Width(), headRect.Height()); } LRESULT CDebugCommandsView::OnSizing(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL & /*bHandled*/) { RedrawCommandsAndRegisters(); return FALSE; } LRESULT CDebugCommandsView::OnScroll(UINT /*uMsg*/, WPARAM wParam, LPARAM /*lParam*/, BOOL & /*bHandled*/) { WORD type = LOWORD(wParam); switch (type) { case SB_LINEUP: ShowAddress(m_StartAddress - 4, TRUE); break; case SB_LINEDOWN: ShowAddress(m_StartAddress + 4, TRUE); break; case SB_THUMBTRACK: { //int scrollPos = HIWORD(wParam); //ShowAddress(m_StartAddress + (scrollPos - 50) * 4, TRUE); } break; } return FALSE; } void CDebugCommandsView::WaitingForStepChanged(void) { if (WaitingForStep()) { ShowAddress((uint32_t)g_Reg->m_PROGRAM_COUNTER, false); m_Debugger->Debug_RefreshStackWindow(); m_Debugger->Debug_RefreshStackTraceWindow(); m_StepButton.EnableWindow(true); m_StepOverButton.EnableWindow(true); m_GoButton.EnableWindow(true); m_AddressEdit.SetFocus(); } else { m_StepButton.EnableWindow(false); m_StepOverButton.EnableWindow(false); m_GoButton.EnableWindow(false); } } void CDebugCommandsView::SteppingOpsChanged(void) { if (!g_Settings->LoadBool(Debugger_SteppingOps)) { m_Debugger->Debug_RefreshStackWindow(); m_Debugger->Debug_RefreshStackTraceWindow(); m_RegisterTabs.SetColorsEnabled(false); m_RegisterTabs.RefreshEdits(); ShowAddress(m_StartAddress, TRUE); } } void CDebugCommandsView::Reset() { ClearEditedOps(); m_History.clear(); ToggleHistoryButtons(); } void CDebugCommandsView::ClearEditedOps() { m_EditedOps.clear(); } BOOL CDebugCommandsView::IsOpEdited(uint32_t address) { for (size_t i = 0; i < m_EditedOps.size(); i++) { if (m_EditedOps[i].address == address) { return TRUE; } } return FALSE; } void CDebugCommandsView::EditOp(uint32_t address, uint32_t op, bool bRefresh) { uint32_t currentOp; if (!m_Debugger->DebugLoad_VAddr(address, currentOp)) { return; } if (currentOp == op) { return; } m_Debugger->DebugStore_VAddr(address, op); if (!IsOpEdited(address)) { m_EditedOps.push_back({address, currentOp}); } if (bRefresh) { ShowAddress(m_StartAddress, TRUE); } } void CDebugCommandsView::RestoreOp(uint32_t address) { for (size_t i = 0; i < m_EditedOps.size(); i++) { if (m_EditedOps[i].address == address) { m_Debugger->DebugStore_VAddr(m_EditedOps[i].address, m_EditedOps[i].originalOp); m_EditedOps.erase(m_EditedOps.begin() + i); break; } } } void CDebugCommandsView::RestoreAllOps() { int lastIndex = (int)((INT_PTR)(m_EditedOps.size() - 1)); for (int i = lastIndex; i >= 0; i--) { m_Debugger->DebugStore_VAddr(m_EditedOps[i].address, m_EditedOps[i].originalOp); m_EditedOps.erase(m_EditedOps.begin() + i); } } void CDebugCommandsView::ShowPIRegTab() { m_RegisterTabs.SetCurSel(2); m_RegisterTabs.ShowTab(2); } LRESULT CDebugCommandsView::OnRegisterTabChange(NMHDR * /*pNMHDR*/) { int nPage = m_RegisterTabs.GetCurSel(); m_RegisterTabs.ShowTab(nPage); m_RegisterTabs.RedrawCurrentTab(); m_RegisterTabs.RedrawWindow(); return FALSE; } void CDebugCommandsView::ToggleHistoryButtons() { if (m_BackButton.m_hWnd != nullptr) { m_BackButton.EnableWindow(m_History.size() != 0 && m_HistoryIndex > 0 ? TRUE : FALSE); } if (m_ForwardButton.m_hWnd != nullptr) { m_ForwardButton.EnableWindow(m_History.size() != 0 && m_HistoryIndex < (int)m_History.size() - 1 ? TRUE : FALSE); } } // Opcode editor LRESULT CDebugCommandsView::OnOpEditKeyDown(UINT /*uMsg*/, WPARAM wParam, LPARAM /*lParam*/, BOOL & bHandled) { if (wParam == VK_UP) { m_SelectedAddress -= 4; BeginOpEdit(m_SelectedAddress); bHandled = TRUE; } else if (wParam == VK_DOWN) { m_SelectedAddress += 4; BeginOpEdit(m_SelectedAddress); bHandled = TRUE; } else if (wParam == VK_RETURN) { uint32_t op; bool bValid = CAssembler::AssembleLine(GetCWindowText(m_OpEdit).c_str(), &op, m_SelectedAddress); if (bValid) { m_OpEdit.SetWindowText(L""); EditOp(m_SelectedAddress, op); m_SelectedAddress += 4; BeginOpEdit(m_SelectedAddress); } bHandled = TRUE; } else if (wParam == VK_ESCAPE) { EndOpEdit(); bHandled = TRUE; } return 1; } LRESULT CDebugCommandsView::OnOpEditChanged(WORD /*wNotifyCode*/, WORD /*wID*/, HWND /*hwnd*/, BOOL & /*bHandled*/) { // Handle multiline input std::string text = GetCWindowText(m_OpEdit); if (strchr(text.c_str(), '\n') == nullptr) { return FALSE; } EndOpEdit(); for (size_t i = 0, n = text.size(); i < n; i++) { if (text[i] == '\r') { text[i] = '\n'; } } char * tokctx; char * line = strtok_s((char *)text.c_str(), "\n", &tokctx); while (line != nullptr) { if (strlen(line) != 0) { uint32_t op; bool bValid = CAssembler::AssembleLine(line, &op, m_SelectedAddress); if (bValid) { EditOp(m_SelectedAddress, op, false); m_SelectedAddress += 4; } else { ShowAddress(m_StartAddress, TRUE); BeginOpEdit(m_SelectedAddress); m_OpEdit.SetWindowText(stdstr(line).ToUTF16().c_str()); return FALSE; } } line = strtok_s(nullptr, "\n", &tokctx); } ShowAddress(m_StartAddress, TRUE); BeginOpEdit(m_SelectedAddress); return FALSE; } LRESULT CEditOp::OnKeyDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL & bHandled) { if (m_CommandsWindow == nullptr) { return FALSE; } return m_CommandsWindow->OnOpEditKeyDown(uMsg, wParam, lParam, bHandled); } void CEditOp::SetCommandsWindow(CDebugCommandsView * commandsWindow) { m_CommandsWindow = commandsWindow; } BOOL CEditOp::Attach(HWND hWndNew) { return SubclassWindow(hWndNew); }