blob: 52cc24e3d2ce0c2df9a73232afdb4c6b550755d8 [file] [log] [blame] [edit]
//===-- Implementation of apply_irelative_relocs (x86_64) -----------------===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "startup/linux/irelative.h"
#include "hdr/elf_macros.h"
#include "hdr/elf_proxy.h"
#include "hdr/link_macros.h"
#include "src/__support/macros/config.h"
namespace LIBC_NAMESPACE_DECL {
void apply_irelative_relocs(intptr_t base, unsigned long /*hwcap*/,
unsigned long /*hwcap2*/) {
for (const ElfW(Rela) *rela = __rela_iplt_start; rela != __rela_iplt_end;
++rela) {
if (ELF64_R_TYPE(rela->r_info) != R_X86_64_IRELATIVE)
continue;
// x86_64 resolvers take no arguments.
// Use unsigned arithmetic to avoid undefined behavior on signed overflow,
// which can occur with very large binaries or high load addresses.
uintptr_t resolver_addr =
static_cast<uintptr_t>(base) + static_cast<uintptr_t>(rela->r_addend);
auto resolver = reinterpret_cast<uintptr_t (*)(void)>(resolver_addr);
uintptr_t result = resolver();
uintptr_t target_addr = static_cast<uintptr_t>(base) + rela->r_offset;
*reinterpret_cast<uintptr_t *>(target_addr) = result;
}
}
} // namespace LIBC_NAMESPACE_DECL