| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2017-10-15 00:11:38 -04:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2019-03-20 12:40:09 -04:00
										 |  |  | #include "core/hle/kernel/code_set.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:58:59 +01:00
										 |  |  | #include "core/hle/kernel/process.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  | #include "core/hle/kernel/vm_manager.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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |     VAddr entryPoint; | 
					
						
							| 
									
										
										
										
											2014-06-18 18:58:09 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-07-23 23:10:48 -04:00
										 |  |  |     explicit 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); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |     VAddr GetEntryPoint() const { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         return entryPoint; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     u32 GetFlags() const { | 
					
						
							|  |  |  |         return (u32)(header->e_flags); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |     Kernel::CodeSet LoadInto(VAddr 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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  | Kernel::CodeSet ElfReader::LoadInto(VAddr vaddr) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Loader, "String section: {}", 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) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06: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 { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_DEBUG(Loader, "Prerelocated executable"); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_DEBUG(Loader, "{} segments:", header->e_phnum); | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // First pass : Get the bits into RAM
 | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |     const VAddr base_addr = relocate ? vaddr : 0; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:05:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |     u64 total_image_size = 0; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     for (unsigned int i = 0; i < header->e_phnum; ++i) { | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |         const Elf32_Phdr* p = &segments[i]; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |         if (p->p_type == PT_LOAD) { | 
					
						
							|  |  |  |             total_image_size += (p->p_memsz + 0xFFF) & ~0xFFF; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-18 18:15:53 -04:00
										 |  |  |     Kernel::PhysicalMemory program_image(total_image_size); | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |     std::size_t current_image_position = 0; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |     Kernel::CodeSet codeset; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned int i = 0; i < header->e_phnum; ++i) { | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |         const Elf32_Phdr* p = &segments[i]; | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_DEBUG(Loader, "Type: {} Vaddr: {:08X} Filesz: {:08X} Memsz: {:08X} ", p->p_type, | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |                   p->p_vaddr, 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) { | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |             Kernel::CodeSet::Segment* codeset_segment; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |             u32 permission_flags = p->p_flags & (PF_R | PF_W | PF_X); | 
					
						
							|  |  |  |             if (permission_flags == (PF_R | PF_X)) { | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |                 codeset_segment = &codeset.CodeSegment(); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |             } else if (permission_flags == (PF_R)) { | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |                 codeset_segment = &codeset.RODataSegment(); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |             } else if (permission_flags == (PF_R | PF_W)) { | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |                 codeset_segment = &codeset.DataSegment(); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |                 LOG_ERROR(Loader, "Unexpected ELF PT_LOAD segment id {} with flags {:X}", i, | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |                           p->p_flags); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (codeset_segment->size != 0) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |                 LOG_ERROR(Loader, | 
					
						
							| 
									
										
										
										
											2018-07-02 10:20:50 -06:00
										 |  |  |                           "ELF has more than one segment of the same type. Skipping extra " | 
					
						
							|  |  |  |                           "segment (id {})", | 
					
						
							|  |  |  |                           i); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-24 20:01:45 -04:00
										 |  |  |             const VAddr segment_addr = base_addr + p->p_vaddr; | 
					
						
							|  |  |  |             const u32 aligned_size = (p->p_memsz + 0xFFF) & ~0xFFF; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  |             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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |     codeset.entrypoint = base_addr + header->e_entry; | 
					
						
							| 
									
										
										
										
											2019-03-22 14:51:36 -04:00
										 |  |  |     codeset.memory = std::move(program_image); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06: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 { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | AppLoader_ELF::AppLoader_ELF(FileSys::VirtualFile file) : AppLoader(std::move(file)) {} | 
					
						
							| 
									
										
										
										
											2018-01-20 15:48:37 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  | FileType AppLoader_ELF::IdentifyType(const FileSys::VirtualFile& file) { | 
					
						
							| 
									
										
										
										
											2017-08-03 20:27:13 -04:00
										 |  |  |     static constexpr u16 ELF_MACHINE_ARM{0x28}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     u32 magic = 0; | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     if (4 != file->ReadObject(&magic)) | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |         return FileType::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 20:27:13 -04:00
										 |  |  |     u16 machine = 0; | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     if (2 != file->ReadObject(&machine, 18)) | 
					
						
							| 
									
										
										
										
											2017-08-03 20:27:13 -04:00
										 |  |  |         return FileType::Error; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-15 00:11:38 -04:00
										 |  |  |     if (Common::MakeMagic('\x7f', 'E', 'L', 'F') == magic && ELF_MACHINE_ARM == machine) | 
					
						
							| 
									
										
										
										
											2015-01-06 23:10:13 +00:00
										 |  |  |         return FileType::ELF; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return FileType::Error; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-09 17:03:04 -04:00
										 |  |  | AppLoader_ELF::LoadResult AppLoader_ELF::Load(Kernel::Process& process) { | 
					
						
							|  |  |  |     if (is_loaded) { | 
					
						
							|  |  |  |         return {ResultStatus::ErrorAlreadyLoaded, {}}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-18 21:07:11 -04:00
										 |  |  |     std::vector<u8> buffer = file->ReadAllBytes(); | 
					
						
							| 
									
										
										
										
											2019-04-09 17:03:04 -04:00
										 |  |  |     if (buffer.size() != file->GetSize()) { | 
					
						
							|  |  |  |         return {ResultStatus::ErrorIncorrectELFFileSize, {}}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-29 18:47:00 -04:00
										 |  |  |     const VAddr base_address = process.VMManager().GetCodeRegionBaseAddress(); | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  |     ElfReader elf_reader(&buffer[0]); | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |     Kernel::CodeSet codeset = elf_reader.LoadInto(base_address); | 
					
						
							|  |  |  |     const VAddr entry_point = codeset.entrypoint; | 
					
						
							| 
									
										
										
										
											2015-07-09 22:52:15 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-12 11:36:31 -04:00
										 |  |  |     process.LoadModule(std::move(codeset), entry_point); | 
					
						
							| 
									
										
										
										
											2015-01-06 19:49:25 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     is_loaded = true; | 
					
						
							| 
									
										
										
										
											2019-04-09 17:03:04 -04:00
										 |  |  |     return {ResultStatus::Success, LoadParameters{48, Memory::DEFAULT_STACK_SIZE}}; | 
					
						
							| 
									
										
										
										
											2014-06-16 23:18:10 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace Loader
 |