diff options
Diffstat (limited to 'gdb/solib-som.c')
-rw-r--r-- | gdb/solib-som.c | 891 |
1 files changed, 0 insertions, 891 deletions
diff --git a/gdb/solib-som.c b/gdb/solib-som.c deleted file mode 100644 index 27e39a7..0000000 --- a/gdb/solib-som.c +++ /dev/null @@ -1,891 +0,0 @@ -/* Handle SOM shared libraries. - - Copyright (C) 2004-2015 Free Software Foundation, Inc. - - This file is part of GDB. - - This program is free software; you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation; either version 3 of the License, or - (at your option) any later version. - - This program 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 General Public License for more details. - - You should have received a copy of the GNU General Public License - along with this program. If not, see <http://www.gnu.org/licenses/>. */ - -#include "defs.h" -#include "symtab.h" -#include "bfd.h" -#include "symfile.h" -#include "objfiles.h" -#include "gdbcore.h" -#include "target.h" -#include "inferior.h" - -#include "hppa-tdep.h" -#include "solist.h" -#include "solib.h" -#include "solib-som.h" - -#undef SOLIB_SOM_DBG - -/* These ought to be defined in some public interface, but aren't. They - define the meaning of the various bits in the distinguished __dld_flags - variable that is declared in every debuggable a.out on HP-UX, and that - is shared between the debugger and the dynamic linker. */ - -#define DLD_FLAGS_MAPPRIVATE 0x1 -#define DLD_FLAGS_HOOKVALID 0x2 -#define DLD_FLAGS_LISTVALID 0x4 -#define DLD_FLAGS_BOR_ENABLE 0x8 - -struct lm_info - { - /* Version of this structure (it is expected to change again in - hpux10). */ - unsigned char struct_version; - - /* Binding mode for this library. */ - unsigned char bind_mode; - - /* Version of this library. */ - short library_version; - - /* Start of text address, - link-time text location (length of text area), - end of text address. */ - CORE_ADDR text_addr; - CORE_ADDR text_link_addr; - CORE_ADDR text_end; - - /* Start of data, start of bss and end of data. */ - CORE_ADDR data_start; - CORE_ADDR bss_start; - CORE_ADDR data_end; - - /* Value of linkage pointer (%r19). */ - CORE_ADDR got_value; - - /* Address in target of offset from thread-local register of - start of this thread's data. I.e., the first thread-local - variable in this shared library starts at *(tsd_start_addr) - from that area pointed to by cr27 (mpsfu_hi). - - We do the indirection as soon as we read it, so from then - on it's the offset itself. */ - CORE_ADDR tsd_start_addr; - - /* Address of the link map entry in the loader. */ - CORE_ADDR lm_addr; - }; - -/* These addresses should be filled in by som_solib_create_inferior_hook. - They are also used elsewhere in this module. */ - -typedef struct - { - CORE_ADDR address; - struct unwind_table_entry *unwind; - } -addr_and_unwind_t; - -/* When adding fields, be sure to clear them in _initialize_som_solib. */ -static struct - { - int is_valid; - addr_and_unwind_t hook; - addr_and_unwind_t hook_stub; - addr_and_unwind_t load; - addr_and_unwind_t load_stub; - addr_and_unwind_t unload; - addr_and_unwind_t unload2; - addr_and_unwind_t unload_stub; - } -dld_cache; - -static void -som_relocate_section_addresses (struct so_list *so, - struct target_section *sec) -{ - flagword aflag = bfd_get_section_flags(so->abfd, sec->the_bfd_section); - - if (aflag & SEC_CODE) - { - sec->addr += so->lm_info->text_addr - so->lm_info->text_link_addr; - sec->endaddr += so->lm_info->text_addr - so->lm_info->text_link_addr; - } - else if (aflag & SEC_DATA) - { - sec->addr += so->lm_info->data_start; - sec->endaddr += so->lm_info->data_start; - } - else - { - /* Nothing. */ - } -} - - -/* Variable storing HP-UX major release number. - - On non-native system, simply assume that the major release number - is 11. On native systems, hppa-hpux-nat.c initialization code - sets this number to the real one on startup. - - We cannot compute this value here, because we need to make a native - call to "uname". We are are not allowed to do that from here, as - this file is used for both native and cross debugging. */ - -#define DEFAULT_HPUX_MAJOR_RELEASE 11 -int hpux_major_release = DEFAULT_HPUX_MAJOR_RELEASE; - -static int -get_hpux_major_release (void) -{ - return hpux_major_release; -} - -/* DL header flag defines. */ -#define SHLIB_TEXT_PRIVATE_ENABLE 0x4000 - -/* The DL header is documented in <shl.h>. We are only interested - in the flags field to determine whether the executable wants shared - libraries mapped private. */ -struct { - short junk[37]; - short flags; -} dl_header; - -/* This hook gets called just before the first instruction in the - inferior process is executed. - - This is our opportunity to set magic flags in the inferior so - that GDB can be notified when a shared library is mapped in and - to tell the dynamic linker that a private copy of the library is - needed (so GDB can set breakpoints in the library). - - __dld_flags is the location of the magic flags; as of this implementation - there are 3 flags of interest: - - bit 0 when set indicates that private copies of the libraries are needed - bit 1 when set indicates that the callback hook routine is valid - bit 2 when set indicates that the dynamic linker should maintain the - __dld_list structure when loading/unloading libraries. - - Note that shared libraries are not mapped in at this time, so we have - run the inferior until the libraries are mapped in. Typically this - means running until the "_start" is called. */ - -static void -som_solib_create_inferior_hook (int from_tty) -{ - enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ()); - struct bound_minimal_symbol msymbol; - unsigned int dld_flags, status, have_endo; - asection *shlib_info; - gdb_byte buf[4]; - CORE_ADDR anaddr; - - if (symfile_objfile == NULL) - return; - - /* First see if the objfile was dynamically linked. */ - shlib_info = bfd_get_section_by_name (symfile_objfile->obfd, "$SHLIB_INFO$"); - if (!shlib_info) - return; - - /* It's got a $SHLIB_INFO$ section, make sure it's not empty. */ - if (bfd_section_size (symfile_objfile->obfd, shlib_info) == 0) - return; - - /* Read the DL header. */ - bfd_get_section_contents (symfile_objfile->obfd, shlib_info, - (char *) &dl_header, 0, sizeof (dl_header)); - - have_endo = 0; - /* Slam the pid of the process into __d_pid. - - We used to warn when this failed, but that warning is only useful - on very old HP systems (hpux9 and older). The warnings are an - annoyance to users of modern systems and foul up the testsuite as - well. As a result, the warnings have been disabled. */ - msymbol = lookup_minimal_symbol ("__d_pid", NULL, symfile_objfile); - if (msymbol.minsym == NULL) - goto keep_going; - - anaddr = BMSYMBOL_VALUE_ADDRESS (msymbol); - store_unsigned_integer (buf, 4, byte_order, ptid_get_pid (inferior_ptid)); - status = target_write_memory (anaddr, buf, 4); - if (status != 0) - { - warning (_("\ -Unable to write __d_pid.\n\ -Suggest linking with /opt/langtools/lib/end.o.\n\ -GDB will be unable to track shl_load/shl_unload calls")); - goto keep_going; - } - - /* Get the value of _DLD_HOOK (an export stub) and put it in __dld_hook; - This will force the dynamic linker to call __d_trap when significant - events occur. - - Note that the above is the pre-HP-UX 9.0 behaviour. At 9.0 and above, - the dld provides an export stub named "__d_trap" as well as the - function named "__d_trap" itself, but doesn't provide "_DLD_HOOK". - We'll look first for the old flavor and then the new. */ - - msymbol = lookup_minimal_symbol ("_DLD_HOOK", NULL, symfile_objfile); - if (msymbol.minsym == NULL) - msymbol = lookup_minimal_symbol ("__d_trap", NULL, symfile_objfile); - if (msymbol.minsym == NULL) - { - warning (_("\ -Unable to find _DLD_HOOK symbol in object file.\n\ -Suggest linking with /opt/langtools/lib/end.o.\n\ -GDB will be unable to track shl_load/shl_unload calls")); - goto keep_going; - } - anaddr = BMSYMBOL_VALUE_ADDRESS (msymbol); - dld_cache.hook.address = anaddr; - - /* Grrr, this might not be an export symbol! We have to find the - export stub. */ - msymbol - = hppa_lookup_stub_minimal_symbol (MSYMBOL_LINKAGE_NAME (msymbol.minsym), - EXPORT); - if (msymbol.minsym != NULL) - { - anaddr = MSYMBOL_VALUE (msymbol.minsym); - dld_cache.hook_stub.address = anaddr; - } - store_unsigned_integer (buf, 4, byte_order, anaddr); - - msymbol = lookup_minimal_symbol ("__dld_hook", NULL, symfile_objfile); - if (msymbol.minsym == NULL) - { - warning (_("\ -Unable to find __dld_hook symbol in object file.\n\ -Suggest linking with /opt/langtools/lib/end.o.\n\ -GDB will be unable to track shl_load/shl_unload calls")); - goto keep_going; - } - anaddr = BMSYMBOL_VALUE_ADDRESS (msymbol); - status = target_write_memory (anaddr, buf, 4); - - /* Now set a shlib_event breakpoint at __d_trap so we can track - significant shared library events. */ - msymbol = lookup_minimal_symbol ("__d_trap", NULL, symfile_objfile); - if (msymbol.minsym == NULL) - { - warning (_("\ -Unable to find __dld_d_trap symbol in object file.\n\ -Suggest linking with /opt/langtools/lib/end.o.\n\ -GDB will be unable to track shl_load/shl_unload calls")); - goto keep_going; - } - create_solib_event_breakpoint (target_gdbarch (), - BMSYMBOL_VALUE_ADDRESS (msymbol)); - - /* We have all the support usually found in end.o, so we can track - shl_load and shl_unload calls. */ - have_endo = 1; - -keep_going: - - /* Get the address of __dld_flags, if no such symbol exists, then we can - not debug the shared code. */ - msymbol = lookup_minimal_symbol ("__dld_flags", NULL, NULL); - if (msymbol.minsym == NULL) - { - error (_("Unable to find __dld_flags symbol in object file.")); - } - - anaddr = BMSYMBOL_VALUE_ADDRESS (msymbol); - - /* Read the current contents. */ - status = target_read_memory (anaddr, buf, 4); - if (status != 0) - error (_("Unable to read __dld_flags.")); - dld_flags = extract_unsigned_integer (buf, 4, byte_order); - - /* If the libraries were not mapped private on HP-UX 11 and later, warn - the user. On HP-UX 10 and earlier, there is no easy way to specify - that shared libraries should be privately mapped. So, we just force - private mapping. */ - if (get_hpux_major_release () >= 11 - && (dl_header.flags & SHLIB_TEXT_PRIVATE_ENABLE) == 0 - && (dld_flags & DLD_FLAGS_MAPPRIVATE) == 0) - warning - (_("\ -Private mapping of shared library text was not specified\n\ -by the executable; setting a breakpoint in a shared library which\n\ -is not privately mapped will not work. See the HP-UX 11i v3 chatr\n\ -manpage for methods to privately map shared library text.")); - - /* Turn on the flags we care about. */ - if (get_hpux_major_release () < 11) - dld_flags |= DLD_FLAGS_MAPPRIVATE; - if (have_endo) - dld_flags |= DLD_FLAGS_HOOKVALID; - store_unsigned_integer (buf, 4, byte_order, dld_flags); - status = target_write_memory (anaddr, buf, 4); - if (status != 0) - error (_("Unable to write __dld_flags.")); - - /* Now find the address of _start and set a breakpoint there. - We still need this code for two reasons: - - * Not all sites have /opt/langtools/lib/end.o, so it's not always - possible to track the dynamic linker's events. - - * At this time no events are triggered for shared libraries - loaded at startup time (what a crock). */ - - msymbol = lookup_minimal_symbol ("_start", NULL, symfile_objfile); - if (msymbol.minsym == NULL) - error (_("Unable to find _start symbol in object file.")); - - anaddr = BMSYMBOL_VALUE_ADDRESS (msymbol); - - /* Make the breakpoint at "_start" a shared library event breakpoint. */ - create_solib_event_breakpoint (target_gdbarch (), anaddr); - - clear_symtab_users (0); -} - -static void -som_special_symbol_handling (void) -{ -} - -static void -som_solib_desire_dynamic_linker_symbols (void) -{ - struct objfile *objfile; - struct unwind_table_entry *u; - struct bound_minimal_symbol dld_msymbol; - - /* Do we already know the value of these symbols? If so, then - we've no work to do. - - (If you add clauses to this test, be sure to likewise update the - test within the loop.) */ - - if (dld_cache.is_valid) - return; - - ALL_OBJFILES (objfile) - { - dld_msymbol = lookup_minimal_symbol ("shl_load", NULL, objfile); - if (dld_msymbol.minsym != NULL) - { - dld_cache.load.address = MSYMBOL_VALUE (dld_msymbol.minsym); - dld_cache.load.unwind = find_unwind_entry (dld_cache.load.address); - } - - dld_msymbol = lookup_minimal_symbol_solib_trampoline ("shl_load", - objfile); - if (dld_msymbol.minsym != NULL) - { - if (MSYMBOL_TYPE (dld_msymbol.minsym) == mst_solib_trampoline) - { - u = find_unwind_entry (MSYMBOL_VALUE (dld_msymbol.minsym)); - if ((u != NULL) && (u->stub_unwind.stub_type == EXPORT)) - { - dld_cache.load_stub.address - = MSYMBOL_VALUE (dld_msymbol.minsym); - dld_cache.load_stub.unwind = u; - } - } - } - - dld_msymbol = lookup_minimal_symbol ("shl_unload", NULL, objfile); - if (dld_msymbol.minsym != NULL) - { - dld_cache.unload.address = MSYMBOL_VALUE (dld_msymbol.minsym); - dld_cache.unload.unwind = find_unwind_entry (dld_cache.unload.address); - - /* ??rehrauer: I'm not sure exactly what this is, but it appears - that on some HPUX 10.x versions, there's two unwind regions to - cover the body of "shl_unload", the second being 4 bytes past - the end of the first. This is a large hack to handle that - case, but since I don't seem to have any legitimate way to - look for this thing via the symbol table... */ - - if (dld_cache.unload.unwind != NULL) - { - u = find_unwind_entry (dld_cache.unload.unwind->region_end + 4); - if (u != NULL) - { - dld_cache.unload2.address = u->region_start; - dld_cache.unload2.unwind = u; - } - } - } - - dld_msymbol = lookup_minimal_symbol_solib_trampoline ("shl_unload", - objfile); - if (dld_msymbol.minsym != NULL) - { - if (MSYMBOL_TYPE (dld_msymbol.minsym) == mst_solib_trampoline) - { - u = find_unwind_entry (MSYMBOL_VALUE (dld_msymbol.minsym)); - if ((u != NULL) && (u->stub_unwind.stub_type == EXPORT)) - { - dld_cache.unload_stub.address - = MSYMBOL_VALUE (dld_msymbol.minsym); - dld_cache.unload_stub.unwind = u; - } - } - } - - /* Did we find everything we were looking for? If so, stop. */ - if ((dld_cache.load.address != 0) - && (dld_cache.load_stub.address != 0) - && (dld_cache.unload.address != 0) - && (dld_cache.unload_stub.address != 0)) - { - dld_cache.is_valid = 1; - break; - } - } - - dld_cache.hook.unwind = find_unwind_entry (dld_cache.hook.address); - dld_cache.hook_stub.unwind = find_unwind_entry (dld_cache.hook_stub.address); - - /* We're prepared not to find some of these symbols, which is why - this function is a "desire" operation, and not a "require". */ -} - -static int -som_in_dynsym_resolve_code (CORE_ADDR pc) -{ - struct unwind_table_entry *u_pc; - - /* Are we in the dld itself? - - ??rehrauer: Large hack -- We'll assume that any address in a - shared text region is the dld's text. This would obviously - fall down if the user attached to a process, whose shlibs - weren't mapped to a (writeable) private region. However, in - that case the debugger probably isn't able to set the fundamental - breakpoint in the dld callback anyways, so this hack should be - safe. */ - - if ((pc & (CORE_ADDR) 0xc0000000) == (CORE_ADDR) 0xc0000000) - return 1; - - /* Cache the address of some symbols that are part of the dynamic - linker, if not already known. */ - - som_solib_desire_dynamic_linker_symbols (); - - /* Are we in the dld callback? Or its export stub? */ - u_pc = find_unwind_entry (pc); - if (u_pc == NULL) - return 0; - - if ((u_pc == dld_cache.hook.unwind) || (u_pc == dld_cache.hook_stub.unwind)) - return 1; - - /* Or the interface of the dld (i.e., "shl_load" or friends)? */ - if ((u_pc == dld_cache.load.unwind) - || (u_pc == dld_cache.unload.unwind) - || (u_pc == dld_cache.unload2.unwind) - || (u_pc == dld_cache.load_stub.unwind) - || (u_pc == dld_cache.unload_stub.unwind)) - return 1; - - /* Apparently this address isn't part of the dld's text. */ - return 0; -} - -static void -som_clear_solib (void) -{ -} - -struct dld_list { - char name[4]; - char info[4]; - char text_addr[4]; - char text_link_addr[4]; - char text_end[4]; - char data_start[4]; - char bss_start[4]; - char data_end[4]; - char got_value[4]; - char next[4]; - char tsd_start_addr_ptr[4]; -}; - -static CORE_ADDR -link_map_start (void) -{ - enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ()); - struct bound_minimal_symbol sym; - CORE_ADDR addr; - gdb_byte buf[4]; - unsigned int dld_flags; - - sym = lookup_minimal_symbol ("__dld_flags", NULL, NULL); - if (!sym.minsym) - error (_("Unable to find __dld_flags symbol in object file.")); - addr = BMSYMBOL_VALUE_ADDRESS (sym); - read_memory (addr, buf, 4); - dld_flags = extract_unsigned_integer (buf, 4, byte_order); - if ((dld_flags & DLD_FLAGS_LISTVALID) == 0) - error (_("__dld_list is not valid according to __dld_flags.")); - - sym = lookup_minimal_symbol ("__dld_list", NULL, NULL); - if (!sym.minsym) - { - /* Older crt0.o files (hpux8) don't have __dld_list as a symbol, - but the data is still available if you know where to look. */ - sym = lookup_minimal_symbol ("__dld_flags", NULL, NULL); - if (!sym.minsym) - { - error (_("Unable to find dynamic library list.")); - return 0; - } - addr = BMSYMBOL_VALUE_ADDRESS (sym) - 8; - } - else - addr = BMSYMBOL_VALUE_ADDRESS (sym); - - read_memory (addr, buf, 4); - addr = extract_unsigned_integer (buf, 4, byte_order); - if (addr == 0) - return 0; - - read_memory (addr, buf, 4); - return extract_unsigned_integer (buf, 4, byte_order); -} - -/* Does this so's name match the main binary? */ -static int -match_main (const char *name) -{ - return strcmp (name, objfile_name (symfile_objfile)) == 0; -} - -static struct so_list * -som_current_sos (void) -{ - enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ()); - CORE_ADDR lm; - struct so_list *head = 0; - struct so_list **link_ptr = &head; - - for (lm = link_map_start (); lm; ) - { - char *namebuf; - CORE_ADDR addr; - struct so_list *newobj; - struct cleanup *old_chain; - int errcode; - struct dld_list dbuf; - gdb_byte tsdbuf[4]; - - newobj = (struct so_list *) xmalloc (sizeof (struct so_list)); - old_chain = make_cleanup (xfree, newobj); - - memset (newobj, 0, sizeof (*newobj)); - newobj->lm_info = xmalloc (sizeof (struct lm_info)); - make_cleanup (xfree, newobj->lm_info); - - read_memory (lm, (gdb_byte *)&dbuf, sizeof (struct dld_list)); - - addr = extract_unsigned_integer ((gdb_byte *)&dbuf.name, - sizeof (dbuf.name), byte_order); - target_read_string (addr, &namebuf, SO_NAME_MAX_PATH_SIZE - 1, &errcode); - if (errcode != 0) - warning (_("Can't read pathname for load map: %s."), - safe_strerror (errcode)); - else - { - strncpy (newobj->so_name, namebuf, SO_NAME_MAX_PATH_SIZE - 1); - newobj->so_name[SO_NAME_MAX_PATH_SIZE - 1] = '\0'; - xfree (namebuf); - strcpy (newobj->so_original_name, newobj->so_name); - } - - if (newobj->so_name[0] && !match_main (newobj->so_name)) - { - struct lm_info *lmi = newobj->lm_info; - unsigned int tmp; - - lmi->lm_addr = lm; - -#define EXTRACT(_fld) \ - extract_unsigned_integer ((gdb_byte *)&dbuf._fld, \ - sizeof (dbuf._fld), byte_order); - - lmi->text_addr = EXTRACT (text_addr); - tmp = EXTRACT (info); - lmi->library_version = (tmp >> 16) & 0xffff; - lmi->bind_mode = (tmp >> 8) & 0xff; - lmi->struct_version = tmp & 0xff; - lmi->text_link_addr = EXTRACT (text_link_addr); - lmi->text_end = EXTRACT (text_end); - lmi->data_start = EXTRACT (data_start); - lmi->bss_start = EXTRACT (bss_start); - lmi->data_end = EXTRACT (data_end); - lmi->got_value = EXTRACT (got_value); - tmp = EXTRACT (tsd_start_addr_ptr); - read_memory (tmp, tsdbuf, 4); - lmi->tsd_start_addr - = extract_unsigned_integer (tsdbuf, 4, byte_order); - -#ifdef SOLIB_SOM_DBG - printf ("\n+ library \"%s\" is described at %s\n", newobj->so_name, - paddress (target_gdbarch (), lm)); - printf (" 'version' is %d\n", newobj->lm_info->struct_version); - printf (" 'bind_mode' is %d\n", newobj->lm_info->bind_mode); - printf (" 'library_version' is %d\n", - newobj->lm_info->library_version); - printf (" 'text_addr' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->text_addr)); - printf (" 'text_link_addr' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->text_link_addr)); - printf (" 'text_end' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->text_end)); - printf (" 'data_start' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->data_start)); - printf (" 'bss_start' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->bss_start)); - printf (" 'data_end' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->data_end)); - printf (" 'got_value' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->got_value)); - printf (" 'tsd_start_addr' is %s\n", - paddress (target_gdbarch (), newobj->lm_info->tsd_start_addr)); -#endif - - newobj->addr_low = lmi->text_addr; - newobj->addr_high = lmi->text_end; - - /* Link the new object onto the list. */ - newobj->next = NULL; - *link_ptr = newobj; - link_ptr = &newobj->next; - } - else - { - free_so (newobj); - } - - lm = EXTRACT (next); - discard_cleanups (old_chain); -#undef EXTRACT - } - - /* TODO: The original somsolib code has logic to detect and eliminate - duplicate entries. Do we need that? */ - - return head; -} - -static int -som_open_symbol_file_object (void *from_ttyp) -{ - enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ()); - CORE_ADDR lm, l_name; - char *filename; - int errcode; - int from_tty = *(int *)from_ttyp; - gdb_byte buf[4]; - struct cleanup *cleanup; - - if (symfile_objfile) - if (!query (_("Attempt to reload symbols from process? "))) - return 0; - - /* First link map member should be the executable. */ - if ((lm = link_map_start ()) == 0) - return 0; /* failed somehow... */ - - /* Read address of name from target memory to GDB. */ - read_memory (lm + offsetof (struct dld_list, name), buf, 4); - - /* Convert the address to host format. Assume that the address is - unsigned. */ - l_name = extract_unsigned_integer (buf, 4, byte_order); - - if (l_name == 0) - return 0; /* No filename. */ - - /* Now fetch the filename from target memory. */ - target_read_string (l_name, &filename, SO_NAME_MAX_PATH_SIZE - 1, &errcode); - - if (errcode) - { - warning (_("failed to read exec filename from attached file: %s"), - safe_strerror (errcode)); - return 0; - } - - cleanup = make_cleanup (xfree, filename); - /* Have a pathname: read the symbol file. */ - symbol_file_add_main (filename, from_tty); - - do_cleanups (cleanup); - return 1; -} - -static void -som_free_so (struct so_list *so) -{ - xfree (so->lm_info); -} - -static CORE_ADDR -som_solib_thread_start_addr (struct so_list *so) -{ - return so->lm_info->tsd_start_addr; -} - -/* Return the GOT value for the shared library in which ADDR belongs. If - ADDR isn't in any known shared library, return zero. */ - -static CORE_ADDR -som_solib_get_got_by_pc (CORE_ADDR addr) -{ - struct so_list *so_list = master_so_list (); - CORE_ADDR got_value = 0; - - while (so_list) - { - if (so_list->lm_info->text_addr <= addr - && so_list->lm_info->text_end > addr) - { - got_value = so_list->lm_info->got_value; - break; - } - so_list = so_list->next; - } - return got_value; -} - -/* Return the address of the handle of the shared library in which - ADDR belongs. If ADDR isn't in any known shared library, return - zero. */ -/* This function is used in initialize_hp_cxx_exception_support in - hppa-hpux-tdep.c. */ - -static CORE_ADDR -som_solib_get_solib_by_pc (CORE_ADDR addr) -{ - struct so_list *so_list = master_so_list (); - - while (so_list) - { - if (so_list->lm_info->text_addr <= addr - && so_list->lm_info->text_end > addr) - { - break; - } - so_list = so_list->next; - } - if (so_list) - return so_list->lm_info->lm_addr; - else - return 0; -} - - -static struct target_so_ops som_so_ops; - -extern initialize_file_ftype _initialize_som_solib; /* -Wmissing-prototypes */ - -void -_initialize_som_solib (void) -{ - som_so_ops.relocate_section_addresses = som_relocate_section_addresses; - som_so_ops.free_so = som_free_so; - som_so_ops.clear_solib = som_clear_solib; - som_so_ops.solib_create_inferior_hook = som_solib_create_inferior_hook; - som_so_ops.special_symbol_handling = som_special_symbol_handling; - som_so_ops.current_sos = som_current_sos; - som_so_ops.open_symbol_file_object = som_open_symbol_file_object; - som_so_ops.in_dynsym_resolve_code = som_in_dynsym_resolve_code; - som_so_ops.bfd_open = solib_bfd_open; -} - -void -som_solib_select (struct gdbarch *gdbarch) -{ - struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch); - - set_solib_ops (gdbarch, &som_so_ops); - tdep->solib_thread_start_addr = som_solib_thread_start_addr; - tdep->solib_get_got_by_pc = som_solib_get_got_by_pc; - tdep->solib_get_solib_by_pc = som_solib_get_solib_by_pc; -} - -/* The rest of these functions are not part of the solib interface; they - are used by somread.c or hppa-hpux-tdep.c. */ - -int -som_solib_section_offsets (struct objfile *objfile, - struct section_offsets *offsets) -{ - struct so_list *so_list = master_so_list (); - - while (so_list) - { - /* Oh what a pain! We need the offsets before so_list->objfile - is valid. The BFDs will never match. Make a best guess. */ - if (strstr (objfile_name (objfile), so_list->so_name)) - { - asection *private_section; - struct obj_section *sect; - - /* The text offset is easy. */ - offsets->offsets[SECT_OFF_TEXT (objfile)] - = (so_list->lm_info->text_addr - - so_list->lm_info->text_link_addr); - - /* We should look at presumed_dp in the SOM header, but - that's not easily available. This should be OK though. */ - private_section = bfd_get_section_by_name (objfile->obfd, - "$PRIVATE$"); - if (!private_section) - { - warning (_("Unable to find $PRIVATE$ in shared library!")); - offsets->offsets[SECT_OFF_DATA (objfile)] = 0; - offsets->offsets[SECT_OFF_BSS (objfile)] = 0; - return 1; - } - if (objfile->sect_index_data != -1) - { - offsets->offsets[SECT_OFF_DATA (objfile)] - = (so_list->lm_info->data_start - private_section->vma); - if (objfile->sect_index_bss != -1) - offsets->offsets[SECT_OFF_BSS (objfile)] - = ANOFFSET (offsets, SECT_OFF_DATA (objfile)); - } - - ALL_OBJFILE_OSECTIONS (objfile, sect) - { - flagword flags = bfd_get_section_flags (objfile->obfd, - sect->the_bfd_section); - - if ((flags & SEC_CODE) != 0) - offsets->offsets[sect->the_bfd_section->index] - = offsets->offsets[SECT_OFF_TEXT (objfile)]; - else - offsets->offsets[sect->the_bfd_section->index] - = offsets->offsets[SECT_OFF_DATA (objfile)]; - } - - return 1; - } - so_list = so_list->next; - } - return 0; -} |