Home | History | Annotate | Line # | Download | only in dlfcn
dlfcn_elf.c revision 1.8
      1  1.8    joerg /*	$NetBSD: dlfcn_elf.c,v 1.8 2011/03/07 05:09:11 joerg Exp $	*/
      2  1.1  minoura 
      3  1.1  minoura /*
      4  1.1  minoura  * Copyright (c) 2000 Takuya SHIOZAKI
      5  1.1  minoura  * All rights reserved.
      6  1.1  minoura  *
      7  1.1  minoura  * Redistribution and use in source and binary forms, with or without
      8  1.1  minoura  * modification, are permitted provided that the following conditions
      9  1.1  minoura  * are met:
     10  1.1  minoura  * 1. Redistributions of source code must retain the above copyright
     11  1.1  minoura  *    notice, this list of conditions and the following disclaimer.
     12  1.1  minoura  * 2. Redistributions in binary form must reproduce the above copyright
     13  1.1  minoura  *    notice, this list of conditions and the following disclaimer in the
     14  1.1  minoura  *    documentation and/or other materials provided with the distribution.
     15  1.1  minoura  *
     16  1.1  minoura  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
     17  1.1  minoura  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
     18  1.1  minoura  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
     19  1.1  minoura  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
     20  1.1  minoura  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
     21  1.1  minoura  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     22  1.1  minoura  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     23  1.1  minoura  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     24  1.1  minoura  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
     25  1.1  minoura  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     26  1.1  minoura  */
     27  1.1  minoura 
     28  1.1  minoura #include <sys/cdefs.h>
     29  1.1  minoura #if defined(LIBC_SCCS) && !defined(lint)
     30  1.8    joerg __RCSID("$NetBSD: dlfcn_elf.c,v 1.8 2011/03/07 05:09:11 joerg Exp $");
     31  1.1  minoura #endif /* LIBC_SCCS and not lint */
     32  1.1  minoura 
     33  1.8    joerg #include "reentrant.h"
     34  1.1  minoura #include "namespace.h"
     35  1.8    joerg #include <elf.h>
     36  1.8    joerg #include <errno.h>
     37  1.8    joerg #include <string.h>
     38  1.1  minoura 
     39  1.5  thorpej #undef dlopen
     40  1.5  thorpej #undef dlclose
     41  1.5  thorpej #undef dlsym
     42  1.5  thorpej #undef dlerror
     43  1.5  thorpej #undef dladdr
     44  1.6    pooka #undef dfinfo
     45  1.5  thorpej 
     46  1.5  thorpej #define	dlopen		___dlopen
     47  1.5  thorpej #define	dlclose		___dlclose
     48  1.5  thorpej #define	dlsym		___dlsym
     49  1.5  thorpej #define	dlerror		___dlerror
     50  1.5  thorpej #define	dladdr		___dladdr
     51  1.6    pooka #define	dlinfo		___dlinfo
     52  1.7    skrll #define	dl_iterate_phdr		___dl_iterate_phdr
     53  1.5  thorpej 
     54  1.1  minoura #define ELFSIZE ARCH_ELFSIZE
     55  1.1  minoura #include "rtld.h"
     56  1.1  minoura 
     57  1.1  minoura #ifdef __weak_alias
     58  1.5  thorpej __weak_alias(dlopen,___dlopen)
     59  1.5  thorpej __weak_alias(dlclose,___dlclose)
     60  1.5  thorpej __weak_alias(dlsym,___dlsym)
     61  1.5  thorpej __weak_alias(dlerror,___dlerror)
     62  1.5  thorpej __weak_alias(dladdr,___dladdr)
     63  1.6    pooka __weak_alias(dlinfo,___dlinfo)
     64  1.7    skrll __weak_alias(dl_iterate_phdr,___dl_iterate_phdr)
     65  1.5  thorpej 
     66  1.5  thorpej __weak_alias(__dlopen,___dlopen)
     67  1.5  thorpej __weak_alias(__dlclose,___dlclose)
     68  1.5  thorpej __weak_alias(__dlsym,___dlsym)
     69  1.5  thorpej __weak_alias(__dlerror,___dlerror)
     70  1.5  thorpej __weak_alias(__dladdr,___dladdr)
     71  1.6    pooka __weak_alias(__dlinfo,___dlinfo)
     72  1.7    skrll __weak_alias(__dl_iterate_phdr,___dl_iterate_phdr)
     73  1.1  minoura #endif
     74  1.1  minoura 
     75  1.4    skrll /*
     76  1.4    skrll  * For ELF, the dynamic linker directly resolves references to its
     77  1.4    skrll  * services to functions inside the dynamic linker itself.  These
     78  1.4    skrll  * weak-symbol stubs are necessary so that "ld" won't complain about
     79  1.4    skrll  * undefined symbols.  The stubs are executed only when the program is
     80  1.4    skrll  * linked statically, or when a given service isn't implemented in the
     81  1.4    skrll  * dynamic linker.  They must return an error if called, and they must
     82  1.4    skrll  * be weak symbols so that the dynamic linker can override them.
     83  1.4    skrll  */
     84  1.4    skrll 
     85  1.4    skrll static char dlfcn_error[] = "Service unavailable";
     86  1.4    skrll 
     87  1.4    skrll /*ARGSUSED*/
     88  1.4    skrll void *
     89  1.4    skrll dlopen(const char *name, int mode)
     90  1.4    skrll {
     91  1.4    skrll 
     92  1.4    skrll 	return NULL;
     93  1.4    skrll }
     94  1.4    skrll 
     95  1.4    skrll /*ARGSUSED*/
     96  1.4    skrll int
     97  1.4    skrll dlclose(void *fd)
     98  1.4    skrll {
     99  1.4    skrll 
    100  1.4    skrll 	return -1;
    101  1.4    skrll }
    102  1.4    skrll 
    103  1.4    skrll /*ARGSUSED*/
    104  1.4    skrll void *
    105  1.4    skrll dlsym(void *handle, const char *name)
    106  1.4    skrll {
    107  1.4    skrll 
    108  1.4    skrll 	return NULL;
    109  1.4    skrll }
    110  1.4    skrll 
    111  1.4    skrll /*ARGSUSED*/
    112  1.4    skrll __aconst char *
    113  1.4    skrll dlerror()
    114  1.4    skrll {
    115  1.4    skrll 
    116  1.4    skrll 	return dlfcn_error;
    117  1.4    skrll }
    118  1.4    skrll 
    119  1.4    skrll /*ARGSUSED*/
    120  1.4    skrll int
    121  1.4    skrll dladdr(const void *addr, Dl_info *dli)
    122  1.4    skrll {
    123  1.4    skrll 
    124  1.4    skrll 	return 0;
    125  1.4    skrll }
    126  1.6    pooka 
    127  1.6    pooka /*ARGSUSED*/
    128  1.6    pooka int
    129  1.6    pooka dlinfo(void *handle, int req, void *v)
    130  1.6    pooka {
    131  1.6    pooka 
    132  1.6    pooka 	return -1;
    133  1.6    pooka }
    134  1.7    skrll 
    135  1.8    joerg static once_t dl_iterate_phdr_once = ONCE_INITIALIZER;
    136  1.8    joerg static const char *dlpi_name;
    137  1.8    joerg static Elf_Addr dlpi_addr;
    138  1.8    joerg static const Elf_Phdr *dlpi_phdr;
    139  1.8    joerg static Elf_Half dlpi_phnum;
    140  1.8    joerg 
    141  1.8    joerg /*
    142  1.8    joerg  * Declare as common symbol to allow new libc with older binaries to
    143  1.8    joerg  * not trigger an undefined reference.
    144  1.8    joerg  */
    145  1.8    joerg extern __dso_hidden void *__auxinfo;
    146  1.8    joerg 
    147  1.8    joerg static void
    148  1.8    joerg dl_iterate_phdr_setup(void)
    149  1.8    joerg {
    150  1.8    joerg 	const AuxInfo *aux;
    151  1.8    joerg 
    152  1.8    joerg 	if (__auxinfo == NULL)
    153  1.8    joerg 		return;
    154  1.8    joerg 
    155  1.8    joerg 	for (aux = __auxinfo; aux->a_type != AT_NULL; ++aux) {
    156  1.8    joerg 		switch (aux->a_type) {
    157  1.8    joerg 		case AT_BASE:
    158  1.8    joerg 			dlpi_addr = aux->a_v;
    159  1.8    joerg 			break;
    160  1.8    joerg 		case AT_PHDR:
    161  1.8    joerg 			dlpi_phdr = (void *)aux->a_v;
    162  1.8    joerg 			break;
    163  1.8    joerg 		case AT_PHNUM:
    164  1.8    joerg 			dlpi_phnum = aux->a_v;
    165  1.8    joerg 			break;
    166  1.8    joerg 		case AT_SUN_EXECNAME:
    167  1.8    joerg 			dlpi_name = (void *)aux->a_v;
    168  1.8    joerg 			break;
    169  1.8    joerg 		}
    170  1.8    joerg 	}
    171  1.8    joerg }
    172  1.8    joerg 
    173  1.7    skrll /*ARGSUSED*/
    174  1.7    skrll int
    175  1.7    skrll dl_iterate_phdr(int (*callback)(struct dl_phdr_info *, size_t, void *),
    176  1.7    skrll     void *data)
    177  1.7    skrll {
    178  1.8    joerg 	struct dl_phdr_info phdr_info;
    179  1.8    joerg 
    180  1.8    joerg 	if (__auxinfo == NULL)
    181  1.8    joerg 		return EOPNOTSUPP;
    182  1.7    skrll 
    183  1.8    joerg 	thr_once(&dl_iterate_phdr_once, dl_iterate_phdr_setup);
    184  1.8    joerg 
    185  1.8    joerg 	memset(&phdr_info, 0, sizeof(phdr_info));
    186  1.8    joerg 	phdr_info.dlpi_addr = dlpi_addr;
    187  1.8    joerg 	phdr_info.dlpi_phdr = dlpi_phdr;
    188  1.8    joerg 	phdr_info.dlpi_phnum = dlpi_phnum;
    189  1.8    joerg 	phdr_info.dlpi_name = dlpi_name;
    190  1.8    joerg 
    191  1.8    joerg 	return callback(&phdr_info, sizeof(phdr_info), data);
    192  1.7    skrll }
    193