lh | 9ed821d | 2023-04-07 01:36:19 -0700 | [diff] [blame] | 1 | /* Machine-dependent ELF dynamic relocation inline functions. Stub version. |
| 2 | Copyright (C) 1995-2015 Free Software Foundation, Inc. |
| 3 | This file is part of the GNU C Library. |
| 4 | |
| 5 | The GNU C Library is free software; you can redistribute it and/or |
| 6 | modify it under the terms of the GNU Lesser General Public |
| 7 | License as published by the Free Software Foundation; either |
| 8 | version 2.1 of the License, or (at your option) any later version. |
| 9 | |
| 10 | The GNU C Library is distributed in the hope that it will be useful, |
| 11 | but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 12 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 13 | Lesser General Public License for more details. |
| 14 | |
| 15 | You should have received a copy of the GNU Lesser General Public |
| 16 | License along with the GNU C Library; if not, see |
| 17 | <http://www.gnu.org/licenses/>. */ |
| 18 | |
| 19 | #define ELF_MACHINE_NAME "stub" |
| 20 | |
| 21 | #include <string.h> |
| 22 | #include <link.h> |
| 23 | |
| 24 | |
| 25 | /* Return nonzero iff ELF header is compatible with the running host. */ |
| 26 | static inline int |
| 27 | elf_machine_matches_host (const Elf32_Ehdr *ehdr) |
| 28 | { |
| 29 | switch (ehdr->e_machine) |
| 30 | { |
| 31 | default: |
| 32 | return 0; |
| 33 | } |
| 34 | } |
| 35 | |
| 36 | |
| 37 | /* Return the link-time address of _DYNAMIC. */ |
| 38 | static inline Elf32_Addr |
| 39 | elf_machine_dynamic (void) |
| 40 | { |
| 41 | #error "Damn, no _DYNAMIC" |
| 42 | } |
| 43 | |
| 44 | |
| 45 | /* Return the run-time load address of the shared object. */ |
| 46 | static inline Elf32_Addr |
| 47 | elf_machine_load_address (void) |
| 48 | { |
| 49 | #error "Where am I?" |
| 50 | } |
| 51 | |
| 52 | /* Fixup a PLT entry to bounce directly to the function at VALUE. */ |
| 53 | |
| 54 | static inline Elf32_Addr |
| 55 | elf_machine_fixup_plt (struct link_map *map, lookup_t t, |
| 56 | const Elf32_Rel *reloc, |
| 57 | Elf32_Addr *reloc_addr, Elf32_Addr value) |
| 58 | { |
| 59 | return *reloc_addr = value; |
| 60 | } |
| 61 | |
| 62 | /* Perform the relocation specified by RELOC and SYM (which is fully resolved). |
| 63 | LOADADDR is the load address of the object; INFO is an array indexed |
| 64 | by DT_* of the .dynamic section info. */ |
| 65 | |
| 66 | auto inline void |
| 67 | __attribute__ ((always_inline)) |
| 68 | elf_machine_rel (Elf32_Addr loadaddr, Elf32_Dyn *info[DT_NUM], |
| 69 | const Elf32_Rel *reloc, const Elf32_Sym *sym, |
| 70 | Elf32_Addr (*resolve) (const Elf32_Sym **ref, |
| 71 | Elf32_Addr reloc_addr, |
| 72 | int noplt)) |
| 73 | { |
| 74 | Elf32_Addr *const reloc_addr = (Elf32_Addr *) reloc->r_offset; |
| 75 | Elf32_Addr loadbase; |
| 76 | |
| 77 | switch (ELF32_R_TYPE (reloc->r_info)) |
| 78 | { |
| 79 | case R_MACHINE_COPY: |
| 80 | loadbase = (*resolve) (&sym, (Elf32_Addr) reloc_addr, 0); |
| 81 | memcpy (reloc_addr, (void *) (loadbase + sym->st_value), sym->st_size); |
| 82 | break; |
| 83 | default: |
| 84 | _dl_reloc_bad_type (map, ELF32_R_TYPE (reloc->r_info), 0); |
| 85 | break; |
| 86 | } |
| 87 | } |
| 88 | |
| 89 | |
| 90 | auto inline Elf32_Addr |
| 91 | __attribute__ ((always_inline)) |
| 92 | elf_machine_rela (Elf32_Addr loadaddr, Elf32_Dyn *info[DT_NUM], |
| 93 | const Elf32_Rel *reloc, const Elf32_Sym *sym, |
| 94 | Elf32_Addr (*resolve) (const Elf32_Sym **ref, |
| 95 | Elf32_Addr reloc_addr, |
| 96 | int noplt)) |
| 97 | { |
| 98 | _dl_signal_error (0, "Elf32_Rela relocation requested -- unused on " |
| 99 | NULL, ELF_MACHINE_NAME); |
| 100 | } |
| 101 | |
| 102 | |
| 103 | /* Set up the loaded object described by L so its unrelocated PLT |
| 104 | entries will jump to the on-demand fixup code in dl-runtime.c. */ |
| 105 | |
| 106 | static inline int |
| 107 | elf_machine_runtime_setup (struct link_map *l, int lazy) |
| 108 | { |
| 109 | extern void _dl_runtime_resolve (Elf32_Word); |
| 110 | |
| 111 | if (lazy) |
| 112 | { |
| 113 | /* The GOT entries for functions in the PLT have not yet been filled |
| 114 | in. Their initial contents will arrange when called to push an |
| 115 | offset into the .rel.plt section, push _GLOBAL_OFFSET_TABLE_[1], |
| 116 | and then jump to _GLOBAL_OFFSET_TABLE[2]. */ |
| 117 | Elf32_Addr *got = (Elf32_Addr *) D_PTR (l, l_info[DT_PLTGOT]); |
| 118 | got[1] = (Elf32_Addr) l; /* Identify this shared object. */ |
| 119 | |
| 120 | /* This function will get called to fix up the GOT entry indicated by |
| 121 | the offset on the stack, and then jump to the resolved address. */ |
| 122 | got[2] = (Elf32_Addr) &_dl_runtime_resolve; |
| 123 | } |
| 124 | |
| 125 | return lazy; |
| 126 | } |
| 127 | |
| 128 | |
| 129 | /* Initial entry point code for the dynamic linker. |
| 130 | The C function `_dl_start' is the real entry point; |
| 131 | its return value is the user program's entry point. */ |
| 132 | |
| 133 | #define RTLD_START #error need some startup code |