mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-08 14:20:07 +00:00
riscv: support GNU indirect function
Enable riscv glibc to support GNU indirect function
This commit is contained in:
parent
2cd361b511
commit
c31b1f5231
@ -48,3 +48,4 @@ IFUNC sparc64-*-linux*
|
||||
IFUNC sparc-*-linux*
|
||||
# Absolute (SHN_ABS) symbols working correctly.
|
||||
ABSOLUTE
|
||||
IFUNC riscv*-linux*
|
||||
|
56
sysdeps/riscv/dl-irel.h
Normal file
56
sysdeps/riscv/dl-irel.h
Normal file
@ -0,0 +1,56 @@
|
||||
/* Machine-dependent ELF indirect relocation inline functions.
|
||||
RISC-V version.
|
||||
Copyright (C) 2020 Free Software Foundation, Inc.
|
||||
This file is part of the GNU C Library.
|
||||
|
||||
The GNU C Library is free software; you can redistribute it and/or
|
||||
modify it under the terms of the GNU Lesser General Public
|
||||
License as published by the Free Software Foundation; either
|
||||
version 2.1 of the License, or (at your option) any later version.
|
||||
|
||||
The GNU C Library is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||
Lesser General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Lesser General Public
|
||||
License along with the GNU C Library; if not, see
|
||||
<https://www.gnu.org/licenses/>. */
|
||||
|
||||
#ifndef _DL_IREL_H
|
||||
#define _DL_IREL_H
|
||||
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <ldsodefs.h>
|
||||
#include <sysdep.h>
|
||||
|
||||
#define ELF_MACHINE_IRELA 1
|
||||
|
||||
static inline ElfW(Addr)
|
||||
__attribute ((always_inline))
|
||||
elf_ifunc_invoke (ElfW(Addr) addr)
|
||||
{
|
||||
/* The second argument is a void pointer to preserve the extension
|
||||
fexibility. */
|
||||
return ((ElfW(Addr) (*) (uint64_t, void *)) (addr))
|
||||
(GLRO(dl_hwcap), NULL);
|
||||
}
|
||||
|
||||
static inline void
|
||||
__attribute ((always_inline))
|
||||
elf_irela (const ElfW(Rela) *reloc)
|
||||
{
|
||||
ElfW(Addr) *const reloc_addr = (void *) reloc->r_offset;
|
||||
const unsigned long int r_type = ELFW(R_TYPE) (reloc->r_info);
|
||||
|
||||
if (__glibc_likely (r_type == R_RISCV_IRELATIVE))
|
||||
{
|
||||
ElfW(Addr) value = elf_ifunc_invoke (reloc->r_addend);
|
||||
*reloc_addr = value;
|
||||
}
|
||||
else
|
||||
__libc_fatal ("Unexpected reloc type in static binary.\n");
|
||||
}
|
||||
|
||||
#endif
|
@ -25,6 +25,7 @@
|
||||
#include <elf/elf.h>
|
||||
#include <sys/asm.h>
|
||||
#include <dl-tls.h>
|
||||
#include <dl-irel.h>
|
||||
|
||||
#ifndef _RTLD_PROLOGUE
|
||||
# define _RTLD_PROLOGUE(entry) \
|
||||
@ -176,6 +177,13 @@ elf_machine_rela (struct link_map *map, const ElfW(Rela) *reloc,
|
||||
if (sym_map != NULL)
|
||||
value = SYMBOL_ADDRESS (sym_map, sym, true) + reloc->r_addend;
|
||||
|
||||
if (sym != NULL
|
||||
&& __glibc_unlikely (ELFW(ST_TYPE) (sym->st_info) == STT_GNU_IFUNC)
|
||||
&& __glibc_likely (sym->st_shndx != SHN_UNDEF)
|
||||
&& __glibc_likely (!skip_ifunc))
|
||||
value = elf_ifunc_invoke (value);
|
||||
|
||||
|
||||
switch (r_type)
|
||||
{
|
||||
#ifndef RTLD_BOOTSTRAP
|
||||
@ -251,6 +259,13 @@ elf_machine_rela (struct link_map *map, const ElfW(Rela) *reloc,
|
||||
}
|
||||
#endif
|
||||
|
||||
case R_RISCV_IRELATIVE:
|
||||
value = map->l_addr + reloc->r_addend;
|
||||
if (__glibc_likely (!skip_ifunc))
|
||||
value = elf_ifunc_invoke (value);
|
||||
*addr_field = value;
|
||||
break;
|
||||
|
||||
case R_RISCV_JUMP_SLOT:
|
||||
case __WORDSIZE == 64 ? R_RISCV_64 : R_RISCV_32:
|
||||
*addr_field = value;
|
||||
@ -292,6 +307,13 @@ elf_machine_lazy_rel (struct link_map *map, ElfW(Addr) l_addr,
|
||||
else
|
||||
*reloc_addr = map->l_mach.plt;
|
||||
}
|
||||
else if (__glibc_unlikely (r_type == R_RISCV_IRELATIVE))
|
||||
{
|
||||
ElfW(Addr) value = map->l_addr + reloc->r_addend;
|
||||
if (__glibc_likely (!skip_ifunc))
|
||||
value = elf_ifunc_invoke (value);
|
||||
*reloc_addr = value;
|
||||
}
|
||||
else
|
||||
_dl_reloc_bad_type (map, r_type, 1);
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user