diff options
author | Gunnar Beutner <gbeutner@serenityos.org> | 2021-05-02 01:32:28 +0200 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2021-05-03 08:42:39 +0200 |
commit | a050b4329057bb61acdd67b836a8bf09d7ab366b (patch) | |
tree | 6d617ed8dbf284b49ffb2841449901928b6b748b /Userland | |
parent | 6990ab41c8d4ce9865bde0a75e750240f7c5157a (diff) | |
download | serenity-a050b4329057bb61acdd67b836a8bf09d7ab366b.zip |
LibELF: Implement x86_64 relocation support
There are definitely some relocations missing and this is untested
for now.
Diffstat (limited to 'Userland')
-rw-r--r-- | Userland/Libraries/LibC/elf.h | 6 | ||||
-rw-r--r-- | Userland/Libraries/LibELF/Arch/x86_64/plt_trampoline.S | 12 | ||||
-rw-r--r-- | Userland/Libraries/LibELF/DynamicLoader.cpp | 20 |
3 files changed, 38 insertions, 0 deletions
diff --git a/Userland/Libraries/LibC/elf.h b/Userland/Libraries/LibC/elf.h index 342dbd1f2e..3b1808188e 100644 --- a/Userland/Libraries/LibC/elf.h +++ b/Userland/Libraries/LibC/elf.h @@ -791,3 +791,9 @@ struct elf_args { #define R_386_RELATIVE 8 /* Base address + Addned */ #define R_386_TLS_TPOFF 14 /* Negative offset into the static TLS storage */ #define R_386_TLS_TPOFF32 37 + +#define R_X86_64_NONE 0 +#define R_X86_64_64 1 +#define R_X86_64_GLOB_DAT 6 +#define R_X86_64_JUMP_SLOT 7 +#define R_X86_64_RELATIVE 8 diff --git a/Userland/Libraries/LibELF/Arch/x86_64/plt_trampoline.S b/Userland/Libraries/LibELF/Arch/x86_64/plt_trampoline.S new file mode 100644 index 0000000000..b3d3125ca9 --- /dev/null +++ b/Userland/Libraries/LibELF/Arch/x86_64/plt_trampoline.S @@ -0,0 +1,12 @@ +/* + * Copyright (c) 2021, Gunnar Beutner <gbeutner@serenityos.org> + * + * SPDX-License-Identifier: BSD-2-Clause + */ + + .align 4 + .globl _plt_trampoline + .hidden _plt_trampoline + .type _plt_trampoline,@function +_plt_trampoline: + int3 diff --git a/Userland/Libraries/LibELF/DynamicLoader.cpp b/Userland/Libraries/LibELF/DynamicLoader.cpp index 5b7573d260..c65f2dd1e0 100644 --- a/Userland/Libraries/LibELF/DynamicLoader.cpp +++ b/Userland/Libraries/LibELF/DynamicLoader.cpp @@ -398,11 +398,19 @@ DynamicLoader::RelocationResult DynamicLoader::do_relocation(const ELF::DynamicO patch_ptr = (FlatPtr*)(FlatPtr)relocation.offset(); switch (relocation.type()) { +#ifndef __LP64__ case R_386_NONE: +#else + case R_X86_64_NONE: +#endif // Apparently most loaders will just skip these? // Seems if the 'link editor' generates one something is funky with your code break; +#ifndef __LP64__ case R_386_32: { +#else + case R_X86_64_64: { +#endif auto symbol = relocation.symbol(); auto res = lookup_symbol(symbol); if (!res.has_value()) { @@ -415,6 +423,7 @@ DynamicLoader::RelocationResult DynamicLoader::do_relocation(const ELF::DynamicO *patch_ptr += symbol_address.get(); break; } +#ifndef __LP64__ case R_386_PC32: { auto symbol = relocation.symbol(); auto result = lookup_symbol(symbol); @@ -425,6 +434,9 @@ DynamicLoader::RelocationResult DynamicLoader::do_relocation(const ELF::DynamicO break; } case R_386_GLOB_DAT: { +#else + case R_X86_64_GLOB_DAT: { +#endif auto symbol = relocation.symbol(); auto res = lookup_symbol(symbol); VirtualAddress symbol_location; @@ -444,13 +456,18 @@ DynamicLoader::RelocationResult DynamicLoader::do_relocation(const ELF::DynamicO *patch_ptr = symbol_location.get(); break; } +#ifndef __LP64__ case R_386_RELATIVE: { +#else + case R_X86_64_RELATIVE: { +#endif // FIXME: According to the spec, R_386_relative ones must be done first. // We could explicitly do them first using m_number_of_relocations from DT_RELCOUNT // However, our compiler is nice enough to put them at the front of the relocations for us :) *patch_ptr += (FlatPtr)m_dynamic_object->base_address().as_ptr(); // + addend for RelA (addend for Rel is stored at addr) break; } +#ifndef __LP64__ case R_386_TLS_TPOFF32: case R_386_TLS_TPOFF: { auto symbol = relocation.symbol(); @@ -466,6 +483,9 @@ DynamicLoader::RelocationResult DynamicLoader::do_relocation(const ELF::DynamicO break; } case R_386_JMP_SLOT: { +#else + case R_X86_64_JUMP_SLOT: { +#endif // FIXME: Or BIND_NOW flag passed in? if (m_dynamic_object->must_bind_now()) { // Eagerly BIND_NOW the PLT entries, doing all the symbol looking goodness |