| 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 |