| xf.li | bdd93d5 | 2023-05-12 07:10:14 -0700 | [diff] [blame] | 1 | /* Machine-dependent ELF dynamic relocation inline functions.  Stub version. | 
|  | 2 | Copyright (C) 1995-2016 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 |