//===-- 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(base) + static_cast(rela->r_addend); auto resolver = reinterpret_cast( resolver_addr); uintptr_t result = resolver(hwcap, hwcap2); // Write the resolved function pointer to the target location. uintptr_t target_addr = static_cast(base) + rela->r_offset; *reinterpret_cast(target_addr) = result; } } } // namespace LIBC_NAMESPACE_DECL