project64/Source/Project64/N64 System/Debugger/Debugger - Memory Search.cpp

490 lines
15 KiB
C++

#include "Debugger UI.h"
CDebugMemorySearch::CDebugMemorySearch(CN64System * System, CMipsMemory * MMU, CDebugger * debugger) :
CDebugDialog<CDebugMemorySearch>(MMU,debugger),
m_System(System)
{
}
CDebugMemorySearch::~CDebugMemorySearch()
{
}
void CDebugMemorySearch::AddAlignmentOptions (CComboBox & ctrl)
{
int Index = ctrl.AddString("32 bits (aligned)");
ctrl.SetItemData(Index,_32Bit);
Index = ctrl.AddString("16bits (aligned)");
ctrl.SetItemData(Index,_16Bit);
Index = ctrl.AddString("8bits");
ctrl.SetCurSel(Index);
ctrl.SetItemData(Index,_8Bit);
}
LRESULT CDebugMemorySearch::OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/)
{
m_PAddrStart.Attach(GetDlgItem(IDC_PADDR_START));
m_PAddrStart.SetDisplayType(CEditNumber::DisplayHex);
m_SearchLen.Attach(GetDlgItem(IDC_ADDR_END));
m_SearchLen.SetDisplayType(CEditNumber::DisplayHex);
m_SearchValue.Attach(GetDlgItem(IDC_SEARCH_VALUE));
m_SearchValue.SetDisplayType(CEditNumber::DisplayDec);
m_SearchValue.SetValue(0);
m_MaxSearch.Attach(GetDlgItem(IDC_MAX_SEARCH));
m_MaxSearch.SetDisplayType(CEditNumber::DisplayDec);
m_MaxSearch.SetValue(50000);
m_UnknownSize.Attach(GetDlgItem(IDC_UNKNOWN_ALIGN));
AddAlignmentOptions(m_UnknownSize);
m_ValueSize.Attach(GetDlgItem(IDC_VALUE_ALIGN));
AddAlignmentOptions(m_ValueSize);
m_SearchResults.Attach(GetDlgItem(IDC_LST_RESULTS));
m_SearchResults.AddColumn("ID",0);
m_SearchResults.AddColumn("PAddr",1);
m_SearchResults.AddColumn("Value",2);
m_SearchResults.SetColumnWidth(0,50);
m_SearchResults.SetColumnWidth(1,75);
m_SearchResults.SetColumnWidth(2,75);
m_SearchResults.SetExtendedListViewStyle(LVS_EX_FULLROWSELECT);
m_UnknownOptions.Attach(GetDlgItem(IDC_CMB_UNKNOWN));
m_HaveResults = false;
FixUnknownOptions(true);
SendMessage(GetDlgItem(IDC_RADIO_VALUE),BM_SETCHECK, BST_CHECKED,0);
BOOL bHandled;
OnClicked(0,IDC_BTN_RDRAM,NULL,bHandled);
OnClicked(0,IDC_RADIO_VALUE,NULL,bHandled);
WindowCreated();
return TRUE;
}
LRESULT CDebugMemorySearch::OnClicked(WORD wNotifyCode, WORD wID, HWND hWndCtl, BOOL& bHandled)
{
switch(wID)
{
case IDCANCEL:
EndDialog(0);
break;
case IDC_BTN_RDRAM:
m_PAddrStart.SetValue(0,true,true);
m_SearchLen.SetValue(m_MMU->RdramSize(),true,true);
break;
case IDC_BTN_ROM:
m_PAddrStart.SetValue(0x10000000,true,true);
m_SearchLen.SetValue(m_MMU->RomFileSize(),true,true);
break;
case IDC_BTN_SPMEM:
m_PAddrStart.SetValue(0x04000000,true,true);
m_SearchLen.SetValue(0x2000,true,true);
break;
case IDC_SEARCH_HEX:
{
bool bChecked = (SendMessage(hWndCtl, BM_GETSTATE, 0,0) & BST_CHECKED) != 0;
m_SearchValue.SetDisplayType( bChecked ? CEditNumber::DisplayHex : CEditNumber::DisplayDec);
}
break;
case ID_POPUP_SHOWINMEMORYVIEWER:
{
LONG iItem = m_SearchResults.GetNextItem(-1, LVNI_SELECTED);
if (iItem == -1)
{
break;
}
int ItemId = m_SearchResults.GetItemData(iItem);
SearchResultItem & Result = m_SearchResult[ItemId];
m_Debugger->Debug_ShowMemoryLocation(Result.PAddr, false);
}
break;
case IDC_RADIO_UNKNOWN:
EnableUnknownOptions(true);
EnableValueOptions(false);
EnableTextOptions(false);
EnableJalOptions(false);
break;
case IDC_RADIO_VALUE:
EnableUnknownOptions(false);
EnableValueOptions(true);
EnableTextOptions(false);
EnableJalOptions(false);
break;
case IDC_RADIO_TEXT:
EnableUnknownOptions(false);
EnableValueOptions(false);
EnableTextOptions(true);
EnableJalOptions(false);
break;
case IDC_RADIO_JAL:
EnableUnknownOptions(false);
EnableValueOptions(false);
EnableTextOptions(false);
EnableJalOptions(true);
break;
case IDC_BTN_SEARCH:
if (SendMessage(GetDlgItem(IDC_RADIO_UNKNOWN),BM_GETSTATE, 0,0) == BST_CHECKED)
{
m_System->ExternalEvent(PauseCPU_SearchMemory);
SearchForUnknown();
m_System->ExternalEvent(ResumeCPU_SearchMemory);
break;
}
if (SendMessage(GetDlgItem(IDC_RADIO_VALUE),BM_GETSTATE, 0,0) == BST_CHECKED)
{
m_System->ExternalEvent(PauseCPU_SearchMemory);
SearchForValue();
m_System->ExternalEvent(ResumeCPU_SearchMemory);
break;
}
if (SendMessage(GetDlgItem(IDC_RADIO_TEXT),BM_GETSTATE, 0,0) == BST_CHECKED)
{
m_System->ExternalEvent(PauseCPU_SearchMemory);
SearchForText();
m_System->ExternalEvent(ResumeCPU_SearchMemory);
break;
}
case IDC_RESET_BUTTON:
Reset();
break;
}
return FALSE;
}
LRESULT CDebugMemorySearch::OnResultRClick ( LPNMHDR lpnmh )
{
LONG iItem = m_SearchResults.GetNextItem(-1, LVNI_SELECTED);
if (iItem == -1)
{
return true;
}
//Load the menu
HMENU hMenu = LoadMenu(GetModuleHandle(NULL),MAKEINTRESOURCE(IDR_MEM_SEARCH));
HMENU hPopupMenu = GetSubMenu(hMenu,0);
//Get the current Mouse location
POINT Mouse;
GetCursorPos(&Mouse);
//Show the menu
TrackPopupMenu(hPopupMenu, 0, Mouse.x, Mouse.y, 0,m_hWnd, NULL);
DestroyMenu(hMenu);
return true;
}
void CDebugMemorySearch::EnableValueOptions( bool Enable )
{
if (Enable)
{
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),m_HaveResults ? "Search Results" : "Search");
}
::EnableWindow(GetDlgItem(IDC_SEARCH_VALUE),Enable);
::EnableWindow(GetDlgItem(IDC_SEARCH_HEX),Enable);
::EnableWindow(GetDlgItem(IDC_VALUE_ALIGN),m_HaveResults ? false : Enable );
}
void CDebugMemorySearch::EnableTextOptions( bool Enable )
{
if (Enable)
{
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),m_HaveResults ? "Search Results" : "Search");
}
::EnableWindow(GetDlgItem(IDC_SEARCH_TEXT),Enable);
::EnableWindow(GetDlgItem(IDC_CASE_SENSITIVE),Enable);
}
void CDebugMemorySearch::EnableJalOptions( bool Enable )
{
if (Enable)
{
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),m_HaveResults ? "Search Results" : "Search");
}
::EnableWindow(GetDlgItem(IDC_JAL_ADDR),Enable);
}
void CDebugMemorySearch::EnableUnknownOptions( bool Enable )
{
if (m_UnknownOptions.GetCount() > 1)
{
::EnableWindow(GetDlgItem(IDC_UNKNOWN_ALIGN),m_HaveResults ? false : Enable );
if (Enable)
{
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH), m_HaveResults ? "Search Results" : "Search");
}
} else {
::EnableWindow(GetDlgItem(IDC_UNKNOWN_ALIGN), false );
if (Enable)
{
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH), "Create");
}
}
::EnableWindow(GetDlgItem(IDC_CMB_UNKNOWN),Enable);
}
void CDebugMemorySearch::SearchForValue( void )
{
MemorySize Size = (MemorySize)m_ValueSize.GetItemData(m_ValueSize.GetCurSel());
DWORD Value = m_SearchValue.GetValue();
DWORD StartAddress = m_PAddrStart.GetValue();
DWORD Len = m_SearchLen.GetValue();
DWORD MaxSearch = m_MaxSearch.GetValue();
DWORD MoveSize = (Size == _32Bit ? 4 : (Size == _16Bit ? 2 : 1));
m_UnknownSize.SetCurSel(m_ValueSize.GetCurSel());
LPCTSTR DisplayStr = "0x%08X";
if (Size == _16Bit) { DisplayStr = "0x%04X"; }
else if (Size == _8Bit) { DisplayStr = "0x%04X"; }
if (!m_HaveResults)
{
m_HaveResults = true;
FixUnknownOptions(false);
m_SearchResults.DeleteAllItems();
DWORD ItemsAdded = 0;
while (m_MMU->SearchForValue(Value,Size,StartAddress,Len))
{
SearchResultItem Result;
Result.PAddr = StartAddress;
Result.Value = Value;
char LocationStr[20];
sprintf(LocationStr,"%d",ItemsAdded + 1);
int Index = m_SearchResults.AddItem(ItemsAdded,0,LocationStr);
m_SearchResults.SetItemData(Index,m_SearchResult.size());
m_SearchResult.push_back(Result);
sprintf(LocationStr,"0x%08X",StartAddress);
m_SearchResults.SetItemText(Index,1,LocationStr);
sprintf(LocationStr,DisplayStr,Value);
m_SearchResults.SetItemText(Index,2,LocationStr);
sprintf(LocationStr,DisplayStr,Value);
m_SearchResults.SetItemText(Index,3,LocationStr);
StartAddress += MoveSize;
Len -= MoveSize;
ItemsAdded += 1;
if (ItemsAdded >= MaxSearch)
{
break;
}
}
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),"Search Results");
::ShowWindow(GetDlgItem(IDC_RESET_BUTTON),SW_SHOW);
::EnableWindow(GetDlgItem(IDC_VALUE_ALIGN),false);
} else {
int ItemCount = m_SearchResults.GetItemCount();
for (int i = ItemCount - 1; i >= 0; i--)
{
int ItemId = m_SearchResults.GetItemData(i);
SearchResultItem & Result = m_SearchResult[ItemId];
DWORD NewValue;
m_MMU->LoadPhysical32(Result.PAddr,NewValue,Size,false);
if (Value == NewValue)
{
char LocationStr[20];
sprintf(LocationStr,DisplayStr,NewValue);
m_SearchResults.SetItemText(i,2,LocationStr);
sprintf(LocationStr,DisplayStr,Result.Value);
m_SearchResults.SetItemText(i,3,LocationStr);
Result.Value = NewValue;
} else {
m_SearchResults.DeleteItem(i);
}
}
}
::SetWindowText(GetDlgItem(IDC_BORDER_RESULTS),stdstr_f("Results (%d)",m_SearchResults.GetItemCount()).c_str());
}
void CDebugMemorySearch::SearchForUnknown()
{
CMipsMemory::SearchMemChangeState Option = (CMipsMemory::SearchMemChangeState)m_UnknownOptions.GetItemData(m_UnknownOptions.GetCurSel());
if (Option == CMipsMemory::SearchChangeState_Reset)
{
m_SearchResults.DeleteAllItems();
m_MMU->SearchSetBaseForChanges();
FixUnknownOptions(false);
::ShowWindow(GetDlgItem(IDC_RESET_BUTTON),SW_SHOW);
::EnableWindow(GetDlgItem(IDC_UNKNOWN_ALIGN),true );
return;
}
MemorySize Size = (MemorySize)m_UnknownSize.GetItemData(m_UnknownSize.GetCurSel());
m_ValueSize.SetCurSel(m_UnknownSize.GetCurSel());
LPCTSTR DisplayStr = "0x%08X";
if (Size == _16Bit) { DisplayStr = "0x%04X"; }
else if (Size == _8Bit) { DisplayStr = "0x%04X"; }
if (!m_HaveResults)
{
m_HaveResults = true;
::EnableWindow(GetDlgItem(IDC_UNKNOWN_ALIGN),false );
DWORD StartAddress = m_PAddrStart.GetValue();
DWORD Len = m_SearchLen.GetValue();
DWORD MaxSearch = m_MaxSearch.GetValue();
DWORD MoveSize = (Size == _32Bit ? 4 : (Size == _16Bit ? 2 : 1));
for (int i = 2; i < 10; i++)
{
if (!m_SearchResults.DeleteColumn(i))
{
break;
}
}
m_SearchResults.AddColumn("New Value",2);
m_SearchResults.AddColumn("Old Value",3);
m_SearchResults.SetColumnWidth(0,50);
m_SearchResults.SetColumnWidth(1,75);
m_SearchResults.SetColumnWidth(2,75);
m_SearchResults.SetColumnWidth(3,75);
m_SearchResults.DeleteAllItems();
DWORD ItemsAdded = 0, OldValue, NewValue;
while (m_MMU->SearchForChanges(Option,Size,StartAddress,Len,OldValue,NewValue))
{
SearchResultItem Result;
Result.PAddr = StartAddress;
Result.Value = NewValue;
//if list size > max, then break
char LocationStr[20];
sprintf(LocationStr,"%d",ItemsAdded + 1);
int Index = m_SearchResults.AddItem(ItemsAdded,0,LocationStr);
m_SearchResults.SetItemData(Index,m_SearchResult.size());
m_SearchResult.push_back(Result);
sprintf(LocationStr,"0x%08X",StartAddress);
m_SearchResults.SetItemText(Index,1,LocationStr);
sprintf(LocationStr,DisplayStr,NewValue);
m_SearchResults.SetItemText(Index,2,LocationStr);
sprintf(LocationStr,DisplayStr,OldValue);
m_SearchResults.SetItemText(Index,3,LocationStr);
StartAddress += MoveSize;
Len -= MoveSize;
ItemsAdded += 1;
if (ItemsAdded >= MaxSearch)
{
break;
}
}
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),"Search Results");
::ShowWindow(GetDlgItem(IDC_RESET_BUTTON),SW_SHOW);
::EnableWindow(GetDlgItem(IDC_RADIO_TEXT),false);
::EnableWindow(GetDlgItem(IDC_RADIO_JAL),false);
} else {
int ItemCount = m_SearchResults.GetItemCount();
for (int i = ItemCount - 1; i >= 0; i--)
{
int ItemId = m_SearchResults.GetItemData(i);
SearchResultItem & Result = m_SearchResult[ItemId];
bool UpdateResult = false;
DWORD NewValue;
m_MMU->LoadPhysical32(Result.PAddr,NewValue,Size,false);
switch (Option)
{
case CMipsMemory::SearchChangeState_Changed:
if (Result.Value != NewValue)
{
UpdateResult = true;
}
break;
case CMipsMemory::SearchChangeState_Unchanged:
if (Result.Value == NewValue)
{
UpdateResult = true;
}
break;
case CMipsMemory::SearchChangeState_Greater:
if (NewValue > Result.Value)
{
UpdateResult = true;
}
break;
case CMipsMemory::SearchChangeState_Lessthan:
if (NewValue < Result.Value)
{
UpdateResult = true;
}
break;
default:
Notify().BreakPoint(__FILE__,__LINE__);
}
if (UpdateResult)
{
char LocationStr[20];
sprintf(LocationStr,DisplayStr,NewValue);
m_SearchResults.SetItemText(i,2,LocationStr);
sprintf(LocationStr,DisplayStr,Result.Value);
m_SearchResults.SetItemText(i,3,LocationStr);
Result.Value = NewValue;
} else {
m_SearchResults.DeleteItem(i);
}
}
}
::SetWindowText(GetDlgItem(IDC_BORDER_RESULTS),stdstr_f("Results (%d)",m_SearchResults.GetItemCount()).c_str());
}
void CDebugMemorySearch::SearchForText()
{
//
}
void CDebugMemorySearch::Reset ( void )
{
m_HaveResults = false;
SendMessage(GetDlgItem(IDC_RADIO_VALUE),BM_SETCHECK, BST_CHECKED,0);
EnableUnknownOptions(false);
EnableValueOptions(true);
EnableTextOptions(false);
EnableJalOptions(false);
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),"Search");
::SetWindowText(GetDlgItem(IDC_BORDER_RESULTS),"Results");
::ShowWindow(GetDlgItem(IDC_RESET_BUTTON),SW_HIDE);
::EnableWindow(GetDlgItem(IDC_RADIO_UNKNOWN),true);
::EnableWindow(GetDlgItem(IDC_RADIO_VALUE),true);
::EnableWindow(GetDlgItem(IDC_RADIO_TEXT),false);
::EnableWindow(GetDlgItem(IDC_RADIO_JAL),false);
for (int i = 1; i < 10; i++)
{
if (!m_SearchResults.DeleteColumn(i))
{
break;
}
}
m_SearchResults.AddColumn("Value",2);
m_SearchResults.DeleteAllItems();
m_SearchResults.SetColumnWidth(0,50);
m_SearchResults.SetColumnWidth(1,75);
m_SearchResults.SetColumnWidth(2,75);
m_SearchResults.SetExtendedListViewStyle(LVS_EX_FULLROWSELECT);
FixUnknownOptions(true);
}
void CDebugMemorySearch::FixUnknownOptions ( bool Reset )
{
CComboBox & cb = m_UnknownOptions ;
if (!Reset && cb.GetCount() > 1)
{
return;
}
cb.ResetContent();
if (Reset)
{
cb.SetItemData(cb.AddString("Create compare base"),CMipsMemory::SearchChangeState_Reset);
cb.SetCurSel(0);
return;
}
cb.SetItemData(cb.AddString("memory changed"),CMipsMemory::SearchChangeState_Changed);
cb.SetItemData(cb.AddString("memory unchanged"),CMipsMemory::SearchChangeState_Unchanged);
cb.SetItemData(cb.AddString("Value has increased"),CMipsMemory::SearchChangeState_Greater);
cb.SetItemData(cb.AddString("Value has descreased"),CMipsMemory::SearchChangeState_Lessthan);
cb.SetCurSel(1);
::SetWindowText(GetDlgItem(IDC_BTN_SEARCH),"Search");
}