forked from eden-emu/eden
		
	common: consolidate ELF structure definitions
This commit is contained in:
		
							parent
							
								
									5c0a31e29f
								
							
						
					
					
						commit
						3c313a43fd
					
				
					 5 changed files with 371 additions and 263 deletions
				
			
		|  | @ -3,73 +3,14 @@ | |||
| 
 | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/elf.h" | ||||
| #include "core/arm/symbols.h" | ||||
| #include "core/core.h" | ||||
| #include "core/memory.h" | ||||
| 
 | ||||
| using namespace Common::ELF; | ||||
| 
 | ||||
| namespace Core { | ||||
| namespace { | ||||
| 
 | ||||
| constexpr u64 ELF_DYNAMIC_TAG_NULL = 0; | ||||
| constexpr u64 ELF_DYNAMIC_TAG_STRTAB = 5; | ||||
| constexpr u64 ELF_DYNAMIC_TAG_SYMTAB = 6; | ||||
| constexpr u64 ELF_DYNAMIC_TAG_SYMENT = 11; | ||||
| 
 | ||||
| enum class ELFSymbolType : u8 { | ||||
|     None = 0, | ||||
|     Object = 1, | ||||
|     Function = 2, | ||||
|     Section = 3, | ||||
|     File = 4, | ||||
|     Common = 5, | ||||
|     TLS = 6, | ||||
| }; | ||||
| 
 | ||||
| enum class ELFSymbolBinding : u8 { | ||||
|     Local = 0, | ||||
|     Global = 1, | ||||
|     Weak = 2, | ||||
| }; | ||||
| 
 | ||||
| enum class ELFSymbolVisibility : u8 { | ||||
|     Default = 0, | ||||
|     Internal = 1, | ||||
|     Hidden = 2, | ||||
|     Protected = 3, | ||||
| }; | ||||
| 
 | ||||
| struct ELF64Symbol { | ||||
|     u32 name_index; | ||||
|     union { | ||||
|         u8 info; | ||||
| 
 | ||||
|         BitField<0, 4, ELFSymbolType> type; | ||||
|         BitField<4, 4, ELFSymbolBinding> binding; | ||||
|     }; | ||||
|     ELFSymbolVisibility visibility; | ||||
|     u16 sh_index; | ||||
|     u64 value; | ||||
|     u64 size; | ||||
| }; | ||||
| static_assert(sizeof(ELF64Symbol) == 0x18, "ELF64Symbol has incorrect size."); | ||||
| 
 | ||||
| struct ELF32Symbol { | ||||
|     u32 name_index; | ||||
|     u32 value; | ||||
|     u32 size; | ||||
|     union { | ||||
|         u8 info; | ||||
| 
 | ||||
|         BitField<0, 4, ELFSymbolType> type; | ||||
|         BitField<4, 4, ELFSymbolBinding> binding; | ||||
|     }; | ||||
|     ELFSymbolVisibility visibility; | ||||
|     u16 sh_index; | ||||
| }; | ||||
| static_assert(sizeof(ELF32Symbol) == 0x10, "ELF32Symbol has incorrect size."); | ||||
| 
 | ||||
| } // Anonymous namespace
 | ||||
| 
 | ||||
| namespace Symbols { | ||||
| 
 | ||||
| template <typename Word, typename ELFSymbol, typename ByteReader> | ||||
|  | @ -110,15 +51,15 @@ static Symbols GetSymbols(ByteReader ReadBytes) { | |||
|         const Word value = ReadWord(dynamic_index + sizeof(Word)); | ||||
|         dynamic_index += 2 * sizeof(Word); | ||||
| 
 | ||||
|         if (tag == ELF_DYNAMIC_TAG_NULL) { | ||||
|         if (tag == ElfDtNull) { | ||||
|             break; | ||||
|         } | ||||
| 
 | ||||
|         if (tag == ELF_DYNAMIC_TAG_STRTAB) { | ||||
|         if (tag == ElfDtStrtab) { | ||||
|             string_table_offset = value; | ||||
|         } else if (tag == ELF_DYNAMIC_TAG_SYMTAB) { | ||||
|         } else if (tag == ElfDtSymtab) { | ||||
|             symbol_table_offset = value; | ||||
|         } else if (tag == ELF_DYNAMIC_TAG_SYMENT) { | ||||
|         } else if (tag == ElfDtSyment) { | ||||
|             symbol_entry_size = value; | ||||
|         } | ||||
|     } | ||||
|  | @ -134,14 +75,14 @@ static Symbols GetSymbols(ByteReader ReadBytes) { | |||
|         ELFSymbol symbol{}; | ||||
|         ReadBytes(&symbol, symbol_index, sizeof(ELFSymbol)); | ||||
| 
 | ||||
|         VAddr string_offset = string_table_offset + symbol.name_index; | ||||
|         VAddr string_offset = string_table_offset + symbol.st_name; | ||||
|         std::string name; | ||||
|         for (u8 c = Read8(string_offset); c != 0; c = Read8(++string_offset)) { | ||||
|             name += static_cast<char>(c); | ||||
|         } | ||||
| 
 | ||||
|         symbol_index += symbol_entry_size; | ||||
|         out[name] = std::make_pair(symbol.value, symbol.size); | ||||
|         out[name] = std::make_pair(symbol.st_value, symbol.st_size); | ||||
|     } | ||||
| 
 | ||||
|     return out; | ||||
|  | @ -152,9 +93,9 @@ Symbols GetSymbols(VAddr base, Core::Memory::Memory& memory, bool is_64) { | |||
|         [&](void* ptr, size_t offset, size_t size) { memory.ReadBlock(base + offset, ptr, size); }}; | ||||
| 
 | ||||
|     if (is_64) { | ||||
|         return GetSymbols<u64, ELF64Symbol>(ReadBytes); | ||||
|         return GetSymbols<u64, Elf64_Sym>(ReadBytes); | ||||
|     } else { | ||||
|         return GetSymbols<u32, ELF32Symbol>(ReadBytes); | ||||
|         return GetSymbols<u32, Elf32_Sym>(ReadBytes); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | @ -164,9 +105,9 @@ Symbols GetSymbols(std::span<const u8> data, bool is_64) { | |||
|     }}; | ||||
| 
 | ||||
|     if (is_64) { | ||||
|         return GetSymbols<u64, ELF64Symbol>(ReadBytes); | ||||
|         return GetSymbols<u64, Elf64_Sym>(ReadBytes); | ||||
|     } else { | ||||
|         return GetSymbols<u32, ELF32Symbol>(ReadBytes); | ||||
|         return GetSymbols<u32, Elf32_Sym>(ReadBytes); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  |  | |||
|  | @ -11,10 +11,13 @@ | |||
| #include "common/alignment.h" | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/div_ceil.h" | ||||
| #include "common/elf.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "core/hle/service/jit/jit_context.h" | ||||
| #include "core/memory.h" | ||||
| 
 | ||||
| using namespace Common::ELF; | ||||
| 
 | ||||
| namespace Service::JIT { | ||||
| 
 | ||||
| constexpr std::array<u8, 8> SVC0_ARM64 = { | ||||
|  | @ -26,25 +29,6 @@ constexpr std::array HELPER_FUNCTIONS{ | |||
|     "_stop", "_resolve", "_panic", "memcpy", "memmove", "memset", | ||||
| }; | ||||
| 
 | ||||
| struct Elf64_Dyn { | ||||
|     u64 d_tag; | ||||
|     u64 d_un; | ||||
| }; | ||||
| 
 | ||||
| struct Elf64_Rela { | ||||
|     u64 r_offset; | ||||
|     u64 r_info; | ||||
|     s64 r_addend; | ||||
| }; | ||||
| 
 | ||||
| static constexpr u32 Elf64_RelaType(const Elf64_Rela* rela) { | ||||
|     return static_cast<u32>(rela->r_info); | ||||
| } | ||||
| 
 | ||||
| constexpr int DT_RELA = 7;               /* Address of Rela relocs */ | ||||
| constexpr int DT_RELASZ = 8;             /* Total size of Rela relocs */ | ||||
| constexpr int R_AARCH64_RELATIVE = 1027; /* Adjust by program base.  */ | ||||
| 
 | ||||
| constexpr size_t STACK_ALIGN = 16; | ||||
| 
 | ||||
| class JITContextImpl; | ||||
|  | @ -206,17 +190,17 @@ public: | |||
|             if (!dyn.d_tag) { | ||||
|                 break; | ||||
|             } | ||||
|             if (dyn.d_tag == DT_RELA) { | ||||
|                 rela_dyn = dyn.d_un; | ||||
|             if (dyn.d_tag == ElfDtRela) { | ||||
|                 rela_dyn = dyn.d_un.d_ptr; | ||||
|             } | ||||
|             if (dyn.d_tag == DT_RELASZ) { | ||||
|                 num_rela = dyn.d_un / sizeof(Elf64_Rela); | ||||
|             if (dyn.d_tag == ElfDtRelasz) { | ||||
|                 num_rela = dyn.d_un.d_val / sizeof(Elf64_Rela); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         for (size_t i = 0; i < num_rela; i++) { | ||||
|             const auto rela{callbacks->ReadMemory<Elf64_Rela>(rela_dyn + i * sizeof(Elf64_Rela))}; | ||||
|             if (Elf64_RelaType(&rela) != R_AARCH64_RELATIVE) { | ||||
|             if (Elf64RelType(rela.r_info) != ElfAArch64Relative) { | ||||
|                 continue; | ||||
|             } | ||||
|             const VAddr contents{callbacks->MemoryRead64(rela.r_offset)}; | ||||
|  |  | |||
|  | @ -6,6 +6,7 @@ | |||
| #include <memory> | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/elf.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "core/hle/kernel/code_set.h" | ||||
| #include "core/hle/kernel/k_page_table.h" | ||||
|  | @ -13,159 +14,7 @@ | |||
| #include "core/loader/elf.h" | ||||
| #include "core/memory.h" | ||||
| 
 | ||||
| ////////////////////////////////////////////////////////////////////////////////////////////////////
 | ||||
| // ELF Header Constants
 | ||||
| 
 | ||||
| // File type
 | ||||
| enum ElfType { | ||||
|     ET_NONE = 0, | ||||
|     ET_REL = 1, | ||||
|     ET_EXEC = 2, | ||||
|     ET_DYN = 3, | ||||
|     ET_CORE = 4, | ||||
|     ET_LOPROC = 0xFF00, | ||||
|     ET_HIPROC = 0xFFFF, | ||||
| }; | ||||
| 
 | ||||
| // Machine/Architecture
 | ||||
| enum ElfMachine { | ||||
|     EM_NONE = 0, | ||||
|     EM_M32 = 1, | ||||
|     EM_SPARC = 2, | ||||
|     EM_386 = 3, | ||||
|     EM_68K = 4, | ||||
|     EM_88K = 5, | ||||
|     EM_860 = 7, | ||||
|     EM_MIPS = 8 | ||||
| }; | ||||
| 
 | ||||
| // File version
 | ||||
| #define EV_NONE 0 | ||||
| #define EV_CURRENT 1 | ||||
| 
 | ||||
| // Identification index
 | ||||
| #define EI_MAG0 0 | ||||
| #define EI_MAG1 1 | ||||
| #define EI_MAG2 2 | ||||
| #define EI_MAG3 3 | ||||
| #define EI_CLASS 4 | ||||
| #define EI_DATA 5 | ||||
| #define EI_VERSION 6 | ||||
| #define EI_PAD 7 | ||||
| #define EI_NIDENT 16 | ||||
| 
 | ||||
| // Sections constants
 | ||||
| 
 | ||||
| // Section types
 | ||||
| #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 | ||||
| #define SHT_LOPROC 0x70000000 | ||||
| #define SHT_HIPROC 0x7FFFFFFF | ||||
| #define SHT_LOUSER 0x80000000 | ||||
| #define SHT_HIUSER 0xFFFFFFFF | ||||
| 
 | ||||
| // Section flags
 | ||||
| enum ElfSectionFlags { | ||||
|     SHF_WRITE = 0x1, | ||||
|     SHF_ALLOC = 0x2, | ||||
|     SHF_EXECINSTR = 0x4, | ||||
|     SHF_MASKPROC = 0xF0000000, | ||||
| }; | ||||
| 
 | ||||
| // Segment types
 | ||||
| #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 | ||||
| 
 | ||||
| // Segment flags
 | ||||
| #define PF_X 0x1 | ||||
| #define PF_W 0x2 | ||||
| #define PF_R 0x4 | ||||
| #define PF_MASKPROC 0xF0000000 | ||||
| 
 | ||||
| typedef unsigned int Elf32_Addr; | ||||
| typedef unsigned short Elf32_Half; | ||||
| typedef unsigned int Elf32_Off; | ||||
| typedef signed int Elf32_Sword; | ||||
| typedef unsigned int Elf32_Word; | ||||
| 
 | ||||
| ////////////////////////////////////////////////////////////////////////////////////////////////////
 | ||||
| // ELF file header
 | ||||
| 
 | ||||
| struct Elf32_Ehdr { | ||||
|     unsigned char e_ident[EI_NIDENT]; | ||||
|     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; | ||||
| }; | ||||
| 
 | ||||
| // Section header
 | ||||
| struct Elf32_Shdr { | ||||
|     Elf32_Word sh_name; | ||||
|     Elf32_Word sh_type; | ||||
|     Elf32_Word sh_flags; | ||||
|     Elf32_Addr sh_addr; | ||||
|     Elf32_Off sh_offset; | ||||
|     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; | ||||
|     Elf32_Off p_offset; | ||||
|     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 { | ||||
|     Elf32_Word st_name; | ||||
|     Elf32_Addr st_value; | ||||
|     Elf32_Word st_size; | ||||
|     unsigned char st_info; | ||||
|     unsigned char st_other; | ||||
|     Elf32_Half st_shndx; | ||||
| }; | ||||
| 
 | ||||
| // Relocation entries
 | ||||
| struct Elf32_Rel { | ||||
|     Elf32_Addr r_offset; | ||||
|     Elf32_Word r_info; | ||||
| }; | ||||
| using namespace Common::ELF; | ||||
| 
 | ||||
| ////////////////////////////////////////////////////////////////////////////////////////////////////
 | ||||
| // ElfReader class
 | ||||
|  | @ -193,11 +42,11 @@ public: | |||
|     } | ||||
| 
 | ||||
|     // Quick accessors
 | ||||
|     ElfType GetType() const { | ||||
|         return (ElfType)(header->e_type); | ||||
|     u16 GetType() const { | ||||
|         return header->e_type; | ||||
|     } | ||||
|     ElfMachine GetMachine() const { | ||||
|         return (ElfMachine)(header->e_machine); | ||||
|     u16 GetMachine() const { | ||||
|         return header->e_machine; | ||||
|     } | ||||
|     VAddr GetEntryPoint() const { | ||||
|         return entryPoint; | ||||
|  | @ -220,13 +69,13 @@ public: | |||
|     const u8* GetSectionDataPtr(int section) const { | ||||
|         if (section < 0 || section >= header->e_shnum) | ||||
|             return nullptr; | ||||
|         if (sections[section].sh_type != SHT_NOBITS) | ||||
|         if (sections[section].sh_type != ElfShtNobits) | ||||
|             return GetPtr(sections[section].sh_offset); | ||||
|         else | ||||
|             return nullptr; | ||||
|     } | ||||
|     bool IsCodeSection(int section) const { | ||||
|         return sections[section].sh_type == SHT_PROGBITS; | ||||
|         return sections[section].sh_type == ElfShtProgBits; | ||||
|     } | ||||
|     const u8* GetSegmentPtr(int segment) { | ||||
|         return GetPtr(segments[segment].p_offset); | ||||
|  | @ -256,7 +105,7 @@ ElfReader::ElfReader(void* ptr) { | |||
| } | ||||
| 
 | ||||
| const char* ElfReader::GetSectionName(int section) const { | ||||
|     if (sections[section].sh_type == SHT_NULL) | ||||
|     if (sections[section].sh_type == ElfShtNull) | ||||
|         return nullptr; | ||||
| 
 | ||||
|     int name_offset = sections[section].sh_name; | ||||
|  | @ -272,7 +121,7 @@ Kernel::CodeSet ElfReader::LoadInto(VAddr vaddr) { | |||
|     LOG_DEBUG(Loader, "String section: {}", header->e_shstrndx); | ||||
| 
 | ||||
|     // Should we relocate?
 | ||||
|     relocate = (header->e_type != ET_EXEC); | ||||
|     relocate = (header->e_type != ElfTypeExec); | ||||
| 
 | ||||
|     if (relocate) { | ||||
|         LOG_DEBUG(Loader, "Relocatable module"); | ||||
|  | @ -288,7 +137,7 @@ Kernel::CodeSet ElfReader::LoadInto(VAddr vaddr) { | |||
|     u64 total_image_size = 0; | ||||
|     for (unsigned int i = 0; i < header->e_phnum; ++i) { | ||||
|         const Elf32_Phdr* p = &segments[i]; | ||||
|         if (p->p_type == PT_LOAD) { | ||||
|         if (p->p_type == ElfPtLoad) { | ||||
|             total_image_size += (p->p_memsz + 0xFFF) & ~0xFFF; | ||||
|         } | ||||
|     } | ||||
|  | @ -303,14 +152,14 @@ Kernel::CodeSet ElfReader::LoadInto(VAddr vaddr) { | |||
|         LOG_DEBUG(Loader, "Type: {} Vaddr: {:08X} Filesz: {:08X} Memsz: {:08X} ", p->p_type, | ||||
|                   p->p_vaddr, p->p_filesz, p->p_memsz); | ||||
| 
 | ||||
|         if (p->p_type == PT_LOAD) { | ||||
|         if (p->p_type == ElfPtLoad) { | ||||
|             Kernel::CodeSet::Segment* codeset_segment; | ||||
|             u32 permission_flags = p->p_flags & (PF_R | PF_W | PF_X); | ||||
|             if (permission_flags == (PF_R | PF_X)) { | ||||
|             u32 permission_flags = p->p_flags & (ElfPfRead | ElfPfWrite | ElfPfExec); | ||||
|             if (permission_flags == (ElfPfRead | ElfPfExec)) { | ||||
|                 codeset_segment = &codeset.CodeSegment(); | ||||
|             } else if (permission_flags == (PF_R)) { | ||||
|             } else if (permission_flags == (ElfPfRead)) { | ||||
|                 codeset_segment = &codeset.RODataSegment(); | ||||
|             } else if (permission_flags == (PF_R | PF_W)) { | ||||
|             } else if (permission_flags == (ElfPfRead | ElfPfWrite)) { | ||||
|                 codeset_segment = &codeset.DataSegment(); | ||||
|             } else { | ||||
|                 LOG_ERROR(Loader, "Unexpected ELF PT_LOAD segment id {} with flags {:X}", i, | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Liam
						Liam