diff options
Diffstat (limited to 'elf')
-rw-r--r-- | elf/Makefile | 2 | ||||
-rw-r--r-- | elf/dl-close.c | 8 | ||||
-rw-r--r-- | elf/dl-debug.c | 56 | ||||
-rw-r--r-- | elf/dl-init.c | 5 | ||||
-rw-r--r-- | elf/dl-lookup.c | 3 | ||||
-rw-r--r-- | elf/dl-open.c | 15 | ||||
-rw-r--r-- | elf/link.h | 10 | ||||
-rw-r--r-- | elf/rtld.c | 34 |
8 files changed, 108 insertions, 25 deletions
diff --git a/elf/Makefile b/elf/Makefile index 07ffbd8..be7604d 100644 --- a/elf/Makefile +++ b/elf/Makefile @@ -26,7 +26,7 @@ routines = $(dl-routines) dl-open dl-close dl-symbol dl-support # The core dynamic linking functions are in libc for the static and # profiled libraries. dl-routines = $(addprefix dl-,load lookup object reloc deps \ - runtime error init fini) + runtime error init fini debug) # But they are absent from the shared libc, because that code is in ld.so. elide-routines.so = $(dl-routines) dl-support diff --git a/elf/dl-close.c b/elf/dl-close.c index 70d7ab4..184d382 100644 --- a/elf/dl-close.c +++ b/elf/dl-close.c @@ -41,6 +41,10 @@ _dl_close (struct link_map *map) /* There are still references to this object. Do nothing more. */ return; + /* Notify the debugger we are about to remove some loaded objects. */ + _r_debug.r_state = RT_DELETE; + _dl_debug_state (); + list = map->l_searchlist; /* The search list contains a counted reference to each object it @@ -105,4 +109,8 @@ _dl_close (struct link_map *map) } free (list); + + /* Notify the debugger those objects are finalized and gone. */ + _r_debug.r_state = RT_CONSISTENT; + _dl_debug_state (); } diff --git a/elf/dl-debug.c b/elf/dl-debug.c new file mode 100644 index 0000000..861e001 --- /dev/null +++ b/elf/dl-debug.c @@ -0,0 +1,56 @@ +/* Communicate dynamic linker state to the debugger at runtime. +Copyright (C) 1996 Free Software Foundation, Inc. +This file is part of the GNU C Library. + +The GNU C Library is free software; you can redistribute it and/or +modify it under the terms of the GNU Library General Public License as +published by the Free Software Foundation; either version 2 of the +License, or (at your option) any later version. + +The GNU C Library is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +Library General Public License for more details. + +You should have received a copy of the GNU Library General Public +License along with the GNU C Library; see the file COPYING.LIB. If +not, write to the Free Software Foundation, Inc., 675 Mass Ave, +Cambridge, MA 02139, USA. */ + +#include <link.h> + +/* This structure communicates dl state to the debugger. The debugger + normally finds it via the DT_DEBUG entry in the dynamic section, but in + a statically-linked program there is no dynamic section for the debugger + to examine and it looks for this particular symbol name. */ +struct r_debug _r_debug; + + +/* Initialize _r_debug if it has not already been done. The argument is + the run-time load address of the dynamic linker, to be put in + _r_debug.r_ldbase. Returns the address of _r_debug. */ + +struct r_debug * +_dl_debug_initialize (ElfW(Addr) ldbase) +{ + if (_r_debug.r_brk == 0) + { + /* Tell the debugger where to find the map of loaded objects. */ + _r_debug.r_version = 1 /* R_DEBUG_VERSION XXX */; + _r_debug.r_ldbase = _dl_rtld_map.l_addr; /* Record our load address. */ + _r_debug.r_map = _dl_loaded; + _r_debug.r_brk = (ElfW(Addr)) &_dl_debug_state; + } + + return &_r_debug; +} + + +/* This function exists solely to have a breakpoint set on it by the + debugger. The debugger is supposed to find this function's address by + examining the r_brk member of struct r_debug, but GDB 4.15 in fact looks + for this particular symbol name in the PT_INTERP file. */ +void +_dl_debug_state (void) +{ +} diff --git a/elf/dl-init.c b/elf/dl-init.c index 66ef83e..6259c35 100644 --- a/elf/dl-init.c +++ b/elf/dl-init.c @@ -65,5 +65,10 @@ _dl_init_next (struct link_map *map) l->l_init_called = 1; } + + /* Notify the debugger all new objects are now ready to go. */ + _r_debug.r_state = RT_CONSISTENT; + _dl_debug_state (); + return 0; } diff --git a/elf/dl-lookup.c b/elf/dl-lookup.c index bbccbc0..7ceffa2 100644 --- a/elf/dl-lookup.c +++ b/elf/dl-lookup.c @@ -129,7 +129,8 @@ _dl_lookup_symbol (const char *undef_name, const ElfW(Sym) **ref, } } - if (weak_value.s == NULL && ELFW(ST_BIND) ((*ref)->st_info) != STB_WEAK) + if (weak_value.s == NULL && + !*ref || ELFW(ST_BIND) ((*ref)->st_info) != STB_WEAK) { /* We could find no value for a strong reference. */ const char msg[] = "undefined symbol: "; diff --git a/elf/dl-open.c b/elf/dl-open.c index 9389303..373d32d 100644 --- a/elf/dl-open.c +++ b/elf/dl-open.c @@ -29,6 +29,7 @@ _dl_open (struct link_map *parent, const char *file, int mode) { struct link_map *new, *l; ElfW(Addr) init; + struct r_debug *r; /* Load the named object. */ @@ -47,7 +48,7 @@ _dl_open (struct link_map *parent, const char *file, int mode) l = new; while (l->l_next) l = l->l_next; - do + while (1) { if (! l->l_relocated) { @@ -56,8 +57,10 @@ _dl_open (struct link_map *parent, const char *file, int mode) *_dl_global_scope_end = NULL; } + if (l == new) + break; l = l->l_prev; - } while (l != new); + } new->l_global = (mode & RTLD_GLOBAL); if (new->l_global) @@ -108,6 +111,14 @@ _dl_open (struct link_map *parent, const char *file, int mode) } } + + /* Notify the debugger we have added some objects. We need to call + _dl_debug_initialize in a static program in case dynamic linking has + not been used before. */ + r = _dl_debug_initialize (0); + r->r_state = RT_ADD; + _dl_debug_state (); + /* Run the initializer functions of new objects. */ while (init = _dl_init_next (new)) (*(void (*) (void)) init) (); @@ -62,6 +62,9 @@ struct r_debug ElfW(Addr) r_ldbase; /* Base address the linker is loaded at. */ }; +/* This is the instance of that structure used by the dynamic linker. */ +extern struct r_debug _r_debug; + /* This symbol refers to the "dynamic structure" in the `.dynamic' section of whatever module refers to `_DYNAMIC'. So, to find its own `struct r_debug', a program could do: @@ -291,7 +294,12 @@ extern void _dl_fini (void); any shared object mappings. The `r_state' member of `struct r_debug' says what change is taking place. This function's address is the value of the `r_brk' member. */ -extern void _dl_r_debug_state (void); +extern void _dl_debug_state (void); + +/* Initialize `struct r_debug' if it has not already been done. The + argument is the run-time load address of the dynamic linker, to be put + in the `r_ldbase' member. Returns the address of the structure. */ +extern struct r_debug *_dl_debug_initialize (ElfW(Addr) ldbase); #endif /* link.h */ @@ -46,8 +46,6 @@ int _dl_argc; char **_dl_argv; const char *_dl_rpath; -struct r_debug _dl_r_debug; - static void dl_main (const ElfW(Phdr) *phdr, ElfW(Half) phent, ElfW(Addr) *user_entry); @@ -229,12 +227,6 @@ of this helper program; chances are you did not intend to run this program.\n", /* Set up our cache of pointers into the hash table. */ _dl_setup_hash (l); - if (l->l_info[DT_DEBUG]) - /* There is a DT_DEBUG entry in the dynamic section. Fill it in - with the run-time address of the r_debug structure, which we - will set up later to communicate with the debugger. */ - l->l_info[DT_DEBUG]->d_un.d_ptr = (ElfW(Addr)) &_dl_r_debug; - /* Put the link_map for ourselves on the chain so it can be found by name. */ _dl_rtld_map.l_name = (char *) _dl_rtld_map.l_libname = interpreter_name; @@ -343,11 +335,20 @@ of this helper program; chances are you did not intend to run this program.\n", _dl_relocate_object (&_dl_rtld_map, &_dl_default_scope[2], 0); } - /* Tell the debugger where to find the map of loaded objects. */ - _dl_r_debug.r_version = 1 /* R_DEBUG_VERSION XXX */; - _dl_r_debug.r_ldbase = _dl_rtld_map.l_addr; /* Record our load address. */ - _dl_r_debug.r_map = _dl_loaded; - _dl_r_debug.r_brk = (ElfW(Addr)) &_dl_r_debug_state; + { + /* Initialize _r_debug. */ + struct r_debug *r = _dl_debug_initialize (_dl_rtld_map.l_addr); + + l = _dl_loaded; + if (l->l_info[DT_DEBUG]) + /* There is a DT_DEBUG entry in the dynamic section. Fill it in + with the run-time address of the r_debug structure */ + l->l_info[DT_DEBUG]->d_un.d_ptr = (ElfW(Addr)) r; + + /* Notify the debugger that all objects are now mapped in. */ + r->r_state = RT_ADD; + _dl_debug_state (); + } if (_dl_rtld_map.l_info[DT_INIT]) { @@ -365,10 +366,3 @@ of this helper program; chances are you did not intend to run this program.\n", /* Once we return, _dl_sysdep_start will invoke the DT_INIT functions and then *USER_ENTRY. */ } - -/* This function exists solely to have a breakpoint set on it by the - debugger. */ -void -_dl_r_debug_state (void) -{ -} |