2016-01-27 09:11:59 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* *
|
|
|
|
* Project64 - A Nintendo 64 emulator. *
|
|
|
|
* http://www.pj64-emu.com/ *
|
|
|
|
* Copyright (C) 2012 Project64. All rights reserved. *
|
|
|
|
* *
|
|
|
|
* License: *
|
|
|
|
* GNU/GPLv2 http://www.gnu.org/licenses/gpl-2.0.html *
|
|
|
|
* *
|
|
|
|
****************************************************************************/
|
|
|
|
#include "stdafx.h"
|
|
|
|
|
|
|
|
#include "DebuggerUI.h"
|
|
|
|
|
2017-08-18 05:08:22 +00:00
|
|
|
#include <Project64-core/N64System/Mips/OpCodeName.h>
|
|
|
|
|
2016-01-27 09:11:59 +00:00
|
|
|
CDumpMemory::CDumpMemory(CDebuggerUI * debugger) :
|
|
|
|
CDebugDialog<CDumpMemory>(debugger)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
CDumpMemory::~CDumpMemory()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
LRESULT CDumpMemory::OnInitDialog(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/)
|
|
|
|
{
|
|
|
|
m_StartAddress.Attach(GetDlgItem(IDC_E_START_ADDR));
|
|
|
|
m_EndAddress.Attach(GetDlgItem(IDC_E_END_ADDR));
|
|
|
|
m_PC.Attach(GetDlgItem(IDC_E_ALT_PC));
|
2017-08-18 05:08:22 +00:00
|
|
|
m_FormatList.Attach(GetDlgItem(IDC_FORMAT));
|
|
|
|
m_FileName.Attach(GetDlgItem(IDC_FILENAME));
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2017-11-21 20:37:45 +00:00
|
|
|
m_StartAddress.SetDisplayType(CEditNumber32::DisplayHex);
|
|
|
|
m_EndAddress.SetDisplayType(CEditNumber32::DisplayHex);
|
|
|
|
m_PC.SetDisplayType(CEditNumber32::DisplayHex);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2017-08-18 05:08:22 +00:00
|
|
|
uint32_t startAddress = 0x80000000;
|
|
|
|
uint32_t endAddress = startAddress + (g_MMU ? g_MMU->RdramSize() : 0x400000);
|
|
|
|
|
|
|
|
m_StartAddress.SetValue(startAddress, true, true);
|
|
|
|
m_EndAddress.SetValue(endAddress, true, true);
|
|
|
|
m_PC.SetValue(startAddress);
|
|
|
|
|
|
|
|
int nIndex = m_FormatList.AddString("TEXT - Disassembly + PC");
|
|
|
|
m_FormatList.SetItemData(nIndex, (DWORD_PTR)DisassemblyWithPC);
|
|
|
|
|
|
|
|
nIndex = m_FormatList.AddString("RAW - Big Endian (N64)");
|
|
|
|
m_FormatList.SetItemData(nIndex, (LPARAM)RawBigEndian);
|
|
|
|
|
|
|
|
m_FormatList.SetCurSel(0);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
|
|
|
WindowCreated();
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
LRESULT CDumpMemory::OnClicked(WORD /*wNotifyCode*/, WORD wID, HWND /*hWndCtl*/, BOOL& /*bHandled*/)
|
|
|
|
{
|
|
|
|
switch (wID)
|
|
|
|
{
|
|
|
|
case IDCANCEL:
|
|
|
|
EndDialog(0);
|
|
|
|
break;
|
|
|
|
case IDC_BTN_CHOOSE_FILE:
|
|
|
|
{
|
|
|
|
g_BaseSystem->ExternalEvent(SysEvent_PauseCPU_DumpMemory);
|
2017-01-20 23:31:09 +00:00
|
|
|
|
2017-08-18 05:08:22 +00:00
|
|
|
int CurrentFormatSel = m_FormatList.GetCurSel();
|
|
|
|
DumpFormat Format = (DumpFormat)m_FormatList.GetItemData(CurrentFormatSel);
|
|
|
|
|
2017-09-13 12:00:53 +00:00
|
|
|
const char* FileFilter = "All files (*.*)\0*.*\0";
|
2017-08-18 05:08:22 +00:00
|
|
|
|
|
|
|
if (Format == RawBigEndian)
|
|
|
|
{
|
|
|
|
FileFilter = "Binary file (*.bin)\0*.bin;\0All files (*.*)\0*.*\0";
|
|
|
|
}
|
|
|
|
else if (Format == DisassemblyWithPC)
|
|
|
|
{
|
|
|
|
FileFilter = "Text file (*.txt)\0*.txt;\0All files (*.*)\0*.*\0";
|
|
|
|
}
|
|
|
|
|
2017-01-20 23:31:09 +00:00
|
|
|
CPath FileName;
|
2017-08-18 05:08:22 +00:00
|
|
|
|
|
|
|
if (FileName.SelectFile(m_hWnd, CPath(CPath::MODULE_DIRECTORY), FileFilter, false))
|
2016-01-27 09:11:59 +00:00
|
|
|
{
|
2017-01-20 23:31:09 +00:00
|
|
|
if (FileName.GetExtension().length() == 0)
|
2016-01-27 09:11:59 +00:00
|
|
|
{
|
2017-08-18 05:08:22 +00:00
|
|
|
FileName.SetExtension(Format == RawBigEndian ? "bin" : "txt");
|
|
|
|
m_FileName.SetWindowTextA(FileName);
|
2016-01-27 09:11:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
g_BaseSystem->ExternalEvent(SysEvent_ResumeCPU_DumpMemory);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IDOK:
|
|
|
|
{
|
|
|
|
TCHAR FileName[MAX_PATH];
|
2017-08-18 05:08:22 +00:00
|
|
|
int CurrentFormatSel = m_FormatList.GetCurSel();
|
|
|
|
DumpFormat Format = (DumpFormat) m_FormatList.GetItemData(CurrentFormatSel);
|
|
|
|
|
2016-01-27 09:11:59 +00:00
|
|
|
DWORD StartPC = m_StartAddress.GetValue();
|
|
|
|
DWORD EndPC = m_EndAddress.GetValue();
|
|
|
|
DWORD DumpPC = m_PC.GetValue();
|
|
|
|
GetDlgItemText(IDC_FILENAME, FileName, sizeof(FileName));
|
|
|
|
if (strlen(FileName) == 0)
|
|
|
|
{
|
|
|
|
g_Notify->DisplayError("Please Choose target file");
|
|
|
|
::SetFocus(GetDlgItem(IDC_FILENAME));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (SendDlgItemMessage(IDC_USE_ALT_PC, BM_GETSTATE, 0, 0) != BST_CHECKED)
|
|
|
|
{
|
2017-08-18 05:08:22 +00:00
|
|
|
DumpPC = StartPC;
|
2016-01-27 09:11:59 +00:00
|
|
|
}
|
|
|
|
//disable buttons
|
|
|
|
::EnableWindow(GetDlgItem(IDC_E_START_ADDR), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDC_E_END_ADDR), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDC_E_ALT_PC), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDC_USE_ALT_PC), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDC_FILENAME), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDC_BTN_CHOOSE_FILE), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDC_FORMAT), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDOK), FALSE);
|
|
|
|
::EnableWindow(GetDlgItem(IDCANCEL), FALSE);
|
|
|
|
g_BaseSystem->ExternalEvent(SysEvent_PauseCPU_DumpMemory);
|
|
|
|
if (!DumpMemory(FileName, Format, StartPC, EndPC, DumpPC))
|
|
|
|
{
|
|
|
|
//enable buttons
|
|
|
|
g_BaseSystem->ExternalEvent(SysEvent_ResumeCPU_DumpMemory);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
g_BaseSystem->ExternalEvent(SysEvent_ResumeCPU_DumpMemory);
|
|
|
|
}
|
|
|
|
EndDialog(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-08-18 05:08:22 +00:00
|
|
|
bool CDumpMemory::DumpMemory(LPCSTR FileName, DumpFormat Format, DWORD StartPC, DWORD EndPC, DWORD DumpPC)
|
2016-01-27 09:11:59 +00:00
|
|
|
{
|
2017-08-18 05:08:22 +00:00
|
|
|
if (Format == DisassemblyWithPC)
|
|
|
|
{
|
|
|
|
CLog LogFile;
|
|
|
|
if (!LogFile.Open(FileName))
|
|
|
|
{
|
|
|
|
g_Notify->DisplayError(stdstr_f("Failed to open\n%s", FileName).c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
LogFile.SetFlush(false);
|
|
|
|
LogFile.SetTruncateFile(false);
|
|
|
|
|
|
|
|
for (uint32_t pc = StartPC; pc < EndPC; pc += 4, DumpPC += 4)
|
|
|
|
{
|
|
|
|
OPCODE opcode;
|
|
|
|
g_MMU->LW_VAddr(pc, opcode.Hex);
|
|
|
|
|
|
|
|
const char* command = R4300iOpcodeName(opcode.Hex, DumpPC);
|
|
|
|
|
|
|
|
char* cmdName = strtok((char*)command, "\t");
|
|
|
|
char* cmdArgs = strtok(NULL, "\t");
|
|
|
|
cmdArgs = cmdArgs ? cmdArgs : "";
|
|
|
|
|
|
|
|
LogFile.LogF("%X: %-15s%s\r\n", DumpPC, cmdName, cmdArgs);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_StartAddress.SetValue(StartPC, true, true);
|
|
|
|
m_EndAddress.SetValue(EndPC, true, true);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Format == RawBigEndian)
|
|
|
|
{
|
|
|
|
CFile dumpFile;
|
|
|
|
|
|
|
|
if (!dumpFile.Open(FileName, CFile::modeCreate | CFile::modeWrite))
|
|
|
|
{
|
|
|
|
g_Notify->DisplayError(stdstr_f("Failed to open\n%s", FileName).c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t dumpLen = EndPC - StartPC;
|
|
|
|
uint8_t* dumpBuf = (uint8_t*)malloc(dumpLen);
|
|
|
|
uint32_t dumpIdx = 0;
|
|
|
|
|
|
|
|
for (uint32_t pc = StartPC; pc < EndPC; pc++, dumpIdx++)
|
|
|
|
{
|
|
|
|
bool bReadable = g_MMU->LB_VAddr(pc, dumpBuf[dumpIdx]);
|
|
|
|
|
|
|
|
if (!bReadable)
|
|
|
|
{
|
|
|
|
g_Notify->DisplayError(stdstr_f("Address error\n%s", FileName).c_str());
|
|
|
|
dumpFile.Close();
|
|
|
|
free(dumpBuf);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dumpFile.SeekToBegin();
|
|
|
|
dumpFile.Write(dumpBuf, dumpLen);
|
|
|
|
dumpFile.Close();
|
|
|
|
free(dumpBuf);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-01-27 09:11:59 +00:00
|
|
|
return false;
|
2015-12-15 06:01:06 +00:00
|
|
|
}
|