OSDN Git Service

fix ld.so.cache handling on no-mmu setups
[uclinux-h8/uclibc-ng.git] / ldso / ldso / dl-elf.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * This file contains the helper routines to load an ELF shared
4  * library into memory and add the symbol table info to the chain.
5  *
6  * Copyright (C) 2000-2006 by Erik Andersen <andersen@codepoet.org>
7  * Copyright (c) 1994-2000 Eric Youngdale, Peter MacDonald,
8  *                              David Engel, Hongjiu Lu and Mitch D'Souza
9  *
10  * Redistribution and use in source and binary forms, with or without
11  * modification, are permitted provided that the following conditions
12  * are met:
13  * 1. Redistributions of source code must retain the above copyright
14  *    notice, this list of conditions and the following disclaimer.
15  * 2. The name of the above contributors may not be
16  *    used to endorse or promote products derived from this software
17  *    without specific prior written permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE CONTRIBUTORS ``AS IS'' AND
20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22  * ARE DISCLAIMED.  IN NO EVENT SHALL THE CONTRIBUTORS BE LIABLE
23  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29  * SUCH DAMAGE.
30  */
31
32
33 #include "ldso.h"
34
35 #ifdef __LDSO_CACHE_SUPPORT__
36
37 static caddr_t _dl_cache_addr = NULL;
38 static size_t _dl_cache_size = 0;
39
40 int _dl_map_cache(void)
41 {
42         int fd;
43         struct stat st;
44         header_t *header;
45         libentry_t *libent;
46         int i, strtabsize;
47
48         if (_dl_cache_addr == (caddr_t) - 1)
49                 return -1;
50         else if (_dl_cache_addr != NULL)
51                 return 0;
52
53         if (_dl_stat(LDSO_CACHE, &st)
54             || (fd = _dl_open(LDSO_CACHE, O_RDONLY, 0)) < 0) {
55                 _dl_cache_addr = (caddr_t) - 1; /* so we won't try again */
56                 return -1;
57         }
58
59         _dl_cache_size = st.st_size;
60         _dl_cache_addr = _dl_mmap(0, _dl_cache_size, PROT_READ, LDSO_CACHE_MMAP_FLAGS, fd, 0);
61         _dl_close(fd);
62         if (_dl_mmap_check_error(_dl_cache_addr)) {
63                 _dl_dprintf(2, "%s:%i: can't map '%s'\n",
64                                 _dl_progname, __LINE__, LDSO_CACHE);
65                 return -1;
66         }
67
68         header = (header_t *) _dl_cache_addr;
69
70         if (_dl_cache_size < sizeof(header_t) ||
71                         _dl_memcmp(header->magic, LDSO_CACHE_MAGIC, LDSO_CACHE_MAGIC_LEN)
72                         || _dl_memcmp(header->version, LDSO_CACHE_VER, LDSO_CACHE_VER_LEN)
73                         || _dl_cache_size <
74                         (sizeof(header_t) + header->nlibs * sizeof(libentry_t))
75                         || _dl_cache_addr[_dl_cache_size - 1] != '\0')
76         {
77                 _dl_dprintf(2, "%s: cache '%s' is corrupt\n", _dl_progname,
78                                 LDSO_CACHE);
79                 goto fail;
80         }
81
82         strtabsize = _dl_cache_size - sizeof(header_t) -
83                 header->nlibs * sizeof(libentry_t);
84         libent = (libentry_t *) & header[1];
85
86         for (i = 0; i < header->nlibs; i++) {
87                 if (libent[i].sooffset >= strtabsize ||
88                                 libent[i].liboffset >= strtabsize)
89                 {
90                         _dl_dprintf(2, "%s: cache '%s' is corrupt\n", _dl_progname, LDSO_CACHE);
91                         goto fail;
92                 }
93         }
94
95         return 0;
96
97 fail:
98         _dl_munmap(_dl_cache_addr, _dl_cache_size);
99         _dl_cache_addr = (caddr_t) - 1;
100         return -1;
101 }
102
103 int _dl_unmap_cache(void)
104 {
105         if (_dl_cache_addr == NULL || _dl_cache_addr == (caddr_t) - 1)
106                 return -1;
107
108 #if 1
109         _dl_munmap(_dl_cache_addr, _dl_cache_size);
110         _dl_cache_addr = NULL;
111 #endif
112
113         return 0;
114 }
115 #endif
116
117
118 void
119 _dl_protect_relro (struct elf_resolve *l)
120 {
121         ElfW(Addr) base = (ElfW(Addr)) DL_RELOC_ADDR(l->loadaddr, l->relro_addr);
122         ElfW(Addr) start = (base & ~(_dl_pagesize - 1));
123         ElfW(Addr) end = ((base + l->relro_size) & ~(_dl_pagesize - 1));
124         _dl_if_debug_dprint("RELRO protecting %s:  start:%x, end:%x\n", l->libname, start, end);
125         if (start != end &&
126             _dl_mprotect ((void *) start, end - start, PROT_READ) < 0) {
127                 _dl_dprintf(2, "%s: cannot apply additional memory protection after relocation", l->libname);
128                 _dl_exit(0);
129         }
130 }
131
132 /* This function's behavior must exactly match that
133  * in uClibc/ldso/util/ldd.c */
134 static struct elf_resolve *
135 search_for_named_library(const char *name, int secure, const char *path_list,
136         struct dyn_elf **rpnt)
137 {
138         char *path, *path_n, *mylibname;
139         struct elf_resolve *tpnt;
140         int done;
141
142         if (path_list==NULL)
143                 return NULL;
144
145         /* We need a writable copy of this string, but we don't
146          * need this allocated permanently since we don't want
147          * to leak memory, so use alloca to put path on the stack */
148         done = _dl_strlen(path_list);
149         path = alloca(done + 1);
150
151         /* another bit of local storage */
152         mylibname = alloca(2050);
153
154         /* gcc inlines alloca using a single instruction adjusting
155          * the stack pointer and no stack overflow check and thus
156          * no NULL error return.  No point leaving in dead code... */
157 #if 0
158         if (!path || !mylibname) {
159                 _dl_dprintf(2, "Out of memory!\n");
160                 _dl_exit(0);
161         }
162 #endif
163
164         _dl_memcpy(path, path_list, done+1);
165
166         /* Unlike ldd.c, don't bother to eliminate double //s */
167
168         /* Replace colons with zeros in path_list */
169         /* : at the beginning or end of path maps to CWD */
170         /* :: anywhere maps CWD */
171         /* "" maps to CWD */
172         done = 0;
173         path_n = path;
174         do {
175                 if (*path == 0) {
176                         *path = ':';
177                         done = 1;
178                 }
179                 if (*path == ':') {
180                         *path = 0;
181                         if (*path_n)
182                                 _dl_strcpy(mylibname, path_n);
183                         else
184                                 _dl_strcpy(mylibname, "."); /* Assume current dir if empty path */
185                         _dl_strcat(mylibname, "/");
186                         _dl_strcat(mylibname, name);
187                         if ((tpnt = _dl_load_elf_shared_library(secure, rpnt, mylibname)) != NULL)
188                                 return tpnt;
189                         path_n = path+1;
190                 }
191                 path++;
192         } while (!done);
193         return NULL;
194 }
195
196 /* Used to return error codes back to dlopen et. al.  */
197 unsigned long _dl_error_number;
198 unsigned long _dl_internal_error_number;
199
200 struct elf_resolve *_dl_load_shared_library(int secure, struct dyn_elf **rpnt,
201         struct elf_resolve *tpnt, char *full_libname, int __attribute__((unused)) trace_loaded_objects)
202 {
203         char *pnt;
204         struct elf_resolve *tpnt1;
205         char *libname;
206
207         _dl_internal_error_number = 0;
208         libname = full_libname;
209
210         /* quick hack to ensure mylibname buffer doesn't overflow.  don't
211            allow full_libname or any directory to be longer than 1024. */
212         if (_dl_strlen(full_libname) > 1024)
213                 goto goof;
214
215         /* Skip over any initial initial './' and '/' stuff to
216          * get the short form libname with no path garbage */
217         pnt = _dl_strrchr(libname, '/');
218         if (pnt) {
219                 libname = pnt + 1;
220         }
221
222         _dl_if_debug_dprint("\tfind library='%s'; searching\n", libname);
223         /* If the filename has any '/', try it straight and leave it at that.
224            For IBCS2 compatibility under linux, we substitute the string
225            /usr/i486-sysv4/lib for /usr/lib in library names. */
226
227         if (libname != full_libname) {
228                 _dl_if_debug_dprint("\ttrying file='%s'\n", full_libname);
229                 tpnt1 = _dl_load_elf_shared_library(secure, rpnt, full_libname);
230                 if (tpnt1) {
231                         return tpnt1;
232                 }
233         }
234
235         /*
236          * The ABI specifies that RPATH is searched before LD_LIBRARY_PATH or
237          * the default path of /usr/lib.  Check in rpath directories.
238          */
239 #ifdef __LDSO_RUNPATH__
240         pnt = (tpnt ? (char *) tpnt->dynamic_info[DT_RPATH] : NULL);
241         if (pnt) {
242                 pnt += (unsigned long) tpnt->dynamic_info[DT_STRTAB];
243                 _dl_if_debug_dprint("\tsearching RPATH='%s'\n", pnt);
244                 if ((tpnt1 = search_for_named_library(libname, secure, pnt, rpnt)) != NULL)
245                         return tpnt1;
246         }
247 #endif
248
249         /* Check in LD_{ELF_}LIBRARY_PATH, if specified and allowed */
250         if (_dl_library_path) {
251                 _dl_if_debug_dprint("\tsearching LD_LIBRARY_PATH='%s'\n", _dl_library_path);
252                 if ((tpnt1 = search_for_named_library(libname, secure, _dl_library_path, rpnt)) != NULL)
253                 {
254                         return tpnt1;
255                 }
256         }
257
258         /*
259          * The ABI specifies that RUNPATH is searched after LD_LIBRARY_PATH.
260          */
261 #ifdef __LDSO_RUNPATH__
262         pnt = (tpnt ? (char *)tpnt->dynamic_info[DT_RUNPATH] : NULL);
263         if (pnt) {
264                 pnt += (unsigned long) tpnt->dynamic_info[DT_STRTAB];
265                 _dl_if_debug_dprint("\tsearching RUNPATH='%s'\n", pnt);
266                 if ((tpnt1 = search_for_named_library(libname, secure, pnt, rpnt)) != NULL)
267                         return tpnt1;
268         }
269 #endif
270
271         /*
272          * Where should the cache be searched?  There is no such concept in the
273          * ABI, so we have some flexibility here.  For now, search it before
274          * the hard coded paths that follow (i.e before /lib and /usr/lib).
275          */
276 #ifdef __LDSO_CACHE_SUPPORT__
277         if (_dl_cache_addr != NULL && _dl_cache_addr != (caddr_t) - 1) {
278                 int i;
279                 header_t *header = (header_t *) _dl_cache_addr;
280                 libentry_t *libent = (libentry_t *) & header[1];
281                 char *strs = (char *) &libent[header->nlibs];
282
283                 _dl_if_debug_dprint("\tsearching cache='%s'\n", LDSO_CACHE);
284                 for (i = 0; i < header->nlibs; i++) {
285                         if ((libent[i].flags == LIB_ELF ||
286                                                 libent[i].flags == LIB_ELF_LIBC0 ||
287                                                 libent[i].flags == LIB_ELF_LIBC5) &&
288                                         _dl_strcmp(libname, strs + libent[i].sooffset) == 0 &&
289                                         (tpnt1 = _dl_load_elf_shared_library(secure,
290                                                                                                                  rpnt, strs + libent[i].liboffset)))
291                                 return tpnt1;
292                 }
293         }
294 #endif
295
296         /* Look for libraries wherever the shared library loader
297          * was installed */
298         _dl_if_debug_dprint("\tsearching ldso dir='%s'\n", _dl_ldsopath);
299         if ((tpnt1 = search_for_named_library(libname, secure, _dl_ldsopath, rpnt)) != NULL)
300         {
301                 return tpnt1;
302         }
303
304
305         /* Lastly, search the standard list of paths for the library.
306            This list must exactly match the list in uClibc/ldso/util/ldd.c */
307         _dl_if_debug_dprint("\tsearching full lib path list\n");
308         if ((tpnt1 = search_for_named_library(libname, secure,
309                                         UCLIBC_RUNTIME_PREFIX "lib:"
310                                         UCLIBC_RUNTIME_PREFIX "usr/lib"
311 #ifndef __LDSO_CACHE_SUPPORT__
312                                         ":" UCLIBC_RUNTIME_PREFIX "usr/X11R6/lib"
313 #endif
314                                         , rpnt)
315                 ) != NULL)
316         {
317                 return tpnt1;
318         }
319
320 goof:
321         /* Well, we shot our wad on that one.  All we can do now is punt */
322         if (_dl_internal_error_number)
323                 _dl_error_number = _dl_internal_error_number;
324         else
325                 _dl_error_number = LD_ERROR_NOFILE;
326         _dl_if_debug_dprint("Bummer: could not find '%s'!\n", libname);
327         return NULL;
328 }
329
330
331 /*
332  * Read one ELF library into memory, mmap it into the correct locations and
333  * add the symbol info to the symbol chain.  Perform any relocations that
334  * are required.
335  */
336
337 struct elf_resolve *_dl_load_elf_shared_library(int secure,
338         struct dyn_elf **rpnt, char *libname)
339 {
340         ElfW(Ehdr) *epnt;
341         unsigned long dynamic_addr = 0;
342         ElfW(Dyn) *dpnt;
343         struct elf_resolve *tpnt;
344         ElfW(Phdr) *ppnt;
345         char *status, *header;
346         unsigned long dynamic_info[DYNAMIC_SIZE];
347         unsigned long *lpnt;
348         unsigned long libaddr;
349         unsigned long minvma = 0xffffffff, maxvma = 0;
350         int i, flags, piclib, infile;
351         ElfW(Addr) relro_addr = 0;
352         size_t relro_size = 0;
353         struct stat st;
354         DL_LOADADDR_TYPE lib_loadaddr;
355         DL_INIT_LOADADDR_EXTRA_DECLS
356
357         infile = _dl_open(libname, O_RDONLY, 0);
358         if (infile < 0) {
359                 _dl_internal_error_number = LD_ERROR_NOFILE;
360                 return NULL;
361         }
362
363         if (_dl_fstat(infile, &st) < 0) {
364                 _dl_internal_error_number = LD_ERROR_NOFILE;
365                 _dl_close(infile);
366                 return NULL;
367         }
368         /* If we are in secure mode (i.e. a setu/gid binary using LD_PRELOAD),
369            we don't load the library if it isn't setuid. */
370         if (secure)
371                 if (!(st.st_mode & S_ISUID)) {
372                         _dl_close(infile);
373                         return NULL;
374                 }
375
376         /* Check if file is already loaded */
377         for (tpnt = _dl_loaded_modules; tpnt; tpnt = tpnt->next) {
378                 if (tpnt->st_dev == st.st_dev && tpnt->st_ino == st.st_ino) {
379                         /* Already loaded */
380                         tpnt->usage_count++;
381                         _dl_close(infile);
382                         return tpnt;
383                 }
384         }
385         header = _dl_mmap((void *) 0, _dl_pagesize, PROT_READ | PROT_WRITE,
386                         MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
387         if (_dl_mmap_check_error(header)) {
388                 _dl_dprintf(2, "%s:%i: can't map '%s'\n", _dl_progname, __LINE__, libname);
389                 _dl_internal_error_number = LD_ERROR_MMAP_FAILED;
390                 _dl_close(infile);
391                 return NULL;
392         }
393
394         _dl_read(infile, header, _dl_pagesize);
395         epnt = (ElfW(Ehdr) *) (intptr_t) header;
396         if (epnt->e_ident[0] != 0x7f ||
397                         epnt->e_ident[1] != 'E' ||
398                         epnt->e_ident[2] != 'L' ||
399                         epnt->e_ident[3] != 'F')
400         {
401                 _dl_dprintf(2, "%s: '%s' is not an ELF file\n", _dl_progname,
402                                 libname);
403                 _dl_internal_error_number = LD_ERROR_NOTELF;
404                 _dl_close(infile);
405                 _dl_munmap(header, _dl_pagesize);
406                 return NULL;
407         }
408
409         if ((epnt->e_type != ET_DYN) || (epnt->e_machine != MAGIC1
410 #ifdef MAGIC2
411                                 && epnt->e_machine != MAGIC2
412 #endif
413                                 ))
414         {
415                 _dl_internal_error_number =
416                         (epnt->e_type != ET_DYN ? LD_ERROR_NOTDYN : LD_ERROR_NOTMAGIC);
417                 _dl_dprintf(2, "%s: '%s' is not an ELF executable for " ELF_TARGET
418                                 "\n", _dl_progname, libname);
419                 _dl_close(infile);
420                 _dl_munmap(header, _dl_pagesize);
421                 return NULL;
422         }
423
424         ppnt = (ElfW(Phdr) *)(intptr_t) & header[epnt->e_phoff];
425
426         piclib = 1;
427         for (i = 0; i < epnt->e_phnum; i++) {
428
429                 if (ppnt->p_type == PT_DYNAMIC) {
430                         if (dynamic_addr)
431                                 _dl_dprintf(2, "%s: '%s' has more than one dynamic section\n",
432                                                 _dl_progname, libname);
433                         dynamic_addr = ppnt->p_vaddr;
434                 }
435
436                 if (ppnt->p_type == PT_LOAD) {
437                         /* See if this is a PIC library. */
438                         if (i == 0 && ppnt->p_vaddr > 0x1000000) {
439                                 piclib = 0;
440                                 minvma = ppnt->p_vaddr;
441                         }
442                         if (piclib && ppnt->p_vaddr < minvma) {
443                                 minvma = ppnt->p_vaddr;
444                         }
445                         if (((unsigned long) ppnt->p_vaddr + ppnt->p_memsz) > maxvma) {
446                                 maxvma = ppnt->p_vaddr + ppnt->p_memsz;
447                         }
448                 }
449                 ppnt++;
450         }
451
452         maxvma = (maxvma + ADDR_ALIGN) & ~ADDR_ALIGN;
453         minvma = minvma & ~0xffffU;
454
455         flags = MAP_PRIVATE /*| MAP_DENYWRITE */ ;
456         if (!piclib)
457                 flags |= MAP_FIXED;
458
459         status = (char *) _dl_mmap((char *) (piclib ? 0 : minvma),
460                         maxvma - minvma, PROT_NONE, flags | MAP_ANONYMOUS, -1, 0);
461         if (_dl_mmap_check_error(status)) {
462                 _dl_dprintf(2, "%s:%i: can't map '%s'\n", _dl_progname, __LINE__, libname);
463                 _dl_internal_error_number = LD_ERROR_MMAP_FAILED;
464                 _dl_close(infile);
465                 _dl_munmap(header, _dl_pagesize);
466                 return NULL;
467         }
468         libaddr = (unsigned long) status;
469         flags |= MAP_FIXED;
470
471         /* Get the memory to store the library */
472         ppnt = (ElfW(Phdr) *)(intptr_t) & header[epnt->e_phoff];
473
474         DL_INIT_LOADADDR(lib_loadaddr, libaddr, ppnt, epnt->e_phnum);
475
476         for (i = 0; i < epnt->e_phnum; i++) {
477                 if (ppnt->p_type == PT_GNU_RELRO) {
478                         relro_addr = ppnt->p_vaddr;
479                         relro_size = ppnt->p_memsz;
480                 }
481                 if (ppnt->p_type == PT_LOAD) {
482
483                         /* See if this is a PIC library. */
484                         if (i == 0 && ppnt->p_vaddr > 0x1000000) {
485                                 piclib = 0;
486                                 /* flags |= MAP_FIXED; */
487                         }
488
489                         if (ppnt->p_flags & PF_W) {
490                                 unsigned long map_size;
491                                 char *cpnt;
492
493                                 status = (char *) _dl_mmap((char *) ((piclib ? libaddr : 0) +
494                                                         (ppnt->p_vaddr & PAGE_ALIGN)), (ppnt->p_vaddr & ADDR_ALIGN)
495                                                 + ppnt->p_filesz, LXFLAGS(ppnt->p_flags), flags, infile,
496                                                 ppnt->p_offset & OFFS_ALIGN);
497
498                                 if (_dl_mmap_check_error(status)) {
499                                         _dl_dprintf(2, "%s:%i: can't map '%s'\n",
500                                                         _dl_progname, __LINE__, libname);
501                                         _dl_internal_error_number = LD_ERROR_MMAP_FAILED;
502                                         _dl_munmap((char *) libaddr, maxvma - minvma);
503                                         _dl_close(infile);
504                                         _dl_munmap(header, _dl_pagesize);
505                                         return NULL;
506                                 }
507
508                                 /* Pad the last page with zeroes. */
509                                 cpnt = (char *) (status + (ppnt->p_vaddr & ADDR_ALIGN) +
510                                                 ppnt->p_filesz);
511                                 while (((unsigned long) cpnt) & ADDR_ALIGN)
512                                         *cpnt++ = 0;
513
514                                 /* I am not quite sure if this is completely
515                                  * correct to do or not, but the basic way that
516                                  * we handle bss segments is that we mmap
517                                  * /dev/zero if there are any pages left over
518                                  * that are not mapped as part of the file */
519
520                                 map_size = (ppnt->p_vaddr + ppnt->p_filesz + ADDR_ALIGN) & PAGE_ALIGN;
521
522                                 if (map_size < ppnt->p_vaddr + ppnt->p_memsz)
523                                         status = (char *) _dl_mmap((char *) map_size +
524                                                         (piclib ? libaddr : 0),
525                                                         ppnt->p_vaddr + ppnt->p_memsz - map_size,
526                                                         LXFLAGS(ppnt->p_flags), flags | MAP_ANONYMOUS, -1, 0);
527                         } else
528                                 status = (char *) _dl_mmap((char *) (ppnt->p_vaddr & PAGE_ALIGN)
529                                                 + (piclib ? libaddr : 0), (ppnt->p_vaddr & ADDR_ALIGN) +
530                                                 ppnt->p_filesz, LXFLAGS(ppnt->p_flags), flags,
531                                                 infile, ppnt->p_offset & OFFS_ALIGN);
532                         if (_dl_mmap_check_error(status)) {
533                                 _dl_dprintf(2, "%s:%i: can't map '%s'\n", _dl_progname, __LINE__, libname);
534                                 _dl_internal_error_number = LD_ERROR_MMAP_FAILED;
535                                 _dl_munmap((char *) libaddr, maxvma - minvma);
536                                 _dl_close(infile);
537                                 _dl_munmap(header, _dl_pagesize);
538                                 return NULL;
539                         }
540
541                         /* if (libaddr == 0 && piclib) {
542                            libaddr = (unsigned long) status;
543                            flags |= MAP_FIXED;
544                            } */
545                 }
546                 ppnt++;
547         }
548         _dl_close(infile);
549
550         /* For a non-PIC library, the addresses are all absolute */
551         if (piclib) {
552                 dynamic_addr = (unsigned long) DL_RELOC_ADDR(lib_loadaddr, dynamic_addr);
553         }
554
555         /*
556          * OK, the ELF library is now loaded into VM in the correct locations
557          * The next step is to go through and do the dynamic linking (if needed).
558          */
559
560         /* Start by scanning the dynamic section to get all of the pointers */
561
562         if (!dynamic_addr) {
563                 _dl_internal_error_number = LD_ERROR_NODYNAMIC;
564                 _dl_dprintf(2, "%s: '%s' is missing a dynamic section\n",
565                                 _dl_progname, libname);
566                 _dl_munmap(header, _dl_pagesize);
567                 return NULL;
568         }
569
570         dpnt = (ElfW(Dyn) *) dynamic_addr;
571         _dl_memset(dynamic_info, 0, sizeof(dynamic_info));
572         _dl_parse_dynamic_info(dpnt, dynamic_info, NULL, lib_loadaddr);
573         /* If the TEXTREL is set, this means that we need to make the pages
574            writable before we perform relocations.  Do this now. They get set
575            back again later. */
576
577         if (dynamic_info[DT_TEXTREL]) {
578 #ifndef __FORCE_SHAREABLE_TEXT_SEGMENTS__
579                 ppnt = (ElfW(Phdr) *)(intptr_t) & header[epnt->e_phoff];
580                 for (i = 0; i < epnt->e_phnum; i++, ppnt++) {
581                         if (ppnt->p_type == PT_LOAD && !(ppnt->p_flags & PF_W))
582                                 _dl_mprotect((void *) ((piclib ? libaddr : 0) +
583                                                         (ppnt->p_vaddr & PAGE_ALIGN)),
584                                                 (ppnt->p_vaddr & ADDR_ALIGN) + (unsigned long) ppnt->p_filesz,
585                                                 PROT_READ | PROT_WRITE | PROT_EXEC);
586                 }
587 #else
588                 _dl_dprintf(_dl_debug_file, "Can't modify %s's text section. Use GCC option -fPIC for shared objects, please.\n",libname);
589                 _dl_exit(1);
590 #endif
591         }
592
593         tpnt = _dl_add_elf_hash_table(libname, lib_loadaddr, dynamic_info,
594                         dynamic_addr, 0);
595         tpnt->relro_addr = relro_addr;
596         tpnt->relro_size = relro_size;
597         tpnt->st_dev = st.st_dev;
598         tpnt->st_ino = st.st_ino;
599         tpnt->ppnt = (ElfW(Phdr) *) DL_RELOC_ADDR(tpnt->loadaddr, epnt->e_phoff);
600         tpnt->n_phent = epnt->e_phnum;
601
602         /*
603          * Add this object into the symbol chain
604          */
605         if (*rpnt) {
606                 (*rpnt)->next = (struct dyn_elf *) _dl_malloc(sizeof(struct dyn_elf));
607                 _dl_memset((*rpnt)->next, 0, sizeof(struct dyn_elf));
608                 (*rpnt)->next->prev = (*rpnt);
609                 *rpnt = (*rpnt)->next;
610         }
611 #ifndef SHARED
612         /* When statically linked, the first time we dlopen a DSO
613          * the *rpnt is NULL, so we need to allocate memory for it,
614          * and initialize the _dl_symbol_table.
615          */ 
616         else {
617                 *rpnt = _dl_symbol_tables = (struct dyn_elf *) _dl_malloc(sizeof(struct dyn_elf));
618                 _dl_memset(*rpnt, 0, sizeof(struct dyn_elf));
619         }
620 #endif
621         (*rpnt)->dyn = tpnt;
622         tpnt->symbol_scope = _dl_symbol_tables;
623         tpnt->usage_count++;
624         tpnt->libtype = elf_lib;
625
626         /*
627          * OK, the next thing we need to do is to insert the dynamic linker into
628          * the proper entry in the GOT so that the PLT symbols can be properly
629          * resolved.
630          */
631
632         lpnt = (unsigned long *) dynamic_info[DT_PLTGOT];
633
634         if (lpnt) {
635                 lpnt = (unsigned long *) (dynamic_info[DT_PLTGOT]);
636                 INIT_GOT(lpnt, tpnt);
637         }
638
639         _dl_if_debug_dprint("\n\tfile='%s';  generating link map\n", libname);
640         _dl_if_debug_dprint("\t\tdynamic: %x  base: %x\n", dynamic_addr, DL_LOADADDR_BASE(lib_loadaddr));
641         _dl_if_debug_dprint("\t\t  entry: %x  phdr: %x  phnum: %x\n\n",
642                         DL_RELOC_ADDR(lib_loadaddr, epnt->e_entry), tpnt->ppnt, tpnt->n_phent);
643
644         _dl_munmap(header, _dl_pagesize);
645
646         return tpnt;
647 }
648
649 /* now_flag must be RTLD_NOW or zero */
650 int _dl_fixup(struct dyn_elf *rpnt, int now_flag)
651 {
652         int goof = 0;
653         struct elf_resolve *tpnt;
654         ElfW(Word) reloc_size, relative_count;
655         ElfW(Addr) reloc_addr;
656
657         if (rpnt->next)
658                 goof = _dl_fixup(rpnt->next, now_flag);
659         if (goof)
660                 return goof;
661         tpnt = rpnt->dyn;
662
663         if (!(tpnt->init_flag & RELOCS_DONE))
664                 _dl_if_debug_dprint("relocation processing: %s\n", tpnt->libname);
665
666         if (unlikely(tpnt->dynamic_info[UNSUPPORTED_RELOC_TYPE])) {
667                 _dl_if_debug_dprint("%s: can't handle %s relocation records\n",
668                                 _dl_progname, UNSUPPORTED_RELOC_STR);
669                 goof++;
670                 return goof;
671         }
672
673         reloc_size = tpnt->dynamic_info[DT_RELOC_TABLE_SIZE];
674 /* On some machines, notably SPARC & PPC, DT_REL* includes DT_JMPREL in its
675    range.  Note that according to the ELF spec, this is completely legal! */
676 #ifdef ELF_MACHINE_PLTREL_OVERLAP
677         reloc_size -= tpnt->dynamic_info [DT_PLTRELSZ];
678 #endif
679         if (tpnt->dynamic_info[DT_RELOC_TABLE_ADDR] &&
680             !(tpnt->init_flag & RELOCS_DONE)) {
681                 reloc_addr = tpnt->dynamic_info[DT_RELOC_TABLE_ADDR];
682                 relative_count = tpnt->dynamic_info[DT_RELCONT_IDX];
683                 if (relative_count) { /* Optimize the XX_RELATIVE relocations if possible */
684                         reloc_size -= relative_count * sizeof(ELF_RELOC);
685                         elf_machine_relative(tpnt->loadaddr, reloc_addr, relative_count);
686                         reloc_addr += relative_count * sizeof(ELF_RELOC);
687                 }
688                 goof += _dl_parse_relocation_information(rpnt,
689                                 reloc_addr,
690                                 reloc_size);
691                 tpnt->init_flag |= RELOCS_DONE;
692         }
693         if (tpnt->dynamic_info[DT_BIND_NOW])
694                 now_flag = RTLD_NOW;
695         if (tpnt->dynamic_info[DT_JMPREL] &&
696             (!(tpnt->init_flag & JMP_RELOCS_DONE) ||
697              (now_flag && !(tpnt->rtld_flags & now_flag)))) {
698                 tpnt->rtld_flags |= now_flag;
699                 if (!(tpnt->rtld_flags & RTLD_NOW)) {
700                         _dl_parse_lazy_relocation_information(rpnt,
701                                         tpnt->dynamic_info[DT_JMPREL],
702                                         tpnt->dynamic_info [DT_PLTRELSZ]);
703                 } else {
704                         goof += _dl_parse_relocation_information(rpnt,
705                                         tpnt->dynamic_info[DT_JMPREL],
706                                         tpnt->dynamic_info[DT_PLTRELSZ]);
707                 }
708                 tpnt->init_flag |= JMP_RELOCS_DONE;
709         }
710         return goof;
711 }
712
713 /* Minimal printf which handles only %s, %d, and %x */
714 void _dl_dprintf(int fd, const char *fmt, ...)
715 {
716 #if __WORDSIZE > 32
717         long int num;
718 #else
719         int num;
720 #endif
721         va_list args;
722         char *start, *ptr, *string;
723         static char *buf;
724
725         if (!fmt)
726                 return;
727
728         buf = _dl_mmap((void *) 0, _dl_pagesize, PROT_READ | PROT_WRITE,
729                         MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
730         if (_dl_mmap_check_error(buf)) {
731                 _dl_write(fd, "mmap of a spare page failed!\n", 29);
732                 _dl_exit(20);
733         }
734
735         start = ptr = buf;
736
737         if (_dl_strlen(fmt) >= (_dl_pagesize - 1)) {
738                 _dl_write(fd, "overflow\n", 11);
739                 _dl_exit(20);
740         }
741
742         _dl_strcpy(buf, fmt);
743         va_start(args, fmt);
744
745         while (start) {
746                 while (*ptr != '%' && *ptr) {
747                         ptr++;
748                 }
749
750                 if (*ptr == '%') {
751                         *ptr++ = '\0';
752                         _dl_write(fd, start, _dl_strlen(start));
753
754                         switch (*ptr++) {
755                                 case 's':
756                                         string = va_arg(args, char *);
757
758                                         if (!string)
759                                                 _dl_write(fd, "(null)", 6);
760                                         else
761                                                 _dl_write(fd, string, _dl_strlen(string));
762                                         break;
763
764                                 case 'i':
765                                 case 'd':
766                                         {
767                                                 char tmp[22];
768 #if __WORDSIZE > 32
769                                                 num = va_arg(args, long int);
770 #else
771                                                 num = va_arg(args, int);
772 #endif
773                                                 string = _dl_simple_ltoa(tmp, num);
774                                                 _dl_write(fd, string, _dl_strlen(string));
775                                                 break;
776                                         }
777                                 case 'x':
778                                 case 'X':
779                                         {
780                                                 char tmp[22];
781 #if __WORDSIZE > 32
782                                                 num = va_arg(args, long int);
783 #else
784                                                 num = va_arg(args, int);
785 #endif
786                                                 string = _dl_simple_ltoahex(tmp, num);
787                                                 _dl_write(fd, string, _dl_strlen(string));
788                                                 break;
789                                         }
790                                 default:
791                                         _dl_write(fd, "(null)", 6);
792                                         break;
793                         }
794
795                         start = ptr;
796                 } else {
797                         _dl_write(fd, start, _dl_strlen(start));
798                         start = NULL;
799                 }
800         }
801         _dl_munmap(buf, _dl_pagesize);
802         return;
803 }
804
805 char *_dl_strdup(const char *string)
806 {
807         char *retval;
808         int len;
809
810         len = _dl_strlen(string);
811         retval = _dl_malloc(len + 1);
812         _dl_strcpy(retval, string);
813         return retval;
814 }
815
816 void _dl_parse_dynamic_info(ElfW(Dyn) *dpnt, unsigned long dynamic_info[],
817                             void *debug_addr, DL_LOADADDR_TYPE load_off)
818 {
819         __dl_parse_dynamic_info(dpnt, dynamic_info, debug_addr, load_off);
820 }
821
822 /* we want this in ldso.so and libdl.a but nowhere else */
823 #ifdef __USE_GNU
824 #if defined IS_IN_rtld || (defined IS_IN_libdl && ! defined SHARED)
825 extern __typeof(dl_iterate_phdr) __dl_iterate_phdr;
826 int
827 __dl_iterate_phdr (int (*callback) (struct dl_phdr_info *info, size_t size, void *data), void *data)
828 {
829         struct elf_resolve *l;
830         struct dl_phdr_info info;
831         int ret = 0;
832
833         for (l = _dl_loaded_modules; l != NULL; l = l->next) {
834                 info.dlpi_addr = l->loadaddr;
835                 info.dlpi_name = l->libname;
836                 info.dlpi_phdr = l->ppnt;
837                 info.dlpi_phnum = l->n_phent;
838                 ret = callback (&info, sizeof (struct dl_phdr_info), data);
839                 if (ret)
840                         break;
841         }
842         return ret;
843 }
844 strong_alias(__dl_iterate_phdr, dl_iterate_phdr)
845 #endif
846 #endif