#include "stdafx.h" #include "Loader.h" #include "ELF.h" #include "SELF.h" #include "PSF.h" #include "Emu/FS/vfsLocalFile.h" static const u64 g_spu_offset = 0x10000; const wxString Ehdr_DataToString(const u8 data) { if(data > 1) return wxString::Format("%d's complement, big endian", data); if(data < 1) return "Data is not found"; return wxString::Format("%d's complement, small endian", data); } const wxString Ehdr_TypeToString(const u16 type) { switch(type) { case 0: return "NULL"; case 2: return "EXEC (Executable file)"; }; return wxString::Format("Unknown (%d)", type); } const wxString Ehdr_OS_ABIToString(const u8 os_abi) { switch(os_abi) { case 0x0 : return "UNIX System V"; case 0x66: return "Cell OS LV-2"; }; return wxString::Format("Unknown (%x)", os_abi); } const wxString Ehdr_MachineToString(const u16 machine) { switch(machine) { case MACHINE_MIPS: return "MIPS"; case MACHINE_PPC64: return "PowerPC64"; case MACHINE_SPU: return "SPU"; case MACHINE_ARM: return "ARM"; }; return wxString::Format("Unknown (%x)", machine); } const wxString Phdr_FlagsToString(u32 flags) { enum {ppu_R = 0x1, ppu_W = 0x2, ppu_E = 0x4}; enum {spu_E = 0x1, spu_W = 0x2, spu_R = 0x4}; enum {rsx_R = 0x1, rsx_W = 0x2, rsx_E = 0x4}; #define FLAGS_TO_STRING(f) \ wxString(f & ##f##_R ? "R" : "-") + \ wxString(f & ##f##_W ? "W" : "-") + \ wxString(f & ##f##_E ? "E" : "-") const u8 ppu = flags & 0xf; const u8 spu = (flags >> 0x14) & 0xf; const u8 rsx = (flags >> 0x18) & 0xf; wxString ret = wxEmptyString; ret += wxString::Format("[0x%x] ", flags); flags &= ~ppu; flags &= ~spu << 0x14; flags &= ~rsx << 0x18; if(flags != 0) return wxString::Format("Unknown %s PPU[%x] SPU[%x] RSX[%x]", ret, ppu, spu, rsx); ret += "PPU[" + FLAGS_TO_STRING(ppu) + "] "; ret += "SPU[" + FLAGS_TO_STRING(spu) + "] "; ret += "RSX[" + FLAGS_TO_STRING(rsx) + "]"; return ret; } const wxString Phdr_TypeToString(const u32 type) { switch(type) { case 0x00000001: return "LOAD"; case 0x00000004: return "NOTE"; case 0x00000007: return "TLS"; case 0x60000001: return "LOOS+1"; case 0x60000002: return "LOOS+2"; }; return wxString::Format("Unknown (%x)", type); } Loader::Loader() : m_stream(nullptr) , m_loader(nullptr) { } Loader::Loader(vfsFileBase& stream) : m_stream(&stream) , m_loader(nullptr) { } Loader::~Loader() { delete m_loader; m_loader = nullptr; } void Loader::Open(vfsFileBase& stream) { m_stream = &stream; } LoaderBase* Loader::SearchLoader() { if(!m_stream) return nullptr; LoaderBase* l; if((l=new ELFLoader(*m_stream))->LoadInfo()) return l; delete l; if((l=new SELFLoader(*m_stream))->LoadInfo()) return l; delete l; return nullptr; } bool Loader::Analyze() { delete m_loader; m_loader = SearchLoader(); if(!m_loader) { ConLog.Error("Unknown file type"); return false; } machine = m_loader->GetMachine(); entry = m_loader->GetMachine() == MACHINE_SPU ? m_loader->GetEntry() + g_spu_offset : m_loader->GetEntry(); return true; } bool Loader::Load() { if(!m_loader) return false; if(!m_loader->LoadData(m_loader->GetMachine() == MACHINE_SPU ? g_spu_offset : 0)) { ConLog.Error("Broken file"); return false; } const wxString& root = wxFileName(wxFileName(m_stream->GetPath()).GetPath()).GetPath(); const wxString& psf_path = root + "\\" + "PARAM.SFO"; if(wxFileExists(psf_path)) { vfsLocalFile f(psf_path); PSFLoader psf_l(f); if(psf_l.Load()) { CurGameInfo = psf_l.m_info; CurGameInfo.root = root; psf_l.Close(); } } return true; }