mirror of
https://github.com/RPCSX/rpcsx.git
synced 2025-12-06 07:12:14 +01:00
move module initialization into a module manager, still has some issues like stopping not working and debug crashing add #idef 0 to modules that aren't in the windows project don't double initialize and don't de-initialize for now, since many modules don't expect it and it leads to many errors remove duplicate module lists for empty modules and implemented ones, make Module non-copyable but movable add secondary project, no real use for it now add some memleak config to the emucore and add asmjit path to rpcs3 small rebase error fixed to get it to compile again add filters for emucore re-add the module manager and static file WIP commit, linker errors abound some more abstraction layer stuff fix the remaining linker errors, re-enable platform specific mouse, pad and keyboard handlers rebasing fix memset undefined and re() usage of se_t before declaration Add wxGUI define by default for cmake builds fix copy constructors of Datetime header fix copy constructors of other wx interface classes remove static declarations of global variables make wxGLCanvas constructor non-ambiguous even with wx2.8. compat mode, fix wrong std::exception constructor calls remove duplicate definition for FromUTF8 and ToUTF8 temp changes
476 lines
12 KiB
C++
476 lines
12 KiB
C++
#include "stdafx.h"
|
|
#include "Emu/ConLog.h"
|
|
#include "Emu/Memory/Memory.h"
|
|
#include "Emu/System.h"
|
|
#include "Emu/Cell/PPUThread.h"
|
|
#include "Emu/SysCalls/SC_FUNC.h"
|
|
#include "Emu/SysCalls/Modules.h"
|
|
#include "ELF64.h"
|
|
#include "Emu/Cell/PPUInstrTable.h"
|
|
#include "Emu/System.h"
|
|
#include "Emu/SysCalls/ModuleManager.h"
|
|
|
|
using namespace PPU_instr;
|
|
|
|
void WriteEhdr(rFile& f, Elf64_Ehdr& ehdr)
|
|
{
|
|
Write32(f, ehdr.e_magic);
|
|
Write8(f, ehdr.e_class);
|
|
Write8(f, ehdr.e_data);
|
|
Write8(f, ehdr.e_curver);
|
|
Write8(f, ehdr.e_os_abi);
|
|
Write64(f, ehdr.e_abi_ver);
|
|
Write16(f, ehdr.e_type);
|
|
Write16(f, ehdr.e_machine);
|
|
Write32(f, ehdr.e_version);
|
|
Write64(f, ehdr.e_entry);
|
|
Write64(f, ehdr.e_phoff);
|
|
Write64(f, ehdr.e_shoff);
|
|
Write32(f, ehdr.e_flags);
|
|
Write16(f, ehdr.e_ehsize);
|
|
Write16(f, ehdr.e_phentsize);
|
|
Write16(f, ehdr.e_phnum);
|
|
Write16(f, ehdr.e_shentsize);
|
|
Write16(f, ehdr.e_shnum);
|
|
Write16(f, ehdr.e_shstrndx);
|
|
}
|
|
|
|
void WritePhdr(rFile& f, Elf64_Phdr& phdr)
|
|
{
|
|
Write32(f, phdr.p_type);
|
|
Write32(f, phdr.p_flags);
|
|
Write64(f, phdr.p_offset);
|
|
Write64(f, phdr.p_vaddr);
|
|
Write64(f, phdr.p_paddr);
|
|
Write64(f, phdr.p_filesz);
|
|
Write64(f, phdr.p_memsz);
|
|
Write64(f, phdr.p_align);
|
|
}
|
|
|
|
void WriteShdr(rFile& f, Elf64_Shdr& shdr)
|
|
{
|
|
Write32(f, shdr.sh_name);
|
|
Write32(f, shdr.sh_type);
|
|
Write64(f, shdr.sh_flags);
|
|
Write64(f, shdr.sh_addr);
|
|
Write64(f, shdr.sh_offset);
|
|
Write64(f, shdr.sh_size);
|
|
Write32(f, shdr.sh_link);
|
|
Write32(f, shdr.sh_info);
|
|
Write64(f, shdr.sh_addralign);
|
|
Write64(f, shdr.sh_entsize);
|
|
}
|
|
|
|
ELF64Loader::ELF64Loader(vfsStream& f)
|
|
: elf64_f(f)
|
|
, LoaderBase()
|
|
{
|
|
int a = 0;
|
|
}
|
|
|
|
bool ELF64Loader::LoadInfo()
|
|
{
|
|
if(!elf64_f.IsOpened()) return false;
|
|
|
|
if(!LoadEhdrInfo()) return false;
|
|
if(!LoadPhdrInfo()) return false;
|
|
if(!LoadShdrInfo()) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::LoadData(u64 offset)
|
|
{
|
|
if(!elf64_f.IsOpened()) return false;
|
|
|
|
if(!LoadEhdrData(offset)) return false;
|
|
if(!LoadPhdrData(offset)) return false;
|
|
if(!LoadShdrData(offset)) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::Close()
|
|
{
|
|
return elf64_f.Close();
|
|
}
|
|
|
|
bool ELF64Loader::LoadEhdrInfo(s64 offset)
|
|
{
|
|
elf64_f.Seek(offset < 0 ? 0 : offset);
|
|
ehdr.Load(elf64_f);
|
|
|
|
if(!ehdr.CheckMagic()) return false;
|
|
|
|
if(ehdr.e_phentsize != sizeof(Elf64_Phdr))
|
|
{
|
|
ConLog.Error("elf64 error: e_phentsize[0x%x] != sizeof(Elf64_Phdr)[0x%x]", ehdr.e_phentsize, sizeof(Elf64_Phdr));
|
|
return false;
|
|
}
|
|
|
|
if(ehdr.e_shentsize != sizeof(Elf64_Shdr))
|
|
{
|
|
ConLog.Error("elf64 error: e_shentsize[0x%x] != sizeof(Elf64_Shdr)[0x%x]", ehdr.e_shentsize, sizeof(Elf64_Shdr));
|
|
return false;
|
|
}
|
|
|
|
switch(ehdr.e_machine)
|
|
{
|
|
case MACHINE_PPC64:
|
|
case MACHINE_SPU:
|
|
machine = (Elf_Machine)ehdr.e_machine;
|
|
break;
|
|
|
|
default:
|
|
machine = MACHINE_Unknown;
|
|
ConLog.Error("Unknown elf64 type: 0x%x", ehdr.e_machine);
|
|
return false;
|
|
}
|
|
|
|
entry = ehdr.GetEntry();
|
|
if(entry == 0)
|
|
{
|
|
ConLog.Error("elf64 error: entry is null!");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::LoadPhdrInfo(s64 offset)
|
|
{
|
|
phdr_arr.clear();
|
|
|
|
if(ehdr.e_phoff == 0 && ehdr.e_phnum)
|
|
{
|
|
ConLog.Error("LoadPhdr64 error: Program header offset is null!");
|
|
return false;
|
|
}
|
|
|
|
elf64_f.Seek(offset < 0 ? ehdr.e_phoff : offset);
|
|
|
|
for(u32 i=0; i<ehdr.e_phnum; ++i)
|
|
{
|
|
phdr_arr.emplace_back();
|
|
phdr_arr.back().Load(elf64_f);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::LoadShdrInfo(s64 offset)
|
|
{
|
|
shdr_arr.clear();
|
|
shdr_name_arr.clear();
|
|
if(ehdr.e_shoff == 0 && ehdr.e_shnum)
|
|
{
|
|
ConLog.Warning("LoadShdr64 error: Section header offset is null!");
|
|
return true;
|
|
}
|
|
|
|
elf64_f.Seek(offset < 0 ? ehdr.e_shoff : offset);
|
|
for(u32 i=0; i<ehdr.e_shnum; ++i)
|
|
{
|
|
shdr_arr.emplace_back();
|
|
shdr_arr.back().Load(elf64_f);
|
|
}
|
|
|
|
if(ehdr.e_shstrndx >= shdr_arr.size())
|
|
{
|
|
ConLog.Warning("LoadShdr64 error: shstrndx too big!");
|
|
return true;
|
|
}
|
|
|
|
for(u32 i=0; i<shdr_arr.size(); ++i)
|
|
{
|
|
elf64_f.Seek((offset < 0 ? shdr_arr[ehdr.e_shstrndx].sh_offset : shdr_arr[ehdr.e_shstrndx].sh_offset - ehdr.e_shoff + offset) + shdr_arr[i].sh_name);
|
|
std::string name;
|
|
while(!elf64_f.Eof())
|
|
{
|
|
char c;
|
|
elf64_f.Read(&c, 1);
|
|
if(c == 0) break;
|
|
name.push_back(c);
|
|
}
|
|
shdr_name_arr.push_back(name);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::LoadEhdrData(u64 offset)
|
|
{
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.SkipLn();
|
|
ehdr.Show();
|
|
ConLog.SkipLn();
|
|
#endif
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::LoadPhdrData(u64 offset)
|
|
{
|
|
for(u32 i=0; i<phdr_arr.size(); ++i)
|
|
{
|
|
phdr_arr[i].Show();
|
|
|
|
if(phdr_arr[i].p_vaddr < min_addr)
|
|
{
|
|
min_addr = phdr_arr[i].p_vaddr;
|
|
}
|
|
|
|
if(phdr_arr[i].p_vaddr + phdr_arr[i].p_memsz > max_addr)
|
|
{
|
|
max_addr = phdr_arr[i].p_vaddr + phdr_arr[i].p_memsz;
|
|
}
|
|
|
|
if(phdr_arr[i].p_vaddr != phdr_arr[i].p_paddr)
|
|
{
|
|
ConLog.Warning
|
|
(
|
|
"ElfProgram different load addrs: paddr=0x%8.8x, vaddr=0x%8.8x",
|
|
phdr_arr[i].p_paddr, phdr_arr[i].p_vaddr
|
|
);
|
|
}
|
|
|
|
if(!Memory.MainMem.IsInMyRange(offset + phdr_arr[i].p_vaddr, phdr_arr[i].p_memsz))
|
|
{
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.Warning("Skipping...");
|
|
ConLog.SkipLn();
|
|
#endif
|
|
continue;
|
|
}
|
|
|
|
switch(phdr_arr[i].p_type)
|
|
{
|
|
case 0x00000001: //LOAD
|
|
if(phdr_arr[i].p_memsz)
|
|
{
|
|
Memory.MainMem.AllocFixed(offset + phdr_arr[i].p_vaddr, phdr_arr[i].p_memsz);
|
|
|
|
if(phdr_arr[i].p_filesz)
|
|
{
|
|
elf64_f.Seek(phdr_arr[i].p_offset);
|
|
elf64_f.Read(&Memory[offset + phdr_arr[i].p_vaddr], phdr_arr[i].p_filesz);
|
|
Emu.GetSFuncManager().StaticAnalyse(&Memory[offset + phdr_arr[i].p_vaddr], phdr_arr[i].p_filesz, phdr_arr[i].p_vaddr);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x00000007: //TLS
|
|
Emu.SetTLSData(offset + phdr_arr[i].p_vaddr, phdr_arr[i].p_filesz, phdr_arr[i].p_memsz);
|
|
break;
|
|
|
|
case 0x60000001: //LOOS+1
|
|
{
|
|
if(!phdr_arr[i].p_filesz) break;
|
|
|
|
const sys_process_param& proc_param = *(sys_process_param*)&Memory[offset + phdr_arr[i].p_vaddr];
|
|
|
|
if(re(proc_param.size) < sizeof(sys_process_param))
|
|
{
|
|
ConLog.Warning("Bad proc param size! [0x%x : 0x%x]", re(proc_param.size), sizeof(sys_process_param));
|
|
}
|
|
|
|
if(re(proc_param.magic) != 0x13bcc5f6)
|
|
{
|
|
ConLog.Error("Bad magic! [0x%x]", Memory.Reverse32(proc_param.magic));
|
|
}
|
|
else
|
|
{
|
|
sys_process_param_info& info = Emu.GetInfo().GetProcParam();
|
|
info.sdk_version = re(proc_param.info.sdk_version);
|
|
info.primary_prio = re(proc_param.info.primary_prio);
|
|
info.primary_stacksize = re(proc_param.info.primary_stacksize);
|
|
info.malloc_pagesize = re(proc_param.info.malloc_pagesize);
|
|
info.ppc_seg = re(proc_param.info.ppc_seg);
|
|
//info.crash_dump_param_addr = re(proc_param.info.crash_dump_param_addr);
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.Write("*** sdk version: 0x%x", info.sdk_version);
|
|
ConLog.Write("*** primary prio: %d", info.primary_prio);
|
|
ConLog.Write("*** primary stacksize: 0x%x", info.primary_stacksize);
|
|
ConLog.Write("*** malloc pagesize: 0x%x", info.malloc_pagesize);
|
|
ConLog.Write("*** ppc seg: 0x%x", info.ppc_seg);
|
|
//ConLog.Write("*** crash dump param addr: 0x%x", info.crash_dump_param_addr);
|
|
#endif
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 0x60000002: //LOOS+2
|
|
{
|
|
if(!phdr_arr[i].p_filesz) break;
|
|
|
|
sys_proc_prx_param proc_prx_param = *(sys_proc_prx_param*)&Memory[offset + phdr_arr[i].p_vaddr];
|
|
|
|
proc_prx_param.size = re(proc_prx_param.size);
|
|
proc_prx_param.magic = re(proc_prx_param.magic);
|
|
proc_prx_param.version = re(proc_prx_param.version);
|
|
proc_prx_param.libentstart = re(proc_prx_param.libentstart);
|
|
proc_prx_param.libentend = re(proc_prx_param.libentend);
|
|
proc_prx_param.libstubstart = re(proc_prx_param.libstubstart);
|
|
proc_prx_param.libstubend = re(proc_prx_param.libstubend);
|
|
proc_prx_param.ver = re(proc_prx_param.ver);
|
|
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.Write("*** size: 0x%x", proc_prx_param.size);
|
|
ConLog.Write("*** magic: 0x%x", proc_prx_param.magic);
|
|
ConLog.Write("*** version: 0x%x", proc_prx_param.version);
|
|
ConLog.Write("*** libentstart: 0x%x", proc_prx_param.libentstart);
|
|
ConLog.Write("*** libentend: 0x%x", proc_prx_param.libentend);
|
|
ConLog.Write("*** libstubstart: 0x%x", proc_prx_param.libstubstart);
|
|
ConLog.Write("*** libstubend: 0x%x", proc_prx_param.libstubend);
|
|
ConLog.Write("*** ver: 0x%x", proc_prx_param.ver);
|
|
#endif
|
|
|
|
if(proc_prx_param.magic != 0x1b434cec)
|
|
{
|
|
ConLog.Error("Bad magic! (0x%x)", proc_prx_param.magic);
|
|
}
|
|
else
|
|
{
|
|
for(u32 s=proc_prx_param.libstubstart; s<proc_prx_param.libstubend; s+=sizeof(Elf64_StubHeader))
|
|
{
|
|
Elf64_StubHeader stub = *(Elf64_StubHeader*)Memory.GetMemFromAddr(offset + s);
|
|
|
|
stub.s_size = re(stub.s_size);
|
|
stub.s_version = re(stub.s_version);
|
|
stub.s_unk0 = re(stub.s_unk0);
|
|
stub.s_unk1 = re(stub.s_unk1);
|
|
stub.s_imports = re(stub.s_imports);
|
|
stub.s_modulename = re(stub.s_modulename);
|
|
stub.s_nid = re(stub.s_nid);
|
|
stub.s_text = re(stub.s_text);
|
|
|
|
const std::string& module_name = Memory.ReadString(stub.s_modulename);
|
|
Module* module = Emu.GetModuleManager().GetModuleByName(module_name);
|
|
if(module)
|
|
{
|
|
//module->SetLoaded();
|
|
}
|
|
else
|
|
{
|
|
ConLog.Warning("Unknown module '%s'", module_name.c_str());
|
|
}
|
|
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.SkipLn();
|
|
ConLog.Write("*** size: 0x%x", stub.s_size);
|
|
ConLog.Write("*** version: 0x%x", stub.s_version);
|
|
ConLog.Write("*** unk0: 0x%x", stub.s_unk0);
|
|
ConLog.Write("*** unk1: 0x%x", stub.s_unk1);
|
|
ConLog.Write("*** imports: %d", stub.s_imports);
|
|
ConLog.Write("*** module name: %s [0x%x]", module_name.c_str(), stub.s_modulename);
|
|
ConLog.Write("*** nid: 0x%x", stub.s_nid);
|
|
ConLog.Write("*** text: 0x%x", stub.s_text);
|
|
#endif
|
|
static const u32 section = 4 * 3;
|
|
u64 tbl = Memory.MainMem.AllocAlign(stub.s_imports * 4 * 2);
|
|
u64 dst = Memory.MainMem.AllocAlign(stub.s_imports * section);
|
|
|
|
for(u32 i=0; i<stub.s_imports; ++i)
|
|
{
|
|
const u32 nid = Memory.Read32(stub.s_nid + i*4);
|
|
const u32 text = Memory.Read32(stub.s_text + i*4);
|
|
|
|
if(module)
|
|
{
|
|
if(!module->Load(nid))
|
|
{
|
|
ConLog.Warning("Unimplemented function '%s' in '%s' module", SysCalls::GetHLEFuncName(nid).c_str(), module_name.c_str());
|
|
}
|
|
}
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.Write("import %d:", i+1);
|
|
ConLog.Write("*** nid: 0x%x (0x%x)", nid, stub.s_nid + i*4);
|
|
ConLog.Write("*** text: 0x%x (0x%x)", text, stub.s_text + i*4);
|
|
#endif
|
|
Memory.Write32(stub.s_text + i*4, tbl + i*8);
|
|
|
|
mem32_ptr_t out_tbl(tbl + i*8);
|
|
out_tbl += dst + i*section;
|
|
out_tbl += Emu.GetModuleManager().GetFuncNumById(nid);
|
|
|
|
mem32_ptr_t out_dst(dst + i*section);
|
|
out_dst += OR(11, 2, 2, 0);
|
|
out_dst += SC(2);
|
|
out_dst += BCLR(0x10 | 0x04, 0, 0, 0);
|
|
}
|
|
}
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.SkipLn();
|
|
#endif
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.SkipLn();
|
|
#endif
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool ELF64Loader::LoadShdrData(u64 offset)
|
|
{
|
|
u64 max_addr = 0;
|
|
|
|
for(uint i=0; i<shdr_arr.size(); ++i)
|
|
{
|
|
Elf64_Shdr& shdr = shdr_arr[i];
|
|
|
|
if(i < shdr_name_arr.size())
|
|
{
|
|
#ifdef LOADER_DEBUG
|
|
ConLog.Write("Name: %s", shdr_name_arr[i].c_str());
|
|
#endif
|
|
}
|
|
|
|
#ifdef LOADER_DEBUG
|
|
shdr.Show();
|
|
ConLog.SkipLn();
|
|
#endif
|
|
if(shdr.sh_addr + shdr.sh_size > max_addr) max_addr = shdr.sh_addr + shdr.sh_size;
|
|
|
|
if((shdr.sh_flags & SHF_ALLOC) != SHF_ALLOC) continue;
|
|
|
|
const u64 addr = shdr.sh_addr;
|
|
const u64 size = shdr.sh_size;
|
|
|
|
if(size == 0 || !Memory.IsGoodAddr(offset + addr, size)) continue;
|
|
|
|
if(shdr.sh_addr && shdr.sh_addr < min_addr)
|
|
{
|
|
min_addr = shdr.sh_addr;
|
|
}
|
|
|
|
if(shdr.sh_addr + shdr.sh_size > max_addr)
|
|
{
|
|
max_addr = shdr.sh_addr + shdr.sh_size;
|
|
}
|
|
|
|
if((shdr.sh_type == SHT_RELA) || (shdr.sh_type == SHT_REL))
|
|
{
|
|
ConLog.Error("ELF64 ERROR: Relocation");
|
|
continue;
|
|
}
|
|
|
|
switch(shdr.sh_type)
|
|
{
|
|
case SHT_NOBITS:
|
|
//ConLog.Warning("SHT_NOBITS: addr=0x%llx, size=0x%llx", offset + addr, size);
|
|
//memset(&Memory[offset + addr], 0, size);
|
|
break;
|
|
|
|
case SHT_PROGBITS:
|
|
//elf64_f.Seek(shdr.sh_offset);
|
|
//elf64_f.Read(&Memory[offset + addr], shdr.sh_size);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|