summaryrefslogtreecommitdiff
path: root/Userland
diff options
context:
space:
mode:
authorGunnar Beutner <gbeutner@serenityos.org>2021-05-02 01:32:28 +0200
committerAndreas Kling <kling@serenityos.org>2021-05-03 08:42:39 +0200
commita050b4329057bb61acdd67b836a8bf09d7ab366b (patch)
tree6d617ed8dbf284b49ffb2841449901928b6b748b /Userland
parent6990ab41c8d4ce9865bde0a75e750240f7c5157a (diff)
downloadserenity-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.h6
-rw-r--r--Userland/Libraries/LibELF/Arch/x86_64/plt_trampoline.S12
-rw-r--r--Userland/Libraries/LibELF/DynamicLoader.cpp20
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