| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Copyright 2013 Dolphin Emulator Project / 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:58:59 +01:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | #include "common/file_util.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:58:59 +01:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 15:25:15 -05:00
										 |  |  | #include "core/hle/kernel/resource_limit.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "core/loader/elf.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-12 22:38:29 -03:00
										 |  |  | #include "core/memory.h"
 | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | using Kernel::SharedPtr; | 
					
						
							|  |  |  | using Kernel::CodeSet; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // ELF Header Constants
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // File type
 | 
					
						
							|  |  |  | enum ElfType { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     ET_NONE = 0, | 
					
						
							|  |  |  |     ET_REL = 1, | 
					
						
							|  |  |  |     ET_EXEC = 2, | 
					
						
							|  |  |  |     ET_DYN = 3, | 
					
						
							|  |  |  |     ET_CORE = 4, | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     ET_LOPROC = 0xFF00, | 
					
						
							|  |  |  |     ET_HIPROC = 0xFFFF, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Machine/Architecture
 | 
					
						
							|  |  |  | enum ElfMachine { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     EM_NONE = 0, | 
					
						
							|  |  |  |     EM_M32 = 1, | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     EM_SPARC = 2, | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     EM_386 = 3, | 
					
						
							|  |  |  |     EM_68K = 4, | 
					
						
							|  |  |  |     EM_88K = 5, | 
					
						
							|  |  |  |     EM_860 = 7, | 
					
						
							|  |  |  |     EM_MIPS = 8 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // File version
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define EV_NONE 0
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | #define EV_CURRENT 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Identification index
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define EI_MAG0 0
 | 
					
						
							|  |  |  | #define EI_MAG1 1
 | 
					
						
							|  |  |  | #define EI_MAG2 2
 | 
					
						
							|  |  |  | #define EI_MAG3 3
 | 
					
						
							|  |  |  | #define EI_CLASS 4
 | 
					
						
							|  |  |  | #define EI_DATA 5
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | #define EI_VERSION 6
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define EI_PAD 7
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | #define EI_NIDENT 16
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Sections constants
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Section types
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define SHT_NULL 0
 | 
					
						
							|  |  |  | #define SHT_PROGBITS 1
 | 
					
						
							|  |  |  | #define SHT_SYMTAB 2
 | 
					
						
							|  |  |  | #define SHT_STRTAB 3
 | 
					
						
							|  |  |  | #define SHT_RELA 4
 | 
					
						
							|  |  |  | #define SHT_HASH 5
 | 
					
						
							|  |  |  | #define SHT_DYNAMIC 6
 | 
					
						
							|  |  |  | #define SHT_NOTE 7
 | 
					
						
							|  |  |  | #define SHT_NOBITS 8
 | 
					
						
							|  |  |  | #define SHT_REL 9
 | 
					
						
							|  |  |  | #define SHT_SHLIB 10
 | 
					
						
							|  |  |  | #define SHT_DYNSYM 11
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | #define SHT_LOPROC 0x70000000
 | 
					
						
							|  |  |  | #define SHT_HIPROC 0x7FFFFFFF
 | 
					
						
							|  |  |  | #define SHT_LOUSER 0x80000000
 | 
					
						
							|  |  |  | #define SHT_HIUSER 0xFFFFFFFF
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Section flags
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | enum ElfSectionFlags { | 
					
						
							|  |  |  |     SHF_WRITE = 0x1, | 
					
						
							|  |  |  |     SHF_ALLOC = 0x2, | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     SHF_EXECINSTR = 0x4, | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     SHF_MASKPROC = 0xF0000000, | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Segment types
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define PT_NULL 0
 | 
					
						
							|  |  |  | #define PT_LOAD 1
 | 
					
						
							|  |  |  | #define PT_DYNAMIC 2
 | 
					
						
							|  |  |  | #define PT_INTERP 3
 | 
					
						
							|  |  |  | #define PT_NOTE 4
 | 
					
						
							|  |  |  | #define PT_SHLIB 5
 | 
					
						
							|  |  |  | #define PT_PHDR 6
 | 
					
						
							|  |  |  | #define PT_LOPROC 0x70000000
 | 
					
						
							|  |  |  | #define PT_HIPROC 0x7FFFFFFF
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | // Segment flags
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #define PF_X 0x1
 | 
					
						
							|  |  |  | #define PF_W 0x2
 | 
					
						
							|  |  |  | #define PF_R 0x4
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | #define PF_MASKPROC 0xF0000000
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | typedef unsigned int Elf32_Addr; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | typedef unsigned short Elf32_Half; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | typedef unsigned int Elf32_Off; | 
					
						
							|  |  |  | typedef signed int Elf32_Sword; | 
					
						
							|  |  |  | typedef unsigned int Elf32_Word; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // ELF file header
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct Elf32_Ehdr { | 
					
						
							|  |  |  |     unsigned char e_ident[EI_NIDENT]; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Elf32_Half e_type; | 
					
						
							|  |  |  |     Elf32_Half e_machine; | 
					
						
							|  |  |  |     Elf32_Word e_version; | 
					
						
							|  |  |  |     Elf32_Addr e_entry; | 
					
						
							|  |  |  |     Elf32_Off e_phoff; | 
					
						
							|  |  |  |     Elf32_Off e_shoff; | 
					
						
							|  |  |  |     Elf32_Word e_flags; | 
					
						
							|  |  |  |     Elf32_Half e_ehsize; | 
					
						
							|  |  |  |     Elf32_Half e_phentsize; | 
					
						
							|  |  |  |     Elf32_Half e_phnum; | 
					
						
							|  |  |  |     Elf32_Half e_shentsize; | 
					
						
							|  |  |  |     Elf32_Half e_shnum; | 
					
						
							|  |  |  |     Elf32_Half e_shstrndx; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Section header
 | 
					
						
							|  |  |  | struct Elf32_Shdr { | 
					
						
							|  |  |  |     Elf32_Word sh_name; | 
					
						
							|  |  |  |     Elf32_Word sh_type; | 
					
						
							|  |  |  |     Elf32_Word sh_flags; | 
					
						
							|  |  |  |     Elf32_Addr sh_addr; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Elf32_Off sh_offset; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     Elf32_Word sh_size; | 
					
						
							|  |  |  |     Elf32_Word sh_link; | 
					
						
							|  |  |  |     Elf32_Word sh_info; | 
					
						
							|  |  |  |     Elf32_Word sh_addralign; | 
					
						
							|  |  |  |     Elf32_Word sh_entsize; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Segment header
 | 
					
						
							|  |  |  | struct Elf32_Phdr { | 
					
						
							|  |  |  |     Elf32_Word p_type; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Elf32_Off p_offset; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     Elf32_Addr p_vaddr; | 
					
						
							|  |  |  |     Elf32_Addr p_paddr; | 
					
						
							|  |  |  |     Elf32_Word p_filesz; | 
					
						
							|  |  |  |     Elf32_Word p_memsz; | 
					
						
							|  |  |  |     Elf32_Word p_flags; | 
					
						
							|  |  |  |     Elf32_Word p_align; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Symbol table entry
 | 
					
						
							|  |  |  | struct Elf32_Sym { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Elf32_Word st_name; | 
					
						
							|  |  |  |     Elf32_Addr st_value; | 
					
						
							|  |  |  |     Elf32_Word st_size; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     unsigned char st_info; | 
					
						
							|  |  |  |     unsigned char st_other; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Elf32_Half st_shndx; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Relocation entries
 | 
					
						
							|  |  |  | struct Elf32_Rel { | 
					
						
							|  |  |  |     Elf32_Addr r_offset; | 
					
						
							|  |  |  |     Elf32_Word r_info; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // ElfReader class
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef int SectionID; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ElfReader { | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     char* base; | 
					
						
							|  |  |  |     u32* base32; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Elf32_Ehdr* header; | 
					
						
							|  |  |  |     Elf32_Phdr* segments; | 
					
						
							|  |  |  |     Elf32_Shdr* sections; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     u32* sectionAddrs; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     bool relocate; | 
					
						
							|  |  |  |     u32 entryPoint; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     ElfReader(void* ptr); | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     u32 Read32(int off) const { | 
					
						
							|  |  |  |         return base32[off >> 2]; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Quick accessors
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     ElfType GetType() const { | 
					
						
							|  |  |  |         return (ElfType)(header->e_type); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ElfMachine GetMachine() const { | 
					
						
							|  |  |  |         return (ElfMachine)(header->e_machine); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     u32 GetEntryPoint() const { | 
					
						
							|  |  |  |         return entryPoint; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     u32 GetFlags() const { | 
					
						
							|  |  |  |         return (u32)(header->e_flags); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     SharedPtr<CodeSet> LoadInto(u32 vaddr); | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     int GetNumSegments() const { | 
					
						
							|  |  |  |         return (int)(header->e_phnum); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     int GetNumSections() const { | 
					
						
							|  |  |  |         return (int)(header->e_shnum); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const u8* GetPtr(int offset) const { | 
					
						
							|  |  |  |         return (u8*)base + offset; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const char* GetSectionName(int section) const; | 
					
						
							|  |  |  |     const u8* GetSectionDataPtr(int section) const { | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         if (section < 0 || section >= header->e_shnum) | 
					
						
							|  |  |  |             return nullptr; | 
					
						
							|  |  |  |         if (sections[section].sh_type != SHT_NOBITS) | 
					
						
							|  |  |  |             return GetPtr(sections[section].sh_offset); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     bool IsCodeSection(int section) const { | 
					
						
							|  |  |  |         return sections[section].sh_type == SHT_PROGBITS; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     const u8* GetSegmentPtr(int segment) { | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return GetPtr(segments[segment].p_offset); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     u32 GetSectionAddr(SectionID section) const { | 
					
						
							|  |  |  |         return sectionAddrs[section]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     unsigned int GetSectionSize(SectionID section) const { | 
					
						
							|  |  |  |         return sections[section].sh_size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     SectionID GetSectionByName(const char* name, int firstSection = 0) const; //-1 for not found
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-02 20:32:16 -05:00
										 |  |  |     bool DidRelocate() const { | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return relocate; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | ElfReader::ElfReader(void* ptr) { | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |     base = (char*)ptr; | 
					
						
							| 
									
										
										
										
											2015-01-07 01:30:32 +00:00
										 |  |  |     base32 = (u32*)ptr; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |     header = (Elf32_Ehdr*)ptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 01:30:32 +00:00
										 |  |  |     segments = (Elf32_Phdr*)(base + header->e_phoff); | 
					
						
							|  |  |  |     sections = (Elf32_Shdr*)(base + header->e_shoff); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     entryPoint = header->e_entry; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | const char* ElfReader::GetSectionName(int section) const { | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |     if (sections[section].sh_type == SHT_NULL) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     int name_offset = sections[section].sh_name; | 
					
						
							| 
									
										
										
										
											2016-01-25 02:34:37 -05:00
										 |  |  |     const char* ptr = reinterpret_cast<const char*>(GetSectionDataPtr(header->e_shstrndx)); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (ptr) | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         return ptr + name_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nullptr; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | SharedPtr<CodeSet> ElfReader::LoadInto(u32 vaddr) { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |     LOG_DEBUG(Loader, "String section: %i", header->e_shstrndx); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Should we relocate?
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     relocate = (header->e_type != ET_EXEC); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     if (relocate) { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |         LOG_DEBUG(Loader, "Relocatable module"); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |         entryPoint += vaddr; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |         LOG_DEBUG(Loader, "Prerelocated executable"); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |     LOG_DEBUG(Loader, "%i segments:", header->e_phnum); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // First pass : Get the bits into RAM
 | 
					
						
							| 
									
										
										
										
											2014-06-19 17:52:15 -04:00
										 |  |  |     u32 base_addr = relocate ? vaddr : 0; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     u32 total_image_size = 0; | 
					
						
							|  |  |  |     for (unsigned int i = 0; i < header->e_phnum; ++i) { | 
					
						
							|  |  |  |         Elf32_Phdr* p = &segments[i]; | 
					
						
							|  |  |  |         if (p->p_type == PT_LOAD) { | 
					
						
							|  |  |  |             total_image_size += (p->p_memsz + 0xFFF) & ~0xFFF; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::vector<u8> program_image(total_image_size); | 
					
						
							|  |  |  |     size_t current_image_position = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SharedPtr<CodeSet> codeset = CodeSet::Create("", 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned int i = 0; i < header->e_phnum; ++i) { | 
					
						
							|  |  |  |         Elf32_Phdr* p = &segments[i]; | 
					
						
							|  |  |  |         LOG_DEBUG(Loader, "Type: %i Vaddr: %08X Filesz: %8X Memsz: %8X ", p->p_type, p->p_vaddr, | 
					
						
							| 
									
										
										
										
											2015-01-07 01:30:32 +00:00
										 |  |  |                   p->p_filesz, p->p_memsz); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |         if (p->p_type == PT_LOAD) { | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |             CodeSet::Segment* codeset_segment; | 
					
						
							|  |  |  |             u32 permission_flags = p->p_flags & (PF_R | PF_W | PF_X); | 
					
						
							|  |  |  |             if (permission_flags == (PF_R | PF_X)) { | 
					
						
							|  |  |  |                 codeset_segment = &codeset->code; | 
					
						
							|  |  |  |             } else if (permission_flags == (PF_R)) { | 
					
						
							|  |  |  |                 codeset_segment = &codeset->rodata; | 
					
						
							|  |  |  |             } else if (permission_flags == (PF_R | PF_W)) { | 
					
						
							|  |  |  |                 codeset_segment = &codeset->data; | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 LOG_ERROR(Loader, "Unexpected ELF PT_LOAD segment id %u with flags %X", i, | 
					
						
							|  |  |  |                           p->p_flags); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (codeset_segment->size != 0) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 LOG_ERROR(Loader, "ELF has more than one segment of the same type. Skipping extra " | 
					
						
							|  |  |  |                                   "segment (id %i)", | 
					
						
							|  |  |  |                           i); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             u32 segment_addr = base_addr + p->p_vaddr; | 
					
						
							|  |  |  |             u32 aligned_size = (p->p_memsz + 0xFFF) & ~0xFFF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             codeset_segment->offset = current_image_position; | 
					
						
							|  |  |  |             codeset_segment->addr = segment_addr; | 
					
						
							|  |  |  |             codeset_segment->size = aligned_size; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             memcpy(&program_image[current_image_position], GetSegmentPtr(i), p->p_filesz); | 
					
						
							|  |  |  |             current_image_position += aligned_size; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     codeset->entrypoint = base_addr + header->e_entry; | 
					
						
							|  |  |  |     codeset->memory = std::make_shared<std::vector<u8>>(std::move(program_image)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 23:53:49 -02:00
										 |  |  |     LOG_DEBUG(Loader, "Done loading."); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return codeset; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | SectionID ElfReader::GetSectionByName(const char* name, int firstSection) const { | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     for (int i = firstSection; i < header->e_shnum; i++) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         const char* secname = GetSectionName(i); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (secname != nullptr && strcmp(name, secname) == 0) | 
					
						
							|  |  |  |             return i; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | ////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Loader namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Loader { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  | FileType AppLoader_ELF::IdentifyType(FileUtil::IOFile& file) { | 
					
						
							|  |  |  |     u32 magic; | 
					
						
							|  |  |  |     file.Seek(0, SEEK_SET); | 
					
						
							|  |  |  |     if (1 != file.ReadArray<u32>(&magic, 1)) | 
					
						
							|  |  |  |         return FileType::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (MakeMagic('\x7f', 'E', 'L', 'F') == magic) | 
					
						
							|  |  |  |         return FileType::ELF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return FileType::Error; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-19 17:46:05 -04:00
										 |  |  | ResultStatus AppLoader_ELF::Load() { | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     if (is_loaded) | 
					
						
							|  |  |  |         return ResultStatus::ErrorAlreadyLoaded; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     if (!file.IsOpen()) | 
					
						
							| 
									
										
										
										
											2015-01-06 21:30:40 +00:00
										 |  |  |         return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-06 22:47:43 +00:00
										 |  |  |     // Reset read pointer in case this file has been read before.
 | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     file.Seek(0, SEEK_SET); | 
					
						
							| 
									
										
										
										
											2015-01-06 22:47:43 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-13 20:43:51 -03:00
										 |  |  |     size_t size = file.GetSize(); | 
					
						
							| 
									
										
										
										
											2015-01-06 21:30:40 +00:00
										 |  |  |     std::unique_ptr<u8[]> buffer(new u8[size]); | 
					
						
							| 
									
										
										
										
											2015-07-11 19:16:33 -03:00
										 |  |  |     if (file.ReadBytes(&buffer[0], size) != size) | 
					
						
							| 
									
										
										
										
											2015-01-07 01:30:32 +00:00
										 |  |  |         return ResultStatus::Error; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     ElfReader elf_reader(&buffer[0]); | 
					
						
							|  |  |  |     SharedPtr<CodeSet> codeset = elf_reader.LoadInto(Memory::PROCESS_IMAGE_VADDR); | 
					
						
							|  |  |  |     codeset->name = filename; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Kernel::g_current_process = Kernel::Process::Create(std::move(codeset)); | 
					
						
							| 
									
										
										
										
											2015-05-08 16:51:48 -03:00
										 |  |  |     Kernel::g_current_process->svc_access_mask.set(); | 
					
						
							| 
									
										
										
										
											2015-05-08 18:12:25 -03:00
										 |  |  |     Kernel::g_current_process->address_mappings = default_address_mappings; | 
					
						
							| 
									
										
										
										
											2017-09-24 22:42:42 +01:00
										 |  |  |     Memory::SetCurrentPageTable(&Kernel::g_current_process->vm_manager.page_table); | 
					
						
							| 
									
										
										
										
											2015-05-04 00:01:16 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 15:25:15 -05:00
										 |  |  |     // Attach the default resource limit (APPLICATION) to the process
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     Kernel::g_current_process->resource_limit = | 
					
						
							|  |  |  |         Kernel::ResourceLimit::GetForCategory(Kernel::ResourceLimitCategory::APPLICATION); | 
					
						
							| 
									
										
										
										
											2015-05-12 15:25:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     Kernel::g_current_process->Run(48, Kernel::DEFAULT_STACK_SIZE); | 
					
						
							| 
									
										
										
										
											2015-01-06 19:49:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     is_loaded = true; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  |     return ResultStatus::Success; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Loader
 |