blob: bc9eb280cedc4b37b8d8da238fc484c74d833f61 [file] [log] [blame] [edit]
//===-- Implementation of apply_irelative_relocs (AArch64) ----------------===//
//
// 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_AARCH64_IRELATIVE)
continue;
// AArch64 resolvers receive hwcap and hwcap2.
// 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 (*)(unsigned long, unsigned long)>(
resolver_addr);
uintptr_t result = resolver(hwcap, hwcap2);
// Write the resolved function pointer to the target location.
uintptr_t target_addr = static_cast<uintptr_t>(base) + rela->r_offset;
*reinterpret_cast<uintptr_t *>(target_addr) = result;
}
}
} // namespace LIBC_NAMESPACE_DECL