mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 07:21:14 +01:00
Merge pull request #5287 from sepalani/rso
Generate Symbols From RSO Modules added
This commit is contained in:
commit
4517231a1a
@ -31,6 +31,7 @@ set(SRCS
|
|||||||
Debugger/Debugger_SymbolMap.cpp
|
Debugger/Debugger_SymbolMap.cpp
|
||||||
Debugger/Dump.cpp
|
Debugger/Dump.cpp
|
||||||
Debugger/PPCDebugInterface.cpp
|
Debugger/PPCDebugInterface.cpp
|
||||||
|
Debugger/RSO.cpp
|
||||||
DSP/DSPAssembler.cpp
|
DSP/DSPAssembler.cpp
|
||||||
DSP/DSPDisassembler.cpp
|
DSP/DSPDisassembler.cpp
|
||||||
DSP/DSPAccelerator.cpp
|
DSP/DSPAccelerator.cpp
|
||||||
|
@ -51,6 +51,7 @@
|
|||||||
<ClCompile Include="Debugger\Debugger_SymbolMap.cpp" />
|
<ClCompile Include="Debugger\Debugger_SymbolMap.cpp" />
|
||||||
<ClCompile Include="Debugger\Dump.cpp" />
|
<ClCompile Include="Debugger\Dump.cpp" />
|
||||||
<ClCompile Include="Debugger\PPCDebugInterface.cpp" />
|
<ClCompile Include="Debugger\PPCDebugInterface.cpp" />
|
||||||
|
<ClCompile Include="Debugger\RSO.cpp" />
|
||||||
<ClCompile Include="DSPEmulator.cpp" />
|
<ClCompile Include="DSPEmulator.cpp" />
|
||||||
<ClCompile Include="DSP\DSPAssembler.cpp" />
|
<ClCompile Include="DSP\DSPAssembler.cpp" />
|
||||||
<ClCompile Include="DSP\DSPDisassembler.cpp" />
|
<ClCompile Include="DSP\DSPDisassembler.cpp" />
|
||||||
@ -312,6 +313,7 @@
|
|||||||
<ClInclude Include="Debugger\Dump.h" />
|
<ClInclude Include="Debugger\Dump.h" />
|
||||||
<ClInclude Include="Debugger\GCELF.h" />
|
<ClInclude Include="Debugger\GCELF.h" />
|
||||||
<ClInclude Include="Debugger\PPCDebugInterface.h" />
|
<ClInclude Include="Debugger\PPCDebugInterface.h" />
|
||||||
|
<ClInclude Include="Debugger\RSO.h" />
|
||||||
<ClInclude Include="DSPEmulator.h" />
|
<ClInclude Include="DSPEmulator.h" />
|
||||||
<ClInclude Include="DSP\DSPAssembler.h" />
|
<ClInclude Include="DSP\DSPAssembler.h" />
|
||||||
<ClInclude Include="DSP\DSPDisassembler.h" />
|
<ClInclude Include="DSP\DSPDisassembler.h" />
|
||||||
|
@ -209,6 +209,9 @@
|
|||||||
<ClCompile Include="Debugger\PPCDebugInterface.cpp">
|
<ClCompile Include="Debugger\PPCDebugInterface.cpp">
|
||||||
<Filter>Debugger</Filter>
|
<Filter>Debugger</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="Debugger\RSO.cpp">
|
||||||
|
<Filter>Debugger</Filter>
|
||||||
|
</ClCompile>
|
||||||
<ClCompile Include="DSP\Interpreter\DSPIntArithmetic.cpp">
|
<ClCompile Include="DSP\Interpreter\DSPIntArithmetic.cpp">
|
||||||
<Filter>DSPCore\Interpreter</Filter>
|
<Filter>DSPCore\Interpreter</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
@ -928,6 +931,9 @@
|
|||||||
<ClInclude Include="Debugger\PPCDebugInterface.h">
|
<ClInclude Include="Debugger\PPCDebugInterface.h">
|
||||||
<Filter>Debugger</Filter>
|
<Filter>Debugger</Filter>
|
||||||
</ClInclude>
|
</ClInclude>
|
||||||
|
<ClInclude Include="Debugger\RSO.h">
|
||||||
|
<Filter>Debugger</Filter>
|
||||||
|
</ClInclude>
|
||||||
<ClInclude Include="DSP\Interpreter\DSPIntCCUtil.h">
|
<ClInclude Include="DSP\Interpreter\DSPIntCCUtil.h">
|
||||||
<Filter>DSPCore\Interpreter</Filter>
|
<Filter>DSPCore\Interpreter</Filter>
|
||||||
</ClInclude>
|
</ClInclude>
|
||||||
|
642
Source/Core/Core/Debugger/RSO.cpp
Normal file
642
Source/Core/Core/Debugger/RSO.cpp
Normal file
@ -0,0 +1,642 @@
|
|||||||
|
// Copyright 2017 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "Core/Debugger/RSO.h"
|
||||||
|
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#include "Common/CommonFuncs.h"
|
||||||
|
#include "Common/Logging/Log.h"
|
||||||
|
#include "Core/PowerPC/PPCSymbolDB.h"
|
||||||
|
#include "Core/PowerPC/PowerPC.h"
|
||||||
|
|
||||||
|
void RSOHeaderView::Load(u32 address)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
m_header.entry.next_entry = PowerPC::HostRead_U32(address);
|
||||||
|
m_header.entry.prev_entry = PowerPC::HostRead_U32(address + 0x04);
|
||||||
|
m_header.entry.section_count = PowerPC::HostRead_U32(address + 0x08);
|
||||||
|
m_header.entry.section_table_offset = PowerPC::HostRead_U32(address + 0xC);
|
||||||
|
m_header.entry.name_offset = PowerPC::HostRead_U32(address + 0x10);
|
||||||
|
m_header.entry.name_size = PowerPC::HostRead_U32(address + 0x14);
|
||||||
|
m_header.entry.version = PowerPC::HostRead_U32(address + 0x18);
|
||||||
|
m_header.entry.bss_size = PowerPC::HostRead_U32(address + 0x1C);
|
||||||
|
m_header.section_info.prolog_section_index = PowerPC::HostRead_U8(address + 0x20);
|
||||||
|
m_header.section_info.epilog_section_index = PowerPC::HostRead_U8(address + 0x21);
|
||||||
|
m_header.section_info.unresolved_section_index = PowerPC::HostRead_U8(address + 0x22);
|
||||||
|
m_header.section_info.bss_section_index = PowerPC::HostRead_U8(address + 0x23);
|
||||||
|
m_header.section_info.prolog_offset = PowerPC::HostRead_U32(address + 0x24);
|
||||||
|
m_header.section_info.epilog_offset = PowerPC::HostRead_U32(address + 0x28);
|
||||||
|
m_header.section_info.unresolved_offset = PowerPC::HostRead_U32(address + 0x2C);
|
||||||
|
m_header.relocation_tables.internals_offset = PowerPC::HostRead_U32(address + 0x30);
|
||||||
|
m_header.relocation_tables.internals_size = PowerPC::HostRead_U32(address + 0x34);
|
||||||
|
m_header.relocation_tables.externals_offset = PowerPC::HostRead_U32(address + 0x38);
|
||||||
|
m_header.relocation_tables.externals_size = PowerPC::HostRead_U32(address + 0x3C);
|
||||||
|
m_header.symbol_tables.exports_offset = PowerPC::HostRead_U32(address + 0x40);
|
||||||
|
m_header.symbol_tables.exports_size = PowerPC::HostRead_U32(address + 0x44);
|
||||||
|
m_header.symbol_tables.exports_name_table = PowerPC::HostRead_U32(address + 0x48);
|
||||||
|
m_header.symbol_tables.imports_offset = PowerPC::HostRead_U32(address + 0x4C);
|
||||||
|
m_header.symbol_tables.imports_size = PowerPC::HostRead_U32(address + 0x50);
|
||||||
|
m_header.symbol_tables.imports_name_table = PowerPC::HostRead_U32(address + 0x54);
|
||||||
|
|
||||||
|
// Prevent an invalid name going wild
|
||||||
|
if (m_header.entry.name_size < 0x100)
|
||||||
|
m_name = PowerPC::HostGetString(m_header.entry.name_offset, m_header.entry.name_size);
|
||||||
|
else
|
||||||
|
m_name = PowerPC::HostGetString(m_header.entry.name_offset, 0x100);
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetNextEntry() const
|
||||||
|
{
|
||||||
|
return m_header.entry.next_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetPrevEntry() const
|
||||||
|
{
|
||||||
|
return m_header.entry.prev_entry;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetSectionCount() const
|
||||||
|
{
|
||||||
|
return m_header.entry.section_count;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetSectionTableOffset() const
|
||||||
|
{
|
||||||
|
return m_header.entry.section_table_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& RSOHeaderView::GetName() const
|
||||||
|
{
|
||||||
|
return m_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetVersion() const
|
||||||
|
{
|
||||||
|
return m_header.entry.version;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetBssSectionSize() const
|
||||||
|
{
|
||||||
|
return m_header.entry.bss_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetPrologSectionIndex() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.prolog_section_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetEpilogSectionIndex() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.epilog_section_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetUnresolvedSectionIndex() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.unresolved_section_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetBssSectionIndex() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.bss_section_index;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetPrologSectionOffset() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.prolog_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetEpilogSectionOffset() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.epilog_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetUnresolvedSectionOffset() const
|
||||||
|
{
|
||||||
|
return m_header.section_info.unresolved_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetInternalsOffset() const
|
||||||
|
{
|
||||||
|
return m_header.relocation_tables.internals_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetInternalsSize() const
|
||||||
|
{
|
||||||
|
return m_header.relocation_tables.internals_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetExternalsOffset() const
|
||||||
|
{
|
||||||
|
return m_header.relocation_tables.externals_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetExternalsSize() const
|
||||||
|
{
|
||||||
|
return m_header.relocation_tables.externals_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetExportsOffset() const
|
||||||
|
{
|
||||||
|
return m_header.symbol_tables.exports_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetExportsSize() const
|
||||||
|
{
|
||||||
|
return m_header.symbol_tables.exports_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetExportsNameTable() const
|
||||||
|
{
|
||||||
|
return m_header.symbol_tables.exports_name_table;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetImportsOffset() const
|
||||||
|
{
|
||||||
|
return m_header.symbol_tables.imports_offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetImportsSize() const
|
||||||
|
{
|
||||||
|
return m_header.symbol_tables.imports_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOHeaderView::GetImportsNameTable() const
|
||||||
|
{
|
||||||
|
return m_header.symbol_tables.imports_name_table;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOSectionsView::Load(u32 address, std::size_t count)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
for (std::size_t i = 0; i < count; ++i)
|
||||||
|
{
|
||||||
|
RSOSection section;
|
||||||
|
section.offset = PowerPC::HostRead_U32(address);
|
||||||
|
section.size = PowerPC::HostRead_U32(address + 4);
|
||||||
|
m_sections.emplace_back(std::move(section));
|
||||||
|
address += sizeof(RSOSection);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t RSOSectionsView::Count() const
|
||||||
|
{
|
||||||
|
return m_sections.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOSection& RSOSectionsView::GetSection(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_sections.at(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOSection>& RSOSectionsView::GetSections() const
|
||||||
|
{
|
||||||
|
return m_sections;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOImportsView::Load(u32 address, std::size_t count)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
for (std::size_t i = 0; i < count; ++i)
|
||||||
|
{
|
||||||
|
RSOImport rso_import;
|
||||||
|
rso_import.name_offset = PowerPC::HostRead_U32(address);
|
||||||
|
rso_import.code_offset = PowerPC::HostRead_U32(address + 4);
|
||||||
|
rso_import.entry_offset = PowerPC::HostRead_U32(address + 8);
|
||||||
|
m_imports.emplace_back(std::move(rso_import));
|
||||||
|
address += sizeof(RSOImport);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOImportsView::Count() const
|
||||||
|
{
|
||||||
|
return m_imports.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOImport& RSOImportsView::GetImport(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_imports.at(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOImport>& RSOImportsView::GetImports() const
|
||||||
|
{
|
||||||
|
return m_imports;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOExportsView::Load(u32 address, std::size_t count)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
for (std::size_t i = 0; i < count; ++i)
|
||||||
|
{
|
||||||
|
RSOExport rso_export;
|
||||||
|
rso_export.name_offset = PowerPC::HostRead_U32(address);
|
||||||
|
rso_export.code_offset = PowerPC::HostRead_U32(address + 4);
|
||||||
|
rso_export.section_index = PowerPC::HostRead_U32(address + 8);
|
||||||
|
rso_export.hash = PowerPC::HostRead_U32(address + 12);
|
||||||
|
m_exports.emplace_back(std::move(rso_export));
|
||||||
|
address += sizeof(RSOExport);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOExportsView::Count() const
|
||||||
|
{
|
||||||
|
return m_exports.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOExport& RSOExportsView::GetExport(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_exports.at(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOExport>& RSOExportsView::GetExports() const
|
||||||
|
{
|
||||||
|
return m_exports;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOInternalsView::Load(u32 address, std::size_t count)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
for (std::size_t i = 0; i < count; ++i)
|
||||||
|
{
|
||||||
|
RSOInternalsEntry entry;
|
||||||
|
entry.r_offset = PowerPC::HostRead_U32(address);
|
||||||
|
entry.r_info = PowerPC::HostRead_U32(address + 4);
|
||||||
|
entry.r_addend = PowerPC::HostRead_U32(address + 8);
|
||||||
|
m_entries.emplace_back(std::move(entry));
|
||||||
|
address += sizeof(RSOInternalsEntry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOInternalsView::Count() const
|
||||||
|
{
|
||||||
|
return m_entries.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOInternalsEntry& RSOInternalsView::GetEntry(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_entries.at(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOInternalsEntry>& RSOInternalsView::GetEntries() const
|
||||||
|
{
|
||||||
|
return m_entries;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOExternalsView::Load(u32 address, std::size_t count)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
for (std::size_t i = 0; i < count; ++i)
|
||||||
|
{
|
||||||
|
RSOExternalsEntry entry;
|
||||||
|
entry.r_offset = PowerPC::HostRead_U32(address);
|
||||||
|
entry.r_info = PowerPC::HostRead_U32(address + 4);
|
||||||
|
entry.r_addend = PowerPC::HostRead_U32(address + 8);
|
||||||
|
m_entries.emplace_back(std::move(entry));
|
||||||
|
address += sizeof(RSOExternalsEntry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOExternalsView::Count() const
|
||||||
|
{
|
||||||
|
return m_entries.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOExternalsEntry& RSOExternalsView::GetEntry(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_entries.at(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOExternalsEntry>& RSOExternalsView::GetEntries() const
|
||||||
|
{
|
||||||
|
return m_entries;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadHeader(u32 address)
|
||||||
|
{
|
||||||
|
m_address = address;
|
||||||
|
m_header.Load(address);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadSections()
|
||||||
|
{
|
||||||
|
m_sections.Load(m_header.GetSectionTableOffset(), m_header.GetSectionCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadImports()
|
||||||
|
{
|
||||||
|
std::size_t size = m_header.GetImportsSize();
|
||||||
|
if (size % sizeof(RSOImport) != 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO Imports Table has an incoherent size (%08zx)", size);
|
||||||
|
m_imports.Load(m_header.GetImportsOffset(), size / sizeof(RSOImport));
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadExports()
|
||||||
|
{
|
||||||
|
std::size_t size = m_header.GetExportsSize();
|
||||||
|
if (size % sizeof(RSOExport) != 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO Exports Table has an incoherent size (%08zx)", size);
|
||||||
|
m_exports.Load(m_header.GetExportsOffset(), size / sizeof(RSOExport));
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadInternals()
|
||||||
|
{
|
||||||
|
std::size_t size = m_header.GetInternalsSize();
|
||||||
|
if (size % sizeof(RSOInternalsEntry) != 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO Internals Relocation Table has an incoherent size (%08zx)", size);
|
||||||
|
m_imports.Load(m_header.GetInternalsOffset(), size / sizeof(RSOInternalsEntry));
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadExternals()
|
||||||
|
{
|
||||||
|
std::size_t size = m_header.GetExternalsSize();
|
||||||
|
if (size % sizeof(RSOExternalsEntry) != 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO Externals Relocation Table has an incoherent size (%08zx)", size);
|
||||||
|
m_imports.Load(m_header.GetExternalsOffset(), size / sizeof(RSOExternalsEntry));
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::LoadAll(u32 address)
|
||||||
|
{
|
||||||
|
LoadHeader(address);
|
||||||
|
LoadSections();
|
||||||
|
LoadImports();
|
||||||
|
LoadExports();
|
||||||
|
LoadInternals();
|
||||||
|
LoadExternals();
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOView::Apply(PPCSymbolDB* symbol_db) const
|
||||||
|
{
|
||||||
|
std::size_t count = 0;
|
||||||
|
for (const RSOExport& rso_export : GetExports())
|
||||||
|
{
|
||||||
|
u32 address = GetExportAddress(rso_export);
|
||||||
|
if (address != 0)
|
||||||
|
{
|
||||||
|
Symbol* symbol = symbol_db->AddFunction(address);
|
||||||
|
if (!symbol)
|
||||||
|
symbol = symbol_db->GetSymbolFromAddr(address);
|
||||||
|
if (symbol)
|
||||||
|
{
|
||||||
|
symbol->name = GetExportName(rso_export);
|
||||||
|
count += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DEBUG_LOG(OSHLE, "RSO(%s): %zu/%zu symbols applied", GetName().c_str(), count, GetExportsCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetNextEntry() const
|
||||||
|
{
|
||||||
|
return m_header.GetNextEntry();
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetPrevEntry() const
|
||||||
|
{
|
||||||
|
return m_header.GetPrevEntry();
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetVersion() const
|
||||||
|
{
|
||||||
|
return m_header.GetVersion();
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetAddress() const
|
||||||
|
{
|
||||||
|
return m_address;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOView::GetSectionCount() const
|
||||||
|
{
|
||||||
|
return m_sections.Count();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOSection& RSOView::GetSection(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_sections.GetSection(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOSection>& RSOView::GetSections() const
|
||||||
|
{
|
||||||
|
return m_sections.GetSections();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOView::GetImportsCount() const
|
||||||
|
{
|
||||||
|
return m_imports.Count();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOImport& RSOView::GetImport(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_imports.GetImport(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string RSOView::GetImportName(const RSOImport& rso_import) const
|
||||||
|
{
|
||||||
|
return PowerPC::HostGetString(m_header.GetImportsNameTable() + rso_import.name_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOImport>& RSOView::GetImports() const
|
||||||
|
{
|
||||||
|
return m_imports.GetImports();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOView::GetExportsCount() const
|
||||||
|
{
|
||||||
|
return m_exports.Count();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOExport& RSOView::GetExport(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_exports.GetExport(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string RSOView::GetExportName(const RSOExport& rso_export) const
|
||||||
|
{
|
||||||
|
return PowerPC::HostGetString(m_header.GetExportsNameTable() + rso_export.name_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetExportAddress(const RSOExport& rso_export) const
|
||||||
|
{
|
||||||
|
u32 address = 0;
|
||||||
|
|
||||||
|
if (rso_export.section_index < GetSectionCount())
|
||||||
|
address = GetSection(rso_export.section_index).offset + rso_export.code_offset;
|
||||||
|
|
||||||
|
return address;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOExport>& RSOView::GetExports() const
|
||||||
|
{
|
||||||
|
return m_exports.GetExports();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOView::GetInternalsCount() const
|
||||||
|
{
|
||||||
|
return m_internals.Count();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOInternalsEntry& RSOView::GetInternalsEntry(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_internals.GetEntry(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOInternalsEntry>& RSOView::GetInternals() const
|
||||||
|
{
|
||||||
|
return m_internals.GetEntries();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::size_t RSOView::GetExternalsCount() const
|
||||||
|
{
|
||||||
|
return m_externals.Count();
|
||||||
|
}
|
||||||
|
|
||||||
|
const RSOExternalsEntry& RSOView::GetExternalsEntry(std::size_t index) const
|
||||||
|
{
|
||||||
|
return m_externals.GetEntry(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::vector<RSOExternalsEntry>& RSOView::GetExternals() const
|
||||||
|
{
|
||||||
|
return m_externals.GetEntries();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& RSOView::GetName() const
|
||||||
|
{
|
||||||
|
return m_header.GetName();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string RSOView::GetName(const RSOImport& rso_import) const
|
||||||
|
{
|
||||||
|
return GetImportName(rso_import);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string RSOView::GetName(const RSOExport& rso_export) const
|
||||||
|
{
|
||||||
|
return GetExportName(rso_export);
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetProlog() const
|
||||||
|
{
|
||||||
|
u32 section_index = m_header.GetPrologSectionIndex();
|
||||||
|
if (section_index == 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO doesn't have a prolog function");
|
||||||
|
else if (section_index >= m_sections.Count())
|
||||||
|
WARN_LOG(OSHLE, "RSO prolog section index out of bound");
|
||||||
|
else
|
||||||
|
return GetSection(section_index).offset + m_header.GetPrologSectionOffset();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetEpilog() const
|
||||||
|
{
|
||||||
|
u32 section_index = m_header.GetEpilogSectionIndex();
|
||||||
|
if (section_index == 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO doesn't have an epilog function");
|
||||||
|
else if (section_index >= m_sections.Count())
|
||||||
|
WARN_LOG(OSHLE, "RSO epilog section index out of bound");
|
||||||
|
else
|
||||||
|
return GetSection(section_index).offset + m_header.GetEpilogSectionOffset();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
u32 RSOView::GetUnresolved() const
|
||||||
|
{
|
||||||
|
u32 section_index = m_header.GetUnresolvedSectionIndex();
|
||||||
|
if (section_index == 0)
|
||||||
|
WARN_LOG(OSHLE, "RSO doesn't have a unresolved function");
|
||||||
|
else if (section_index >= m_sections.Count())
|
||||||
|
WARN_LOG(OSHLE, "RSO unresolved section index out of bound");
|
||||||
|
else
|
||||||
|
return GetSection(section_index).offset + m_header.GetUnresolvedSectionOffset();
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RSOChainView::Load(u32 address)
|
||||||
|
{
|
||||||
|
// Load node
|
||||||
|
RSOView node;
|
||||||
|
node.LoadHeader(address);
|
||||||
|
DEBUG_LOG(OSHLE, "RSOChain node name: %s", node.GetName().c_str());
|
||||||
|
m_chain.emplace_front(std::move(node));
|
||||||
|
|
||||||
|
if (LoadNextChain(m_chain.front()) && LoadPrevChain(m_chain.front()))
|
||||||
|
{
|
||||||
|
for (RSOView& view : m_chain)
|
||||||
|
{
|
||||||
|
view.LoadSections();
|
||||||
|
view.LoadExports();
|
||||||
|
view.LoadImports();
|
||||||
|
view.LoadExternals();
|
||||||
|
view.LoadInternals();
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOChainView::Apply(PPCSymbolDB* symbol_db) const
|
||||||
|
{
|
||||||
|
for (const RSOView& rso_view : m_chain)
|
||||||
|
rso_view.Apply(symbol_db);
|
||||||
|
}
|
||||||
|
|
||||||
|
void RSOChainView::Clear()
|
||||||
|
{
|
||||||
|
m_chain.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::list<RSOView>& RSOChainView::GetChain() const
|
||||||
|
{
|
||||||
|
return m_chain;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RSOChainView::LoadNextChain(const RSOView& view)
|
||||||
|
{
|
||||||
|
u32 prev_address = view.GetAddress();
|
||||||
|
u32 next_address = view.GetNextEntry();
|
||||||
|
|
||||||
|
while (next_address != 0)
|
||||||
|
{
|
||||||
|
RSOView next_node;
|
||||||
|
next_node.LoadHeader(next_address);
|
||||||
|
|
||||||
|
if (prev_address != next_node.GetPrevEntry())
|
||||||
|
{
|
||||||
|
ERROR_LOG(OSHLE, "RSOChain has an incoherent previous link %08x != %08x in %s", prev_address,
|
||||||
|
next_node.GetPrevEntry(), next_node.GetName().c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
prev_address = next_address;
|
||||||
|
next_address = next_node.GetNextEntry();
|
||||||
|
m_chain.emplace_back(std::move(next_node));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RSOChainView::LoadPrevChain(const RSOView& view)
|
||||||
|
{
|
||||||
|
u32 prev_address = view.GetPrevEntry();
|
||||||
|
u32 next_address = view.GetAddress();
|
||||||
|
|
||||||
|
while (prev_address != 0)
|
||||||
|
{
|
||||||
|
RSOView prev_node;
|
||||||
|
prev_node.LoadHeader(prev_address);
|
||||||
|
|
||||||
|
if (next_address != prev_node.GetNextEntry())
|
||||||
|
{
|
||||||
|
ERROR_LOG(OSHLE, "RSOChain has an incoherent next link %08x != %08x in %s", next_address,
|
||||||
|
prev_node.GetNextEntry(), prev_node.GetName().c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
next_address = prev_address;
|
||||||
|
prev_address = prev_node.GetPrevEntry();
|
||||||
|
m_chain.emplace_front(std::move(prev_node));
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
282
Source/Core/Core/Debugger/RSO.h
Normal file
282
Source/Core/Core/Debugger/RSO.h
Normal file
@ -0,0 +1,282 @@
|
|||||||
|
// Copyright 2017 Dolphin Emulator Project
|
||||||
|
// Licensed under GPLv2+
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <cstddef>
|
||||||
|
#include <list>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "Common/CommonTypes.h"
|
||||||
|
|
||||||
|
class PPCSymbolDB;
|
||||||
|
|
||||||
|
struct RSOEntry
|
||||||
|
{
|
||||||
|
u32 next_entry;
|
||||||
|
u32 prev_entry;
|
||||||
|
u32 section_count;
|
||||||
|
u32 section_table_offset;
|
||||||
|
u32 name_offset;
|
||||||
|
u32 name_size;
|
||||||
|
u32 version;
|
||||||
|
u32 bss_size;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSOSectionInfo
|
||||||
|
{
|
||||||
|
u8 prolog_section_index;
|
||||||
|
u8 epilog_section_index;
|
||||||
|
u8 unresolved_section_index;
|
||||||
|
u8 bss_section_index;
|
||||||
|
u32 prolog_offset;
|
||||||
|
u32 epilog_offset;
|
||||||
|
u32 unresolved_offset;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSORelocationTables
|
||||||
|
{
|
||||||
|
u32 internals_offset;
|
||||||
|
u32 internals_size;
|
||||||
|
u32 externals_offset;
|
||||||
|
u32 externals_size;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSOSymbolTables
|
||||||
|
{
|
||||||
|
u32 exports_offset;
|
||||||
|
u32 exports_size;
|
||||||
|
u32 exports_name_table;
|
||||||
|
u32 imports_offset;
|
||||||
|
u32 imports_size;
|
||||||
|
u32 imports_name_table;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSOHeader
|
||||||
|
{
|
||||||
|
RSOEntry entry;
|
||||||
|
RSOSectionInfo section_info;
|
||||||
|
RSORelocationTables relocation_tables;
|
||||||
|
RSOSymbolTables symbol_tables;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSOSection
|
||||||
|
{
|
||||||
|
u32 offset;
|
||||||
|
u32 size;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSOImport
|
||||||
|
{
|
||||||
|
u32 name_offset;
|
||||||
|
u32 code_offset;
|
||||||
|
u32 entry_offset;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct RSOExport
|
||||||
|
{
|
||||||
|
u32 name_offset;
|
||||||
|
u32 code_offset;
|
||||||
|
u32 section_index;
|
||||||
|
u32 hash;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class RSORelocationTableType
|
||||||
|
{
|
||||||
|
Internals,
|
||||||
|
Externals
|
||||||
|
};
|
||||||
|
|
||||||
|
template <RSORelocationTableType Disambiguate>
|
||||||
|
struct RSORelocationTableEntry
|
||||||
|
{
|
||||||
|
u32 r_offset;
|
||||||
|
u32 r_info;
|
||||||
|
u32 r_addend;
|
||||||
|
};
|
||||||
|
|
||||||
|
using RSOInternalsEntry = RSORelocationTableEntry<RSORelocationTableType::Internals>;
|
||||||
|
using RSOExternalsEntry = RSORelocationTableEntry<RSORelocationTableType::Externals>;
|
||||||
|
|
||||||
|
class RSOHeaderView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Load(u32 address);
|
||||||
|
|
||||||
|
u32 GetNextEntry() const;
|
||||||
|
u32 GetPrevEntry() const;
|
||||||
|
u32 GetSectionCount() const;
|
||||||
|
u32 GetSectionTableOffset() const;
|
||||||
|
const std::string& GetName() const;
|
||||||
|
u32 GetVersion() const;
|
||||||
|
u32 GetBssSectionSize() const;
|
||||||
|
u32 GetPrologSectionIndex() const;
|
||||||
|
u32 GetEpilogSectionIndex() const;
|
||||||
|
u32 GetUnresolvedSectionIndex() const;
|
||||||
|
u32 GetBssSectionIndex() const;
|
||||||
|
u32 GetPrologSectionOffset() const;
|
||||||
|
u32 GetEpilogSectionOffset() const;
|
||||||
|
u32 GetUnresolvedSectionOffset() const;
|
||||||
|
u32 GetInternalsOffset() const;
|
||||||
|
u32 GetInternalsSize() const;
|
||||||
|
u32 GetExternalsOffset() const;
|
||||||
|
u32 GetExternalsSize() const;
|
||||||
|
u32 GetExportsOffset() const;
|
||||||
|
u32 GetExportsSize() const;
|
||||||
|
u32 GetExportsNameTable() const;
|
||||||
|
u32 GetImportsOffset() const;
|
||||||
|
u32 GetImportsSize() const;
|
||||||
|
u32 GetImportsNameTable() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
RSOHeader m_header;
|
||||||
|
std::string m_name;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOSectionsView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Load(u32 address, std::size_t count = 1);
|
||||||
|
std::size_t Count() const;
|
||||||
|
|
||||||
|
const RSOSection& GetSection(std::size_t index) const;
|
||||||
|
const std::vector<RSOSection>& GetSections() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<RSOSection> m_sections;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOImportsView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Load(u32 address, std::size_t count = 1);
|
||||||
|
std::size_t Count() const;
|
||||||
|
|
||||||
|
const RSOImport& GetImport(std::size_t index) const;
|
||||||
|
const std::vector<RSOImport>& GetImports() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<RSOImport> m_imports;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOExportsView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Load(u32 address, std::size_t count = 1);
|
||||||
|
std::size_t Count() const;
|
||||||
|
|
||||||
|
const RSOExport& GetExport(std::size_t index) const;
|
||||||
|
const std::vector<RSOExport>& GetExports() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<RSOExport> m_exports;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOInternalsView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Load(u32 address, std::size_t count = 1);
|
||||||
|
std::size_t Count() const;
|
||||||
|
|
||||||
|
const RSOInternalsEntry& GetEntry(std::size_t index) const;
|
||||||
|
const std::vector<RSOInternalsEntry>& GetEntries() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<RSOInternalsEntry> m_entries;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOExternalsView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Load(u32 address, std::size_t count = 1);
|
||||||
|
std::size_t Count() const;
|
||||||
|
|
||||||
|
const RSOExternalsEntry& GetEntry(std::size_t index) const;
|
||||||
|
const std::vector<RSOExternalsEntry>& GetEntries() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<RSOExternalsEntry> m_entries;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void LoadHeader(u32 address);
|
||||||
|
void LoadSections();
|
||||||
|
void LoadImports();
|
||||||
|
void LoadExports();
|
||||||
|
void LoadInternals();
|
||||||
|
void LoadExternals();
|
||||||
|
void LoadAll(u32 address);
|
||||||
|
|
||||||
|
void Apply(PPCSymbolDB* symbol_db) const;
|
||||||
|
|
||||||
|
u32 GetNextEntry() const;
|
||||||
|
u32 GetPrevEntry() const;
|
||||||
|
u32 GetVersion() const;
|
||||||
|
u32 GetAddress() const;
|
||||||
|
|
||||||
|
std::size_t GetSectionCount() const;
|
||||||
|
const RSOSection& GetSection(std::size_t index) const;
|
||||||
|
const std::vector<RSOSection>& GetSections() const;
|
||||||
|
|
||||||
|
std::size_t GetImportsCount() const;
|
||||||
|
const RSOImport& GetImport(std::size_t index) const;
|
||||||
|
std::string GetImportName(const RSOImport& rso_import) const;
|
||||||
|
const std::vector<RSOImport>& GetImports() const;
|
||||||
|
|
||||||
|
std::size_t GetExportsCount() const;
|
||||||
|
const RSOExport& GetExport(std::size_t index) const;
|
||||||
|
std::string GetExportName(const RSOExport& rso_export) const;
|
||||||
|
u32 GetExportAddress(const RSOExport& rso_export) const;
|
||||||
|
const std::vector<RSOExport>& GetExports() const;
|
||||||
|
|
||||||
|
std::size_t GetInternalsCount() const;
|
||||||
|
const RSOInternalsEntry& GetInternalsEntry(std::size_t index) const;
|
||||||
|
const std::vector<RSOInternalsEntry>& GetInternals() const;
|
||||||
|
|
||||||
|
std::size_t GetExternalsCount() const;
|
||||||
|
const RSOExternalsEntry& GetExternalsEntry(std::size_t index) const;
|
||||||
|
const std::vector<RSOExternalsEntry>& GetExternals() const;
|
||||||
|
|
||||||
|
const std::string& GetName() const;
|
||||||
|
std::string GetName(const RSOImport& rso_import) const;
|
||||||
|
std::string GetName(const RSOExport& rso_export) const;
|
||||||
|
|
||||||
|
u32 GetProlog() const;
|
||||||
|
u32 GetEpilog() const;
|
||||||
|
u32 GetUnresolved() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
RSOHeaderView m_header;
|
||||||
|
RSOSectionsView m_sections;
|
||||||
|
RSOImportsView m_imports;
|
||||||
|
RSOExportsView m_exports;
|
||||||
|
RSOInternalsView m_internals;
|
||||||
|
RSOExternalsView m_externals;
|
||||||
|
u32 m_address = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RSOChainView
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
bool Load(u32 address);
|
||||||
|
void Apply(PPCSymbolDB* symbol_db) const;
|
||||||
|
void Clear();
|
||||||
|
const std::list<RSOView>& GetChain() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool LoadNextChain(const RSOView& view);
|
||||||
|
bool LoadPrevChain(const RSOView& view);
|
||||||
|
|
||||||
|
std::list<RSOView> m_chain;
|
||||||
|
};
|
@ -3,8 +3,10 @@
|
|||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
|
#include <cstdlib>
|
||||||
#include <fstream>
|
#include <fstream>
|
||||||
#include <istream>
|
#include <istream>
|
||||||
|
#include <limits>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
@ -29,6 +31,7 @@
|
|||||||
#include "Core/Boot/Boot.h"
|
#include "Core/Boot/Boot.h"
|
||||||
#include "Core/ConfigManager.h"
|
#include "Core/ConfigManager.h"
|
||||||
#include "Core/Core.h"
|
#include "Core/Core.h"
|
||||||
|
#include "Core/Debugger/RSO.h"
|
||||||
#include "Core/HLE/HLE.h"
|
#include "Core/HLE/HLE.h"
|
||||||
#include "Core/Host.h"
|
#include "Core/Host.h"
|
||||||
#include "Core/PowerPC/JitInterface.h"
|
#include "Core/PowerPC/JitInterface.h"
|
||||||
@ -202,6 +205,35 @@ void CCodeWindow::OnSymbolsMenu(wxCommandEvent& event)
|
|||||||
NotifyMapLoaded();
|
NotifyMapLoaded();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case IDM_SCAN_RSO:
|
||||||
|
{
|
||||||
|
wxTextEntryDialog dialog(this, _("Enter the RSO module address:"));
|
||||||
|
if (dialog.ShowModal() == wxID_OK)
|
||||||
|
{
|
||||||
|
unsigned long address;
|
||||||
|
if (dialog.GetValue().ToULong(&address, 0) && address <= std::numeric_limits<u32>::max())
|
||||||
|
{
|
||||||
|
RSOChainView rso_chain;
|
||||||
|
if (rso_chain.Load(static_cast<u32>(address)))
|
||||||
|
{
|
||||||
|
rso_chain.Apply(&g_symbolDB);
|
||||||
|
// Update GUI
|
||||||
|
NotifyMapLoaded();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Parent->StatusBarMessage("Failed to load RSO module at %s",
|
||||||
|
dialog.GetValue().ToStdString().c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Parent->StatusBarMessage("Invalid RSO module address: %s",
|
||||||
|
dialog.GetValue().ToStdString().c_str());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
case IDM_LOAD_MAP_FILE:
|
case IDM_LOAD_MAP_FILE:
|
||||||
if (!map_exists)
|
if (!map_exists)
|
||||||
{
|
{
|
||||||
|
@ -232,6 +232,7 @@ void CFrame::BindDebuggerMenuBarUpdateEvents()
|
|||||||
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_CLEAR_SYMBOLS);
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_CLEAR_SYMBOLS);
|
||||||
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SCAN_FUNCTIONS);
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SCAN_FUNCTIONS);
|
||||||
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SCAN_SIGNATURES);
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SCAN_SIGNATURES);
|
||||||
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SCAN_RSO);
|
||||||
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_LOAD_MAP_FILE);
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_LOAD_MAP_FILE);
|
||||||
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SAVEMAPFILE);
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_SAVEMAPFILE);
|
||||||
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_LOAD_MAP_FILE_AS);
|
Bind(wxEVT_UPDATE_UI, &WxEventUtils::OnEnableIfCoreInitialized, IDM_LOAD_MAP_FILE_AS);
|
||||||
|
@ -217,6 +217,7 @@ enum
|
|||||||
IDM_CLEAR_SYMBOLS,
|
IDM_CLEAR_SYMBOLS,
|
||||||
IDM_SCAN_FUNCTIONS,
|
IDM_SCAN_FUNCTIONS,
|
||||||
IDM_SCAN_SIGNATURES,
|
IDM_SCAN_SIGNATURES,
|
||||||
|
IDM_SCAN_RSO,
|
||||||
IDM_LOAD_MAP_FILE,
|
IDM_LOAD_MAP_FILE,
|
||||||
IDM_LOAD_MAP_FILE_AS,
|
IDM_LOAD_MAP_FILE_AS,
|
||||||
IDM_LOAD_BAD_MAP_FILE,
|
IDM_LOAD_BAD_MAP_FILE,
|
||||||
|
@ -423,6 +423,8 @@ wxMenu* MainMenuBar::CreateSymbolsMenu() const
|
|||||||
IDM_SCAN_SIGNATURES, _("&Signature Database"),
|
IDM_SCAN_SIGNATURES, _("&Signature Database"),
|
||||||
_("Recognise standard functions from Sys/totaldb.dsy, and use generic zz_ "
|
_("Recognise standard functions from Sys/totaldb.dsy, and use generic zz_ "
|
||||||
"names for other functions."));
|
"names for other functions."));
|
||||||
|
generate_symbols_menu->Append(IDM_SCAN_RSO, _("&RSO Modules"),
|
||||||
|
_("Find functions based on RSO modules (experimental)..."));
|
||||||
symbols_menu->AppendSubMenu(generate_symbols_menu, _("&Generate Symbols From"));
|
symbols_menu->AppendSubMenu(generate_symbols_menu, _("&Generate Symbols From"));
|
||||||
symbols_menu->AppendSeparator();
|
symbols_menu->AppendSeparator();
|
||||||
symbols_menu->Append(IDM_LOAD_MAP_FILE, _("&Load Symbol Map"),
|
symbols_menu->Append(IDM_LOAD_MAP_FILE, _("&Load Symbol Map"),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user