1 /* vi: set sw=4 ts=4: */
2 /* mips/mipsel ELF shared library loader suppport
4 Copyright (C) 2002, Steven J. Hill (sjhill@realitydiluted.com)
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. The name of the above contributors may not be
14 * used to endorse or promote products derived from this software
15 * without specific prior written permission.
17 * THIS SOFTWARE IS PROVIDED BY THE CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 extern int _dl_runtime_resolve(void);
34 #define OFFSET_GP_GOT 0x7ff0
36 unsigned long __dl_runtime_resolve(unsigned long sym_index,
37 unsigned long old_gpreg)
39 unsigned long *got = (unsigned long *) (old_gpreg - OFFSET_GP_GOT);
40 struct elf_resolve *tpnt = (struct elf_resolve *) got[1];
43 unsigned long local_gotno;
45 unsigned long new_addr;
46 unsigned long instr_addr;
50 gotsym = tpnt->dynamic_info[DT_MIPS_GOTSYM_IDX];
51 local_gotno = tpnt->dynamic_info[DT_MIPS_LOCAL_GOTNO_IDX];
53 sym = ((Elf32_Sym *) tpnt->dynamic_info[DT_SYMTAB]) + sym_index;
54 strtab = (char *) tpnt->dynamic_info[DT_STRTAB];
55 symname = strtab + sym->st_name;
57 new_addr = (unsigned long) _dl_find_hash(symname,
58 tpnt->symbol_scope, tpnt, ELF_RTYPE_CLASS_PLT);
59 if (unlikely(!new_addr)) {
60 _dl_dprintf (2, "%s: can't resolve symbol '%s'\n",
61 _dl_progname, symname);
65 /* Address of jump instruction to fix up */
66 instr_addr = (unsigned long) (got + local_gotno + sym_index - gotsym);
67 got_addr = (char **) instr_addr;
69 #if defined (__SUPPORT_LD_DEBUG__)
70 if (_dl_debug_bindings)
72 _dl_dprintf(_dl_debug_file, "\nresolve function: %s", symname);
73 if(_dl_debug_detail) _dl_dprintf(_dl_debug_file,
74 "\n\tpatched %x ==> %x @ %x\n", *got_addr, new_addr, got_addr);
76 if (!_dl_debug_nofixups) {
77 *got_addr = (char*)new_addr;
80 *got_addr = (char*)new_addr;
86 void _dl_parse_lazy_relocation_information(struct dyn_elf *rpnt,
87 unsigned long rel_addr, unsigned long rel_size)
93 int _dl_parse_relocation_information(struct dyn_elf *xpnt,
94 unsigned long rel_addr, unsigned long rel_size)
101 unsigned long *reloc_addr=NULL;
102 unsigned long symbol_addr;
103 int reloc_type, symtab_index;
104 struct elf_resolve *tpnt = xpnt->dyn;
105 #if defined (__SUPPORT_LD_DEBUG__)
106 unsigned long old_val=0;
109 /* Now parse the relocation information */
110 rel_size = rel_size / sizeof(Elf32_Rel);
111 rpnt = (Elf32_Rel *) rel_addr;
113 symtab = (Elf32_Sym *) tpnt->dynamic_info[DT_SYMTAB];
114 strtab = (char *) tpnt->dynamic_info[DT_STRTAB];
115 got = (unsigned long *) tpnt->dynamic_info[DT_PLTGOT];
117 for (i = 0; i < rel_size; i++, rpnt++) {
118 reloc_addr = (unsigned long *) (tpnt->loadaddr +
119 (unsigned long) rpnt->r_offset);
120 reloc_type = ELF32_R_TYPE(rpnt->r_info);
121 symtab_index = ELF32_R_SYM(rpnt->r_info);
124 #if defined (__SUPPORT_LD_DEBUG__)
125 debug_sym(symtab,strtab,symtab_index);
126 debug_reloc(symtab,strtab,rpnt);
128 old_val = *reloc_addr;
131 switch (reloc_type) {
134 if (symtab_index < tpnt->dynamic_info[DT_MIPS_GOTSYM_IDX])
136 symtab[symtab_index].st_value +
137 (unsigned long) tpnt->loadaddr;
139 *reloc_addr += got[symtab_index + tpnt->dynamic_info[DT_MIPS_LOCAL_GOTNO_IDX] -
140 tpnt->dynamic_info[DT_MIPS_GOTSYM_IDX]];
144 *reloc_addr += (unsigned long) tpnt->loadaddr;
151 int reloc_type = ELF32_R_TYPE(rpnt->r_info);
152 _dl_dprintf(2, "\n%s: ",_dl_progname);
155 _dl_dprintf(2, "symbol '%s': ", strtab + symtab[symtab_index].st_name);
157 #if defined (__SUPPORT_LD_DEBUG__)
158 _dl_dprintf(2, "can't handle reloc type %s\n ", _dl_reltypes(reloc_type));
160 _dl_dprintf(2, "can't handle reloc type %x\n", reloc_type);
167 #if defined (__SUPPORT_LD_DEBUG__)
168 if(_dl_debug_reloc && _dl_debug_detail && reloc_addr)
169 _dl_dprintf(_dl_debug_file, "\tpatched: %x ==> %x @ %x\n", old_val, *reloc_addr, reloc_addr);
175 /* Relocate the global GOT entries for the object */
176 void _dl_perform_mips_global_got_relocations(struct elf_resolve *tpnt, int lazy)
180 unsigned long i, tmp_lazy;
181 unsigned long *got_entry;
183 for (; tpnt ; tpnt = tpnt->next) {
185 /* We don't touch the dynamic linker */
186 if (tpnt->libtype == program_interpreter)
189 /* Setup the loop variables */
190 got_entry = (unsigned long *) (tpnt->dynamic_info[DT_PLTGOT])
191 + tpnt->dynamic_info[DT_MIPS_LOCAL_GOTNO_IDX];
192 sym = (Elf32_Sym *) tpnt->dynamic_info[DT_SYMTAB] + tpnt->dynamic_info[DT_MIPS_GOTSYM_IDX];
193 strtab = (char *) tpnt->dynamic_info[DT_STRTAB];
194 i = tpnt->dynamic_info[DT_MIPS_SYMTABNO_IDX] - tpnt->dynamic_info[DT_MIPS_GOTSYM_IDX];
196 #if defined (__SUPPORT_LD_DEBUG__)
198 _dl_dprintf(2, "_dl_perform_mips_global_got_relocations for '%s'\n", tpnt->libname);
200 tmp_lazy = lazy && !tpnt->dynamic_info[DT_BIND_NOW];
201 /* Relocate the global GOT entries for the object */
203 if (sym->st_shndx == SHN_UNDEF) {
204 if (ELF32_ST_TYPE(sym->st_info) == STT_FUNC && sym->st_value && tmp_lazy) {
205 *got_entry = sym->st_value + (unsigned long) tpnt->loadaddr;
208 *got_entry = (unsigned long) _dl_find_hash(strtab +
209 sym->st_name, tpnt->symbol_scope, tpnt, ELF_RTYPE_CLASS_PLT);
212 else if (sym->st_shndx == SHN_COMMON) {
213 *got_entry = (unsigned long) _dl_find_hash(strtab +
214 sym->st_name, tpnt->symbol_scope, tpnt, ELF_RTYPE_CLASS_PLT);
216 else if (ELF32_ST_TYPE(sym->st_info) == STT_FUNC &&
217 *got_entry != sym->st_value && tmp_lazy) {
218 *got_entry += (unsigned long) tpnt->loadaddr;
220 else if (ELF32_ST_TYPE(sym->st_info) == STT_SECTION) {
221 if (sym->st_other == 0)
222 *got_entry += (unsigned long) tpnt->loadaddr;
225 *got_entry = (unsigned long) _dl_find_hash(strtab +
226 sym->st_name, tpnt->symbol_scope, tpnt, ELF_RTYPE_CLASS_PLT);