2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/dwarf2read.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/dwarf2read.c 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/dwarf2read.c 2006-07-11 02:56:58.000000000 -0300
|
|
|
|
|
@@ -1208,7 +1208,7 @@ dwarf2_build_psymtabs (struct objfile *o
|
2005-01-11 00:15:58 +00:00
|
|
|
|
else
|
|
|
|
|
dwarf2_per_objfile->loc_buffer = NULL;
|
|
|
|
|
|
|
|
|
|
- if (mainline
|
|
|
|
|
+ if ((mainline == 1)
|
|
|
|
|
|| (objfile->global_psymbols.size == 0
|
|
|
|
|
&& objfile->static_psymbols.size == 0))
|
|
|
|
|
{
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/auxv.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/auxv.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/auxv.c 2006-07-11 02:47:11.000000000 -0300
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -119,7 +119,7 @@ target_auxv_read (struct target_ops *ops
|
|
|
|
|
Return 1 if an entry was read into *TYPEP and *VALP. */
|
|
|
|
|
int
|
2006-07-11 06:33:02 +00:00
|
|
|
|
target_auxv_parse (struct target_ops *ops, gdb_byte **readptr,
|
|
|
|
|
- gdb_byte *endptr, CORE_ADDR *typep, CORE_ADDR *valp)
|
|
|
|
|
+ gdb_byte *endptr, ULONGEST *typep, CORE_ADDR *valp)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
{
|
|
|
|
|
const int sizeof_auxv_field = TYPE_LENGTH (builtin_type_void_data_ptr);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte *ptr = *readptr;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -144,9 +144,10 @@ target_auxv_parse (struct target_ops *op
|
|
|
|
|
an error getting the information. On success, return 1 after
|
|
|
|
|
storing the entry's value field in *VALP. */
|
|
|
|
|
int
|
|
|
|
|
-target_auxv_search (struct target_ops *ops, CORE_ADDR match, CORE_ADDR *valp)
|
|
|
|
|
+target_auxv_search (struct target_ops *ops, ULONGEST match, CORE_ADDR *valp)
|
|
|
|
|
{
|
|
|
|
|
- CORE_ADDR type, val;
|
|
|
|
|
+ CORE_ADDR val;
|
|
|
|
|
+ ULONGEST at_type;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte *data;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
int n = target_auxv_read (ops, &data);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte *ptr = data;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -156,10 +157,10 @@ target_auxv_search (struct target_ops *o
|
|
|
|
|
return n;
|
|
|
|
|
|
|
|
|
|
while (1)
|
|
|
|
|
- switch (target_auxv_parse (ops, &ptr, data + n, &type, &val))
|
|
|
|
|
+ switch (target_auxv_parse (ops, &ptr, data + n, &at_type, &val))
|
|
|
|
|
{
|
|
|
|
|
case 1: /* Here's an entry, check it. */
|
|
|
|
|
- if (type == match)
|
|
|
|
|
+ if (at_type == match)
|
|
|
|
|
{
|
|
|
|
|
xfree (data);
|
|
|
|
|
*valp = val;
|
|
|
|
|
@@ -182,7 +183,8 @@ target_auxv_search (struct target_ops *o
|
|
|
|
|
int
|
|
|
|
|
fprint_target_auxv (struct ui_file *file, struct target_ops *ops)
|
|
|
|
|
{
|
|
|
|
|
- CORE_ADDR type, val;
|
|
|
|
|
+ CORE_ADDR val;
|
|
|
|
|
+ ULONGEST at_type;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte *data;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
int len = target_auxv_read (ops, &data);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte *ptr = data;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -191,14 +193,14 @@ fprint_target_auxv (struct ui_file *file
|
|
|
|
|
if (len <= 0)
|
|
|
|
|
return len;
|
|
|
|
|
|
|
|
|
|
- while (target_auxv_parse (ops, &ptr, data + len, &type, &val) > 0)
|
|
|
|
|
+ while (target_auxv_parse (ops, &ptr, data + len, &at_type, &val) > 0)
|
|
|
|
|
{
|
|
|
|
|
extern int addressprint;
|
|
|
|
|
const char *name = "???";
|
|
|
|
|
const char *description = "";
|
|
|
|
|
enum { dec, hex, str } flavor = hex;
|
|
|
|
|
|
|
|
|
|
- switch (type)
|
|
|
|
|
+ switch (at_type)
|
|
|
|
|
{
|
|
|
|
|
#define TAG(tag, text, kind) \
|
|
|
|
|
case tag: name = #tag; description = text; flavor = kind; break
|
|
|
|
|
@@ -249,7 +251,7 @@ fprint_target_auxv (struct ui_file *file
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf_filtered (file, "%-4s %-20s %-30s ",
|
|
|
|
|
- paddr_d (type), name, description);
|
|
|
|
|
+ paddr_d (at_type), name, description);
|
|
|
|
|
switch (flavor)
|
|
|
|
|
{
|
|
|
|
|
case dec:
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/auxv.h
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/auxv.h 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/auxv.h 2006-07-11 02:47:11.000000000 -0300
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -43,14 +43,14 @@ extern LONGEST target_auxv_read (struct
|
|
|
|
|
Return 1 if an entry was read into *TYPEP and *VALP. */
|
|
|
|
|
extern int target_auxv_parse (struct target_ops *ops,
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte **readptr, gdb_byte *endptr,
|
2005-01-11 00:15:58 +00:00
|
|
|
|
- CORE_ADDR *typep, CORE_ADDR *valp);
|
|
|
|
|
+ ULONGEST *typep, CORE_ADDR *valp);
|
|
|
|
|
|
|
|
|
|
/* Extract the auxiliary vector entry with a_type matching MATCH.
|
|
|
|
|
Return zero if no such entry was found, or -1 if there was
|
|
|
|
|
an error getting the information. On success, return 1 after
|
|
|
|
|
storing the entry's value field in *VALP. */
|
|
|
|
|
extern int target_auxv_search (struct target_ops *ops,
|
|
|
|
|
- CORE_ADDR match, CORE_ADDR *valp);
|
|
|
|
|
+ ULONGEST match, CORE_ADDR *valp);
|
|
|
|
|
|
|
|
|
|
/* Print the contents of the target's AUXV on the specified file. */
|
|
|
|
|
extern int fprint_target_auxv (struct ui_file *file, struct target_ops *ops);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/breakpoint.h
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/breakpoint.h 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/breakpoint.h 2006-07-11 02:47:11.000000000 -0300
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -159,6 +159,7 @@ enum enable_state
|
|
|
|
|
automatically enabled and reset when the call
|
|
|
|
|
"lands" (either completes, or stops at another
|
|
|
|
|
eventpoint). */
|
|
|
|
|
+ bp_startup_disabled,
|
|
|
|
|
bp_permanent /* There is a breakpoint instruction hard-wired into
|
|
|
|
|
the target's code. Don't try to write another
|
|
|
|
|
breakpoint instruction on top of it, or restore
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -807,8 +808,12 @@ extern void remove_thread_event_breakpoi
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
extern void disable_breakpoints_in_shlibs (int silent);
|
|
|
|
|
|
|
|
|
|
+extern void disable_breakpoints_at_startup (int silent);
|
|
|
|
|
+
|
|
|
|
|
extern void re_enable_breakpoints_in_shlibs (void);
|
|
|
|
|
|
|
|
|
|
+void re_enable_breakpoints_at_startup (void);
|
|
|
|
|
+
|
|
|
|
|
extern void create_solib_load_event_breakpoint (char *, int, char *, char *);
|
|
|
|
|
|
|
|
|
|
extern void create_solib_unload_event_breakpoint (char *, int,
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/symfile-mem.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/symfile-mem.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/symfile-mem.c 2006-07-11 02:56:55.000000000 -0300
|
|
|
|
|
@@ -110,7 +110,7 @@ symbol_file_add_from_memory (struct bfd
|
2005-01-11 00:15:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
objf = symbol_file_add_from_bfd (nbfd, from_tty,
|
|
|
|
|
- sai, 0, OBJF_SHARED);
|
|
|
|
|
+ sai, 2, OBJF_SHARED);
|
|
|
|
|
|
|
|
|
|
/* This might change our ideas about frames already looked at. */
|
|
|
|
|
reinit_frame_cache ();
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/infrun.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/infrun.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/infrun.c 2006-07-11 02:56:58.000000000 -0300
|
|
|
|
|
@@ -2230,6 +2230,11 @@ process_event_stop_test:
|
2005-01-11 00:15:58 +00:00
|
|
|
|
code segments in shared libraries might be mapped in now. */
|
|
|
|
|
re_enable_breakpoints_in_shlibs ();
|
|
|
|
|
|
|
|
|
|
+ /* For PIE executables, we dont really know where the
|
|
|
|
|
+ breakpoints are going to be until we start up the
|
|
|
|
|
+ inferior. */
|
|
|
|
|
+ re_enable_breakpoints_at_startup ();
|
|
|
|
|
+
|
|
|
|
|
/* If requested, stop when the dynamic linker notifies
|
|
|
|
|
gdb of events. This allows the user to get control
|
|
|
|
|
and place breakpoints in initializer routines for
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/objfiles.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/objfiles.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/objfiles.c 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
@@ -47,6 +47,9 @@
|
2005-01-11 00:15:58 +00:00
|
|
|
|
#include "dictionary.h"
|
2006-07-11 06:33:02 +00:00
|
|
|
|
#include "source.h"
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+#include "auxv.h"
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+#include "elf/common.h"
|
|
|
|
|
+
|
|
|
|
|
/* Prototypes for local functions */
|
|
|
|
|
|
|
|
|
|
static void objfile_alloc_data (struct objfile *objfile);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -258,7 +261,19 @@ init_entry_point_info (struct objfile *o
|
2005-01-11 00:15:58 +00:00
|
|
|
|
CORE_ADDR
|
|
|
|
|
entry_point_address (void)
|
|
|
|
|
{
|
|
|
|
|
- return symfile_objfile ? symfile_objfile->ei.entry_point : 0;
|
|
|
|
|
+ int ret;
|
|
|
|
|
+ CORE_ADDR entry_addr;
|
|
|
|
|
+
|
|
|
|
|
+ /* Find the address of the entry point of the program from the
|
|
|
|
|
+ auxv vector. */
|
|
|
|
|
+ ret = target_auxv_search (¤t_target, AT_ENTRY, &entry_addr);
|
|
|
|
|
+ if (ret == 1)
|
|
|
|
|
+ return entry_addr;
|
|
|
|
|
+ /*if (ret == 0 || ret == -1)*/
|
|
|
|
|
+ else
|
|
|
|
|
+ {
|
|
|
|
|
+ return symfile_objfile ? symfile_objfile->ei.entry_point : 0;
|
|
|
|
|
+ }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Create the terminating entry of OBJFILE's minimal symbol table.
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/solib-svr4.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/solib-svr4.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/solib-svr4.c 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
@@ -34,6 +34,8 @@
|
2005-01-11 00:15:58 +00:00
|
|
|
|
#include "gdbcore.h"
|
|
|
|
|
#include "target.h"
|
|
|
|
|
#include "inferior.h"
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+#include "auxv.h"
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+#include "command.h"
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
#include "gdb_assert.h"
|
|
|
|
|
|
|
|
|
|
@@ -254,7 +256,9 @@ static CORE_ADDR breakpoint_addr; /* Add
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
/* Local function prototypes */
|
|
|
|
|
|
|
|
|
|
+#if 0
|
|
|
|
|
static int match_main (char *);
|
|
|
|
|
+#endif
|
|
|
|
|
|
|
|
|
|
static CORE_ADDR bfd_lookup_symbol (bfd *, char *, flagword);
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -376,22 +380,79 @@ elf_locate_base (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
{
|
|
|
|
|
struct bfd_section *dyninfo_sect;
|
|
|
|
|
int dyninfo_sect_size;
|
|
|
|
|
- CORE_ADDR dyninfo_addr;
|
|
|
|
|
+ CORE_ADDR dyninfo_addr, relocated_dyninfo_addr, entry_addr;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
gdb_byte *buf;
|
|
|
|
|
gdb_byte *bufend;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
int arch_size;
|
|
|
|
|
+ int ret;
|
|
|
|
|
+
|
|
|
|
|
+ /* Find the address of the entry point of the program from the
|
|
|
|
|
+ auxv vector. */
|
|
|
|
|
+ ret = target_auxv_search (¤t_target, AT_ENTRY, &entry_addr);
|
|
|
|
|
+
|
|
|
|
|
+ if (ret == 0 || ret == -1)
|
|
|
|
|
+ {
|
|
|
|
|
+ /* No auxv info, maybe an older kernel. Fake our way through. */
|
|
|
|
|
+ entry_addr = bfd_get_start_address (exec_bfd);
|
|
|
|
|
+
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: program entry address not found. Using bfd's 0x%s for %s\n",
|
|
|
|
|
+ paddr_nz (entry_addr), exec_bfd->filename);
|
|
|
|
|
+ }
|
|
|
|
|
+ else
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: found program entry address 0x%s for %s\n",
|
|
|
|
|
+ paddr_nz (entry_addr), exec_bfd->filename);
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
|
|
/* Find the start address of the .dynamic section. */
|
|
|
|
|
dyninfo_sect = bfd_get_section_by_name (exec_bfd, ".dynamic");
|
|
|
|
|
if (dyninfo_sect == NULL)
|
|
|
|
|
- return 0;
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: .dynamic section not found in %s -- return now\n",
|
|
|
|
|
+ exec_bfd->filename);
|
|
|
|
|
+ return 0;
|
|
|
|
|
+ }
|
|
|
|
|
+ else
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: .dynamic section found in %s\n",
|
|
|
|
|
+ exec_bfd->filename);
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
dyninfo_addr = bfd_section_vma (exec_bfd, dyninfo_sect);
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: unrelocated .dynamic addr 0x%s\n",
|
|
|
|
|
+ paddr_nz (dyninfo_addr));
|
|
|
|
|
+
|
|
|
|
|
+ relocated_dyninfo_addr = dyninfo_addr
|
|
|
|
|
+ + entry_addr - bfd_get_start_address(exec_bfd);
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: relocated .dyn addr 0x%s for %s\n",
|
|
|
|
|
+ paddr_nz(relocated_dyninfo_addr), exec_bfd->filename);
|
|
|
|
|
|
|
|
|
|
/* Read in .dynamic section, silently ignore errors. */
|
|
|
|
|
dyninfo_sect_size = bfd_section_size (exec_bfd, dyninfo_sect);
|
|
|
|
|
buf = alloca (dyninfo_sect_size);
|
|
|
|
|
- if (target_read_memory (dyninfo_addr, buf, dyninfo_sect_size))
|
|
|
|
|
- return 0;
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: read in .dynamic section\n");
|
|
|
|
|
+ if (target_read_memory (relocated_dyninfo_addr, buf, dyninfo_sect_size))
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: couldn't read .dynamic section at 0x%s -- return now\n",
|
|
|
|
|
+ paddr_nz (relocated_dyninfo_addr));
|
|
|
|
|
+ return 0;
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
|
|
/* Find the DT_DEBUG entry in the the .dynamic section.
|
|
|
|
|
For mips elf we look for DT_MIPS_RLD_MAP, mips elf apparently has
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -418,6 +479,10 @@ elf_locate_base (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
{
|
|
|
|
|
dyn_ptr = bfd_h_get_32 (exec_bfd,
|
|
|
|
|
(bfd_byte *) x_dynp->d_un.d_ptr);
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "elf_locate_base: DT_DEBUG entry has value 0x%s -- return now\n",
|
|
|
|
|
+ paddr_nz (dyn_ptr));
|
|
|
|
|
return dyn_ptr;
|
|
|
|
|
}
|
|
|
|
|
else if (dyn_tag == DT_MIPS_RLD_MAP)
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -543,6 +608,10 @@ solib_svr4_r_map (void)
|
|
|
|
|
{
|
|
|
|
|
struct link_map_offsets *lmo = svr4_fetch_link_map_offsets ();
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ "solib_svr4_r_map: read at 0x%s\n",
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ paddr_nz (debug_base + lmo->r_map_offset));
|
2006-07-11 06:33:02 +00:00
|
|
|
|
return read_memory_typed_address (debug_base + lmo->r_map_offset,
|
|
|
|
|
builtin_type_void_data_ptr);
|
|
|
|
|
}
|
|
|
|
|
@@ -669,6 +738,11 @@ svr4_current_sos (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
struct so_list **link_ptr = &head;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
CORE_ADDR ldsomap = 0;
|
|
|
|
|
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: exec_bfd %s\n",
|
|
|
|
|
+ exec_bfd->filename);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* Make sure we've looked up the inferior's dynamic linker's base
|
|
|
|
|
structure. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
if (! debug_base)
|
|
|
|
|
@@ -678,11 +752,21 @@ svr4_current_sos (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* If we can't find the dynamic linker's base structure, this
|
|
|
|
|
must not be a dynamically linked executable. Hmm. */
|
|
|
|
|
if (! debug_base)
|
|
|
|
|
- return 0;
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: no DT_DEBUG found in %s -- return now\n",
|
|
|
|
|
+ exec_bfd->filename);
|
|
|
|
|
+ return 0;
|
|
|
|
|
+ }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Walk the inferior's link map list, and build our list of
|
|
|
|
|
`struct so_list' nodes. */
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: walk link map in %s\n",
|
|
|
|
|
+ exec_bfd->filename);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
lm = solib_svr4_r_map ();
|
2005-01-11 00:15:58 +00:00
|
|
|
|
while (lm)
|
|
|
|
|
{
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -697,23 +781,103 @@ svr4_current_sos (void)
|
|
|
|
|
new->lm_info->lm = xzalloc (lmo->link_map_size);
|
2005-01-11 00:15:58 +00:00
|
|
|
|
make_cleanup (xfree, new->lm_info->lm);
|
|
|
|
|
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: read lm at 0x%s\n", paddr_nz(lm));
|
|
|
|
|
read_memory (lm, new->lm_info->lm, lmo->link_map_size);
|
|
|
|
|
|
|
|
|
|
lm = LM_NEXT (new);
|
|
|
|
|
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: is first link entry? %d\n",
|
|
|
|
|
+ IGNORE_FIRST_LINK_MAP_ENTRY (new));
|
|
|
|
|
+
|
|
|
|
|
/* For SVR4 versions, the first entry in the link map is for the
|
|
|
|
|
inferior executable, so we must ignore it. For some versions of
|
|
|
|
|
SVR4, it has no name. For others (Solaris 2.3 for example), it
|
|
|
|
|
does have a name, so we can no longer use a missing name to
|
|
|
|
|
decide when to ignore it. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
if (IGNORE_FIRST_LINK_MAP_ENTRY (new) && ldsomap == 0)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
- free_so (new);
|
|
|
|
|
+ {
|
|
|
|
|
+ /* It is the first link map entry, i.e. it is the main executable. */
|
|
|
|
|
+
|
|
|
|
|
+ if (bfd_get_start_address (exec_bfd) == entry_point_address ())
|
|
|
|
|
+ {
|
|
|
|
|
+ /* Non-pie case, main executable has not been relocated. */
|
|
|
|
|
+ free_so (new);
|
|
|
|
|
+ }
|
|
|
|
|
+ else
|
|
|
|
|
+ {
|
|
|
|
|
+ /* Pie case, main executable has been relocated. */
|
|
|
|
|
+ struct so_list *gdb_solib;
|
|
|
|
|
+
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: Processing first link map entry\n");
|
|
|
|
|
+ strncpy (new->so_name, exec_bfd->filename,
|
|
|
|
|
+ SO_NAME_MAX_PATH_SIZE - 1);
|
|
|
|
|
+ new->so_name[SO_NAME_MAX_PATH_SIZE - 1] = '\0';
|
|
|
|
|
+ strcpy (new->so_original_name, new->so_name);
|
|
|
|
|
+ /*new->main = 1;*/
|
|
|
|
|
+ new->main_relocated = 0;
|
|
|
|
|
+
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ {
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: Processing nameless DSO\n");
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: adding name %s\n",
|
|
|
|
|
+ new->so_name);
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ for (gdb_solib = master_so_list ();
|
|
|
|
|
+ gdb_solib;
|
|
|
|
|
+ gdb_solib = gdb_solib->next)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: compare gdb %s and new %s\n",
|
|
|
|
|
+ gdb_solib->so_name, new->so_name);
|
|
|
|
|
+ if (strcmp (gdb_solib->so_name, new->so_name) == 0)
|
|
|
|
|
+ if (gdb_solib->main_relocated)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: found main relocated\n");
|
|
|
|
|
+ break;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ if ((gdb_solib && !gdb_solib->main_relocated) || (!gdb_solib))
|
|
|
|
|
+ {
|
|
|
|
|
+ add_to_target_sections (0 /*from_tty*/, ¤t_target, new);
|
|
|
|
|
+ new->main = 1;
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ /* We need this in the list of shared libs we return because
|
|
|
|
|
+ solib_add_stub will loop through it and add the symbol file. */
|
|
|
|
|
+ new->next = 0;
|
|
|
|
|
+ *link_ptr = new;
|
|
|
|
|
+ link_ptr = &new->next;
|
|
|
|
|
+ }
|
|
|
|
|
+ } /* End of IGNORE_FIRST_LINK_MAP_ENTRY */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
+ /* This is not the first link map entry, i.e. is not the main
|
|
|
|
|
+ executable. Note however that it could be the DSO supplied on
|
|
|
|
|
+ certain systems (i.e. Linux 2.6) containing information about
|
|
|
|
|
+ the vsyscall page. We must ignore such entry. This entry is
|
|
|
|
|
+ nameless (just like the one for the main executable, sigh). */
|
|
|
|
|
+
|
|
|
|
|
int errcode;
|
|
|
|
|
char *buffer;
|
|
|
|
|
|
|
|
|
|
/* Extract this shared object's name. */
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: read LM_NAME\n");
|
|
|
|
|
+
|
|
|
|
|
target_read_string (LM_NAME (new), &buffer,
|
|
|
|
|
SO_NAME_MAX_PATH_SIZE - 1, &errcode);
|
|
|
|
|
if (errcode != 0)
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -721,22 +885,37 @@ svr4_current_sos (void)
|
|
|
|
|
safe_strerror (errcode));
|
2005-01-11 00:15:58 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
2006-07-11 06:33:02 +00:00
|
|
|
|
- strncpy (new->so_name, buffer, SO_NAME_MAX_PATH_SIZE - 1);
|
|
|
|
|
- new->so_name[SO_NAME_MAX_PATH_SIZE - 1] = '\0';
|
|
|
|
|
- xfree (buffer);
|
|
|
|
|
- strcpy (new->so_original_name, new->so_name);
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- /* If this entry has no name, or its name matches the name
|
|
|
|
|
- for the main executable, don't include it in the list. */
|
|
|
|
|
- if (! new->so_name[0]
|
|
|
|
|
- || match_main (new->so_name))
|
|
|
|
|
- free_so (new);
|
|
|
|
|
- else
|
|
|
|
|
- {
|
|
|
|
|
- new->next = 0;
|
|
|
|
|
- *link_ptr = new;
|
|
|
|
|
- link_ptr = &new->next;
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: LM_NAME is <%s>\n",
|
|
|
|
|
+ buffer);
|
|
|
|
|
+ /* The name could be empty, in which case it is the
|
|
|
|
|
+ system supplied DSO. */
|
|
|
|
|
+ if (strcmp (buffer, "") == 0)
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ {
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ xfree (buffer);
|
|
|
|
|
+ free_so (new);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ }
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ else
|
|
|
|
|
+ {
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ strncpy (new->so_name, buffer, SO_NAME_MAX_PATH_SIZE - 1);
|
|
|
|
|
+ new->so_name[SO_NAME_MAX_PATH_SIZE - 1] = '\0';
|
|
|
|
|
+ xfree (buffer);
|
|
|
|
|
+ strcpy (new->so_original_name, new->so_name);
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ {
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: Processing DSO: %s\n",
|
|
|
|
|
+ new->so_name);
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: first link entry %d\n",
|
|
|
|
|
+ IGNORE_FIRST_LINK_MAP_ENTRY (new));
|
|
|
|
|
+ }
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ new->next = 0;
|
|
|
|
|
+ *link_ptr = new;
|
|
|
|
|
+ link_ptr = &new->next;
|
|
|
|
|
+ }
|
|
|
|
|
}
|
2005-01-11 00:15:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -750,6 +929,11 @@ svr4_current_sos (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
discard_cleanups (old_chain);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_current_sos: ENDS %s\n",
|
|
|
|
|
+ exec_bfd->filename);
|
|
|
|
|
+
|
|
|
|
|
return head;
|
|
|
|
|
}
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -826,7 +1010,7 @@ svr4_fetch_objfile_link_map (struct objf
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* On some systems, the only way to recognize the link map entry for
|
|
|
|
|
the main executable file is by looking at its name. Return
|
|
|
|
|
non-zero iff SONAME matches one of the known main executable names. */
|
|
|
|
|
-
|
|
|
|
|
+#if 0
|
|
|
|
|
static int
|
|
|
|
|
match_main (char *soname)
|
|
|
|
|
{
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -840,6 +1024,7 @@ match_main (char *soname)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
+#endif
|
|
|
|
|
|
|
|
|
|
/* Return 1 if PC lies in the dynamic symbol resolution code of the
|
|
|
|
|
SVR4 run time loader. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -939,6 +1124,11 @@ enable_break (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* Find the .interp section; if not found, warn the user and drop
|
|
|
|
|
into the old breakpoint at symbol code. */
|
|
|
|
|
interp_sect = bfd_get_section_by_name (exec_bfd, ".interp");
|
|
|
|
|
+
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "enable_break: search for .interp in %s\n",
|
|
|
|
|
+ exec_bfd->filename);
|
|
|
|
|
if (interp_sect)
|
|
|
|
|
{
|
|
|
|
|
unsigned int interp_sect_size;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -972,6 +1162,9 @@ enable_break (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
if (tmp_fd >= 0)
|
2006-07-11 06:33:02 +00:00
|
|
|
|
tmp_bfd = bfd_fopen (tmp_pathname, gnutarget, FOPEN_RB, tmp_fd);
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "enable_break: opening %s\n", tmp_pathname);
|
|
|
|
|
if (tmp_bfd == NULL)
|
|
|
|
|
goto bkpt_at_symbol;
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -1052,6 +1245,9 @@ enable_break (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
if (sym_addr != 0)
|
|
|
|
|
{
|
|
|
|
|
create_solib_event_breakpoint (load_addr + sym_addr);
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "enable_break: solib bp set\n");
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -1311,6 +1507,8 @@ svr4_solib_create_inferior_hook (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
while (stop_signal != TARGET_SIGNAL_TRAP);
|
|
|
|
|
stop_soon = NO_STOP_QUIETLY;
|
|
|
|
|
#endif /* defined(_SCO_DS) */
|
|
|
|
|
+
|
|
|
|
|
+ disable_breakpoints_at_startup (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -1493,6 +1691,75 @@ svr4_lp64_fetch_link_map_offsets (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
return lmp;
|
|
|
|
|
}
|
|
|
|
|
+void
|
|
|
|
|
+info_linkmap_command (char *cmd, int from_tty)
|
|
|
|
|
+{
|
|
|
|
|
+ CORE_ADDR lm;
|
|
|
|
|
+
|
|
|
|
|
+ /* Make sure we've looked up the inferior's dynamic linker's base
|
|
|
|
|
+ structure. */
|
|
|
|
|
+ if (! debug_base)
|
|
|
|
|
+ {
|
|
|
|
|
+ debug_base = locate_base ();
|
|
|
|
|
+
|
|
|
|
|
+ /* If we can't find the dynamic linker's base structure, this
|
|
|
|
|
+ must not be a dynamically linked executable. Hmm. */
|
|
|
|
|
+ if (! debug_base)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_print_linkmap: no DT_DEBUG found in %s -- return now\n",
|
|
|
|
|
+ exec_bfd->filename);
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+
|
|
|
|
|
+ /* Walk the inferior's link map list, and print the info. */
|
|
|
|
|
+
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ lm = solib_svr4_r_map ();
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ while (lm)
|
|
|
|
|
+ {
|
|
|
|
|
+ int errcode;
|
|
|
|
|
+ char *buffer;
|
|
|
|
|
+ CORE_ADDR load_addr;
|
|
|
|
|
+
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ struct link_map_offsets *lmo = svr4_fetch_link_map_offsets ();
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ struct so_list *new
|
|
|
|
|
+ = (struct so_list *) xmalloc (sizeof (struct so_list));
|
|
|
|
|
+ struct cleanup *old_chain = make_cleanup (xfree, new);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ memset (new, 0, sizeof (*new));
|
|
|
|
|
+
|
|
|
|
|
+ new->lm_info = xmalloc (sizeof (struct lm_info));
|
|
|
|
|
+ make_cleanup (xfree, new->lm_info);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ new->lm_info->lm = xmalloc (lmo->link_map_size);
|
|
|
|
|
+ make_cleanup (xfree, new->lm_info->lm);
|
|
|
|
|
+ memset (new->lm_info->lm, 0, lmo->link_map_size);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "svr4_print_linkmap: read lm at 0x%s\n", paddr_nz(lm));
|
|
|
|
|
+ read_memory (lm, new->lm_info->lm, lmo->link_map_size);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ lm = LM_NEXT (new);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ /* Load address. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ load_addr = LM_ADDR_CHECK (new, NULL);
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ /* Shared object's name. */
|
|
|
|
|
+ target_read_string (LM_NAME (new), &buffer,
|
|
|
|
|
+ SO_NAME_MAX_PATH_SIZE - 1, &errcode);
|
|
|
|
|
+ make_cleanup (xfree, buffer);
|
|
|
|
|
+ if (errcode != 0)
|
|
|
|
|
+ {
|
|
|
|
|
+ warning ("svr4_print_linkmap: Can't read pathname for load map: %s\n",
|
|
|
|
|
+ safe_strerror (errcode));
|
|
|
|
|
+ }
|
|
|
|
|
+ fprintf_filtered (gdb_stdout, "%-8s %-30s\n", paddr(load_addr), buffer);
|
|
|
|
|
+ do_cleanups (old_chain);
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static struct target_so_ops svr4_so_ops;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -1515,4 +1782,8 @@ _initialize_svr4_solib (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
/* FIXME: Don't do this here. *_gdbarch_init() should set so_ops. */
|
|
|
|
|
current_target_so_ops = &svr4_so_ops;
|
|
|
|
|
+
|
|
|
|
|
+ add_info ("linkmap", info_linkmap_command,
|
|
|
|
|
+ "Display the inferior's linkmap.");
|
|
|
|
|
+
|
|
|
|
|
}
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/varobj.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/varobj.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/varobj.c 2006-07-11 02:56:57.000000000 -0300
|
|
|
|
|
@@ -870,6 +870,62 @@ varobj_list (struct varobj ***varlist)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
return rootcount;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
+void
|
|
|
|
|
+varobj_refresh (void)
|
|
|
|
|
+{
|
|
|
|
|
+ struct varobj *var;
|
|
|
|
|
+ struct varobj_root *croot;
|
|
|
|
|
+ int mycount = rootcount;
|
|
|
|
|
+ char * name;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ croot = rootlist;
|
|
|
|
|
+ while ((croot != NULL) && (mycount > 0))
|
|
|
|
|
+ {
|
|
|
|
|
+ var = croot->rootvar;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ /* Get rid of the memory for the old expression. This also
|
|
|
|
|
+ leaves var->root->exp == NULL, which is ok for the parsing
|
|
|
|
|
+ below. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ free_current_contents (&var->root->exp);
|
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ value_free (var->value);
|
|
|
|
|
+ var->type = NULL;
|
|
|
|
|
+
|
|
|
|
|
+ name = xstrdup (var->name);
|
|
|
|
|
+
|
|
|
|
|
+ /* Reparse the expression. Wrap the call to parse expression,
|
|
|
|
|
+ so we can return a sensible error. */
|
|
|
|
|
+ if (!gdb_parse_exp_1 (&name, var->root->valid_block, 0, &var->root->exp))
|
|
|
|
|
+ {
|
|
|
|
|
+ return;
|
|
|
|
|
+ }
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ /* We definitively need to catch errors here.
|
|
|
|
|
+ If evaluate_expression succeeds we got the value we wanted.
|
|
|
|
|
+ But if it fails, we still go on with a call to evaluate_type() */
|
|
|
|
|
+ if (gdb_evaluate_expression (var->root->exp, &var->value))
|
|
|
|
|
+ {
|
|
|
|
|
+ /* no error */
|
|
|
|
|
+ release_value (var->value);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ if (value_lazy (var->value))
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ gdb_value_fetch_lazy (var->value);
|
|
|
|
|
+ }
|
|
|
|
|
+ else
|
|
|
|
|
+ var->value = evaluate_type (var->root->exp);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
|
|
|
|
+ var->type = value_type (var->value);
|
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ mycount--;
|
|
|
|
|
+ croot = croot->next;
|
|
|
|
|
+ }
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+ if (mycount || (croot != NULL))
|
|
|
|
|
+ warning
|
|
|
|
|
+ ("varobj_refresh: assertion failed - wrong tally of root vars (%d:%d)",
|
|
|
|
|
+ rootcount, mycount);
|
|
|
|
|
+}
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+
|
|
|
|
|
/* Update the values for a variable and its children. This is a
|
|
|
|
|
two-pronged attack. First, re-parse the value for the root's
|
|
|
|
|
expression to see if it's changed. Then go all the way
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/solist.h
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/solist.h 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/solist.h 2006-07-11 02:47:11.000000000 -0300
|
2005-01-11 00:15:58 +00:00
|
|
|
|
@@ -62,6 +62,8 @@ struct so_list
|
|
|
|
|
bfd *abfd;
|
|
|
|
|
char symbols_loaded; /* flag: symbols read in yet? */
|
|
|
|
|
char from_tty; /* flag: print msgs? */
|
|
|
|
|
+ char main; /* flag: is this the main executable? */
|
|
|
|
|
+ char main_relocated; /* flag: has it been relocated yet? */
|
|
|
|
|
struct objfile *objfile; /* objfile for loaded lib */
|
|
|
|
|
struct section_table *sections;
|
|
|
|
|
struct section_table *sections_end;
|
|
|
|
|
@@ -113,9 +115,15 @@ void free_so (struct so_list *so);
|
|
|
|
|
/* Return address of first so_list entry in master shared object list. */
|
|
|
|
|
struct so_list *master_so_list (void);
|
|
|
|
|
|
|
|
|
|
+/* Return address of first so_list entry in master shared object list. */
|
|
|
|
|
+struct so_list *master_so_list (void);
|
|
|
|
|
+
|
|
|
|
|
/* Find solib binary file and open it. */
|
|
|
|
|
extern int solib_open (char *in_pathname, char **found_pathname);
|
|
|
|
|
|
|
|
|
|
+/* Add the list of sections in so_list to the target to_sections. */
|
|
|
|
|
+extern void add_to_target_sections (int, struct target_ops *, struct so_list *);
|
|
|
|
|
+
|
|
|
|
|
/* FIXME: gdbarch needs to control this variable */
|
|
|
|
|
extern struct target_so_ops *current_target_so_ops;
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -126,4 +134,6 @@ extern struct target_so_ops *current_tar
|
|
|
|
|
#define TARGET_SO_IN_DYNSYM_RESOLVE_CODE \
|
|
|
|
|
(current_target_so_ops->in_dynsym_resolve_code)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
+/* Controls the printing of debugging output. */
|
|
|
|
|
+extern int debug_solib;
|
|
|
|
|
#endif
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/varobj.h
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/varobj.h 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/varobj.h 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
@@ -99,4 +99,6 @@ extern int varobj_list (struct varobj **
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
extern int varobj_update (struct varobj **varp, struct varobj ***changelist);
|
|
|
|
|
|
|
|
|
|
+extern void varobj_refresh(void);
|
|
|
|
|
+
|
|
|
|
|
#endif /* VAROBJ_H */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/symfile.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/symfile.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/symfile.c 2006-07-11 02:56:57.000000000 -0300
|
|
|
|
|
@@ -49,6 +49,7 @@
|
2005-01-11 00:15:58 +00:00
|
|
|
|
#include "readline/readline.h"
|
|
|
|
|
#include "gdb_assert.h"
|
|
|
|
|
#include "block.h"
|
|
|
|
|
+#include "varobj.h"
|
2006-07-11 06:33:02 +00:00
|
|
|
|
#include "observer.h"
|
|
|
|
|
#include "exec.h"
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -658,7 +659,7 @@ syms_from_objfile (struct objfile *objfi
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
/* Now either addrs or offsets is non-zero. */
|
|
|
|
|
|
|
|
|
|
- if (mainline)
|
|
|
|
|
+ if (mainline == 1)
|
|
|
|
|
{
|
|
|
|
|
/* We will modify the main symbol table, make sure that all its users
|
|
|
|
|
will be cleaned up if an error occurs during symbol reading. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -686,7 +687,7 @@ syms_from_objfile (struct objfile *objfi
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
We no longer warn if the lowest section is not a text segment (as
|
|
|
|
|
happens for the PA64 port. */
|
|
|
|
|
- if (!mainline && addrs && addrs->other[0].name)
|
|
|
|
|
+ if (/*!mainline &&*/ addrs && addrs->other[0].name)
|
|
|
|
|
{
|
|
|
|
|
asection *lower_sect;
|
|
|
|
|
asection *sect;
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -855,17 +856,21 @@ new_symfile_objfile (struct objfile *obj
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* If this is the main symbol file we have to clean up all users of the
|
|
|
|
|
old main symbol file. Otherwise it is sufficient to fixup all the
|
|
|
|
|
breakpoints that may have been redefined by this symbol file. */
|
|
|
|
|
- if (mainline)
|
|
|
|
|
+ if (mainline == 1)
|
|
|
|
|
{
|
|
|
|
|
/* OK, make it the "real" symbol file. */
|
|
|
|
|
symfile_objfile = objfile;
|
|
|
|
|
|
|
|
|
|
clear_symtab_users ();
|
|
|
|
|
}
|
|
|
|
|
- else
|
|
|
|
|
+ else if (mainline == 0)
|
|
|
|
|
{
|
|
|
|
|
breakpoint_re_set ();
|
|
|
|
|
}
|
|
|
|
|
+ else
|
|
|
|
|
+ {
|
|
|
|
|
+ /* Don't reset breakpoints or it will screw up PIE. */
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
|
|
/* We're done reading the symbol file; finish off complaints. */
|
|
|
|
|
clear_complaints (&symfile_complaints, 0, verbo);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -908,7 +913,7 @@ symbol_file_add_with_addrs_or_offsets (b
|
2005-01-11 00:15:58 +00:00
|
|
|
|
interactively wiping out any existing symbols. */
|
|
|
|
|
|
|
|
|
|
if ((have_full_symbols () || have_partial_symbols ())
|
|
|
|
|
- && mainline
|
|
|
|
|
+ && (mainline == 1)
|
|
|
|
|
&& from_tty
|
|
|
|
|
&& !query ("Load new symbol table from \"%s\"? ", name))
|
2006-07-11 06:33:02 +00:00
|
|
|
|
error (_("Not confirmed."));
|
|
|
|
|
@@ -1089,6 +1094,10 @@ symbol_file_clear (int from_tty)
|
|
|
|
|
symfile_objfile->name)
|
|
|
|
|
: !query (_("Discard symbol table? "))))
|
|
|
|
|
error (_("Not confirmed."));
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+#ifdef CLEAR_SOLIB
|
|
|
|
|
+ CLEAR_SOLIB ();
|
|
|
|
|
+#endif
|
|
|
|
|
+
|
|
|
|
|
free_all_objfiles ();
|
|
|
|
|
|
|
|
|
|
/* solib descriptors may have handles to objfiles. Since their
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -2154,6 +2163,8 @@ reread_symbols (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* Discard cleanups as symbol reading was successful. */
|
|
|
|
|
discard_cleanups (old_cleanups);
|
|
|
|
|
|
|
|
|
|
+ init_entry_point_info (objfile);
|
|
|
|
|
+
|
|
|
|
|
/* If the mtime has changed between the time we set new_modtime
|
|
|
|
|
and now, we *want* this to be out of date, so don't call stat
|
|
|
|
|
again now. */
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -2529,6 +2540,7 @@ clear_symtab_users (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
clear_pc_function_cache ();
|
|
|
|
|
if (deprecated_target_new_objfile_hook)
|
|
|
|
|
deprecated_target_new_objfile_hook (NULL);
|
|
|
|
|
+ varobj_refresh ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/breakpoint.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/breakpoint.c 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/breakpoint.c 2006-07-11 02:56:59.000000000 -0300
|
|
|
|
|
@@ -782,15 +782,15 @@ insert_watchpoints_for_new_thread (ptid_
|
|
|
|
|
struct value *v = b->owner->val_chain;
|
|
|
|
|
|
|
|
|
|
/* Look at each value on the value chain. */
|
|
|
|
|
- for (; v; v = v->next)
|
|
|
|
|
+ for (; v; v = value_next (v))
|
|
|
|
|
{
|
|
|
|
|
/* If it's a memory location, and GDB actually needed
|
|
|
|
|
its contents to evaluate the expression, then we
|
|
|
|
|
must watch it. */
|
|
|
|
|
if (VALUE_LVAL (v) == lval_memory
|
|
|
|
|
- && ! VALUE_LAZY (v))
|
|
|
|
|
+ && ! value_lazy (v))
|
|
|
|
|
{
|
|
|
|
|
- struct type *vtype = check_typedef (VALUE_TYPE (v));
|
|
|
|
|
+ struct type *vtype = check_typedef (value_type (v));
|
|
|
|
|
|
|
|
|
|
/* We only watch structs and arrays if user asked
|
|
|
|
|
for it explicitly, never if they just happen to
|
|
|
|
|
@@ -802,8 +802,8 @@ insert_watchpoints_for_new_thread (ptid_
|
|
|
|
|
CORE_ADDR addr;
|
|
|
|
|
int len, type;
|
|
|
|
|
|
|
|
|
|
- addr = VALUE_ADDRESS (v) + VALUE_OFFSET (v);
|
|
|
|
|
- len = TYPE_LENGTH (VALUE_TYPE (v));
|
|
|
|
|
+ addr = VALUE_ADDRESS (v) + value_offset (v);
|
|
|
|
|
+ len = TYPE_LENGTH (value_type (v));
|
|
|
|
|
type = hw_write;
|
|
|
|
|
if (b->owner->type == bp_read_watchpoint)
|
|
|
|
|
type = hw_read;
|
|
|
|
|
@@ -2680,12 +2680,12 @@ mark_triggered_watchpoints (CORE_ADDR st
|
|
|
|
|
|| b->type == bp_read_watchpoint
|
|
|
|
|
|| b->type == bp_access_watchpoint)
|
|
|
|
|
{
|
|
|
|
|
- for (v = b->val_chain; v; v = v->next)
|
|
|
|
|
+ for (v = b->val_chain; v; v = value_next (v))
|
|
|
|
|
{
|
|
|
|
|
if (VALUE_LVAL (v) == lval_memory
|
|
|
|
|
- && ! VALUE_LAZY (v))
|
|
|
|
|
+ && ! value_lazy (v))
|
|
|
|
|
{
|
|
|
|
|
- struct type *vtype = check_typedef (VALUE_TYPE (v));
|
|
|
|
|
+ struct type *vtype = check_typedef (value_type (v));
|
|
|
|
|
|
|
|
|
|
if (v == b->val_chain
|
|
|
|
|
|| (TYPE_CODE (vtype) != TYPE_CODE_STRUCT
|
|
|
|
|
@@ -2693,11 +2693,11 @@ mark_triggered_watchpoints (CORE_ADDR st
|
|
|
|
|
{
|
|
|
|
|
CORE_ADDR vaddr;
|
|
|
|
|
|
|
|
|
|
- vaddr = VALUE_ADDRESS (v) + VALUE_OFFSET (v);
|
|
|
|
|
+ vaddr = VALUE_ADDRESS (v) + value_offset (v);
|
|
|
|
|
/* Exact match not required. Within range is
|
|
|
|
|
sufficient. */
|
|
|
|
|
if (addr >= vaddr &&
|
|
|
|
|
- addr < vaddr + TYPE_LENGTH (VALUE_TYPE (v)))
|
|
|
|
|
+ addr < vaddr + TYPE_LENGTH (value_type (v)))
|
|
|
|
|
b->watchpoint_triggered = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
@@ -2867,12 +2867,12 @@ bpstat_stop_status (CORE_ADDR bp_addr, p
|
|
|
|
|
bs->stop = 0;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
- for (v = b->val_chain; v; v = v->next)
|
|
|
|
|
+ for (v = b->val_chain; v; v = value_next (v))
|
|
|
|
|
{
|
|
|
|
|
if (VALUE_LVAL (v) == lval_memory
|
|
|
|
|
- && ! VALUE_LAZY (v))
|
|
|
|
|
+ && ! value_lazy (v))
|
|
|
|
|
{
|
|
|
|
|
- struct type *vtype = check_typedef (VALUE_TYPE (v));
|
|
|
|
|
+ struct type *vtype = check_typedef (value_type (v));
|
|
|
|
|
|
|
|
|
|
if (v == b->val_chain
|
|
|
|
|
|| (TYPE_CODE (vtype) != TYPE_CODE_STRUCT
|
|
|
|
|
@@ -2880,11 +2880,11 @@ bpstat_stop_status (CORE_ADDR bp_addr, p
|
|
|
|
|
{
|
|
|
|
|
CORE_ADDR vaddr;
|
|
|
|
|
|
|
|
|
|
- vaddr = VALUE_ADDRESS (v) + VALUE_OFFSET (v);
|
|
|
|
|
+ vaddr = VALUE_ADDRESS (v) + value_offset (v);
|
|
|
|
|
/* Exact match not required. Within range is
|
|
|
|
|
sufficient. */
|
|
|
|
|
if (addr >= vaddr &&
|
|
|
|
|
- addr < vaddr + TYPE_LENGTH (VALUE_TYPE (v)))
|
|
|
|
|
+ addr < vaddr + TYPE_LENGTH (value_type (v)))
|
|
|
|
|
must_check_value = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
@@ -3931,6 +3931,7 @@ describe_other_breakpoints (CORE_ADDR pc
|
2005-01-11 00:15:58 +00:00
|
|
|
|
b->number,
|
|
|
|
|
((b->enable_state == bp_disabled ||
|
|
|
|
|
b->enable_state == bp_shlib_disabled ||
|
|
|
|
|
+ b->enable_state == bp_startup_disabled ||
|
|
|
|
|
b->enable_state == bp_call_disabled)
|
|
|
|
|
? " (disabled)"
|
|
|
|
|
: b->enable_state == bp_permanent
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -4609,6 +4610,62 @@ re_enable_breakpoints_in_shlibs (void)
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
+void
|
|
|
|
|
+disable_breakpoints_at_startup (int silent)
|
|
|
|
|
+{
|
|
|
|
|
+ struct breakpoint *b;
|
|
|
|
|
+ int disabled_startup_breaks = 0;
|
|
|
|
|
+
|
|
|
|
|
+ if (bfd_get_start_address (exec_bfd) != entry_point_address ())
|
|
|
|
|
+ {
|
|
|
|
|
+ ALL_BREAKPOINTS (b)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (((b->type == bp_breakpoint) ||
|
|
|
|
|
+ (b->type == bp_hardware_breakpoint)) &&
|
|
|
|
|
+ b->enable_state == bp_enabled &&
|
|
|
|
|
+ !b->loc->duplicate)
|
|
|
|
|
+ {
|
|
|
|
|
+ b->enable_state = bp_startup_disabled;
|
|
|
|
|
+ if (!silent)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (!disabled_startup_breaks)
|
|
|
|
|
+ {
|
|
|
|
|
+ target_terminal_ours_for_output ();
|
|
|
|
|
+ warning ("Temporarily disabling breakpoints:");
|
|
|
|
|
+ }
|
|
|
|
|
+ disabled_startup_breaks = 1;
|
|
|
|
|
+ warning ("breakpoint #%d addr 0x%s", b->number, paddr_nz(b->loc->address));
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+/* Try to reenable any breakpoints after startup. */
|
|
|
|
|
+void
|
|
|
|
|
+re_enable_breakpoints_at_startup (void)
|
|
|
|
|
+{
|
|
|
|
|
+ struct breakpoint *b;
|
|
|
|
|
+
|
|
|
|
|
+ if (bfd_get_start_address (exec_bfd) != entry_point_address ())
|
|
|
|
|
+ {
|
|
|
|
|
+ ALL_BREAKPOINTS (b)
|
|
|
|
|
+ if (b->enable_state == bp_startup_disabled)
|
|
|
|
|
+ {
|
|
|
|
|
+ char buf[1];
|
|
|
|
|
+
|
|
|
|
|
+ /* Do not reenable the breakpoint if the shared library
|
|
|
|
|
+ is still not mapped in. */
|
|
|
|
|
+ if (target_read_memory (b->loc->address, buf, 1) == 0)
|
|
|
|
|
+ {
|
|
|
|
|
+ /*printf ("enabling breakpoint at 0x%s\n", paddr_nz(b->loc->address));*/
|
|
|
|
|
+ b->enable_state = bp_enabled;
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+
|
|
|
|
|
static void
|
|
|
|
|
solib_load_unload_1 (char *hookname, int tempflag, char *dll_pathname,
|
|
|
|
|
char *cond_string, enum bptype bp_kind)
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -6955,6 +7012,7 @@ delete_breakpoint (struct breakpoint *bp
|
2005-01-11 00:15:58 +00:00
|
|
|
|
&& !b->loc->duplicate
|
|
|
|
|
&& b->enable_state != bp_disabled
|
|
|
|
|
&& b->enable_state != bp_shlib_disabled
|
|
|
|
|
+ && b->enable_state != bp_startup_disabled
|
|
|
|
|
&& !b->pending
|
|
|
|
|
&& b->enable_state != bp_call_disabled)
|
|
|
|
|
{
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -7170,7 +7228,8 @@ breakpoint_re_set_one (void *bint)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
save_enable = b->enable_state;
|
|
|
|
|
- if (b->enable_state != bp_shlib_disabled)
|
|
|
|
|
+ if (b->enable_state != bp_shlib_disabled
|
|
|
|
|
+ || b->enable_state != bp_shlib_disabled)
|
|
|
|
|
b->enable_state = bp_disabled;
|
|
|
|
|
else
|
|
|
|
|
/* If resetting a shlib-disabled breakpoint, we don't want to
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/solib.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/solib.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/solib.c 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
@@ -72,6 +72,8 @@ solib_ops (struct gdbarch *gdbarch)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
/* external data declarations */
|
|
|
|
|
|
|
|
|
|
+int debug_solib;
|
|
|
|
|
+
|
|
|
|
|
/* FIXME: gdbarch needs to control this variable */
|
|
|
|
|
struct target_so_ops *current_target_so_ops;
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -102,6 +104,8 @@ The search path for loading non-absolute
|
|
|
|
|
value);
|
|
|
|
|
}
|
2005-01-11 00:15:58 +00:00
|
|
|
|
|
|
|
|
|
+void add_to_target_sections (int, struct target_ops *, struct so_list *);
|
|
|
|
|
+
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
|
|
GLOBAL FUNCTION
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -372,7 +376,6 @@ free_so (struct so_list *so)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
xfree (so);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
-
|
|
|
|
|
/* Return address of first so_list entry in master shared object list. */
|
|
|
|
|
struct so_list *
|
|
|
|
|
master_so_list (void)
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -380,7 +383,6 @@ master_so_list (void)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
return so_list_head;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
-
|
|
|
|
|
/* A small stub to get us past the arg-passing pinhole of catch_errors. */
|
|
|
|
|
|
|
|
|
|
static int
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -392,15 +394,40 @@ symbol_add_stub (void *arg)
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* Have we already loaded this shared object? */
|
|
|
|
|
ALL_OBJFILES (so->objfile)
|
|
|
|
|
{
|
|
|
|
|
- if (strcmp (so->objfile->name, so->so_name) == 0)
|
|
|
|
|
+ /* Found an already loaded shared library. */
|
|
|
|
|
+ if (strcmp (so->objfile->name, so->so_name) == 0
|
|
|
|
|
+ && !so->main)
|
|
|
|
|
return 1;
|
|
|
|
|
+ /* Found an already loaded main executable. This could happen in
|
|
|
|
|
+ two circumstances.
|
|
|
|
|
+ First case: the main file has already been read in
|
|
|
|
|
+ as the first thing that gdb does at startup, and the file
|
|
|
|
|
+ hasn't been relocated properly yet. Therefor we need to read
|
|
|
|
|
+ it in with the proper section info.
|
|
|
|
|
+ Second case: it has been read in with the correct relocation,
|
|
|
|
|
+ and therefore we need to skip it. */
|
|
|
|
|
+ if (strcmp (so->objfile->name, so->so_name) == 0
|
|
|
|
|
+ && so->main
|
|
|
|
|
+ && so->main_relocated)
|
|
|
|
|
+ return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sap = build_section_addr_info_from_section_table (so->sections,
|
|
|
|
|
so->sections_end);
|
|
|
|
|
|
|
|
|
|
- so->objfile = symbol_file_add (so->so_name, so->from_tty,
|
|
|
|
|
- sap, 0, OBJF_SHARED);
|
|
|
|
|
+ if (so->main)
|
|
|
|
|
+ {
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "symbol_add_stub: adding symbols for main\n");
|
|
|
|
|
+ so->objfile = symbol_file_add (so->so_name, /*so->from_tty*/ 0,
|
|
|
|
|
+ sap, 1, 0);
|
|
|
|
|
+ so->main_relocated = 1;
|
|
|
|
|
+ }
|
|
|
|
|
+ else
|
|
|
|
|
+ so->objfile = symbol_file_add (so->so_name, so->from_tty,
|
|
|
|
|
+ sap, 0, OBJF_SHARED);
|
|
|
|
|
+
|
|
|
|
|
free_section_addr_info (sap);
|
|
|
|
|
|
|
|
|
|
return (1);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -523,6 +550,10 @@ update_solib_list (int from_tty, struct
|
2005-01-11 00:15:58 +00:00
|
|
|
|
the inferior's current list. */
|
|
|
|
|
while (i)
|
|
|
|
|
{
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "update_solib_list: compare gdb:%s and inferior:%s\n",
|
|
|
|
|
+ gdb->so_original_name, i->so_original_name);
|
|
|
|
|
if (! strcmp (gdb->so_original_name, i->so_original_name))
|
|
|
|
|
break;
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -576,28 +607,7 @@ update_solib_list (int from_tty, struct
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* Fill in the rest of each of the `struct so_list' nodes. */
|
|
|
|
|
for (i = inferior; i; i = i->next)
|
|
|
|
|
{
|
|
|
|
|
- i->from_tty = from_tty;
|
|
|
|
|
-
|
|
|
|
|
- /* Fill in the rest of the `struct so_list' node. */
|
|
|
|
|
- catch_errors (solib_map_sections, i,
|
|
|
|
|
- "Error while mapping shared library sections:\n",
|
|
|
|
|
- RETURN_MASK_ALL);
|
|
|
|
|
-
|
|
|
|
|
- /* If requested, add the shared object's sections to the TARGET's
|
|
|
|
|
- section table. Do this immediately after mapping the object so
|
|
|
|
|
- that later nodes in the list can query this object, as is needed
|
|
|
|
|
- in solib-osf.c. */
|
|
|
|
|
- if (target)
|
|
|
|
|
- {
|
|
|
|
|
- int count = (i->sections_end - i->sections);
|
|
|
|
|
- if (count > 0)
|
|
|
|
|
- {
|
|
|
|
|
- int space = target_resize_to_sections (target, count);
|
|
|
|
|
- memcpy (target->to_sections + space,
|
|
|
|
|
- i->sections,
|
|
|
|
|
- count * sizeof (i->sections[0]));
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
+ add_to_target_sections (from_tty, target, i);
|
2006-07-11 06:33:02 +00:00
|
|
|
|
|
|
|
|
|
/* Notify any observer that the shared object has been
|
|
|
|
|
loaded now that we've added it to GDB's tables. */
|
|
|
|
|
@@ -606,6 +616,41 @@ update_solib_list (int from_tty, struct
|
2005-01-11 00:15:58 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
+void
|
|
|
|
|
+add_to_target_sections (int from_tty, struct target_ops *target, struct so_list *solib)
|
|
|
|
|
+{
|
|
|
|
|
+ /* If this is set, then the sections have been already added to the
|
|
|
|
|
+ target list. */
|
|
|
|
|
+ if (solib->main)
|
|
|
|
|
+ return;
|
|
|
|
|
+
|
|
|
|
|
+ solib->from_tty = from_tty;
|
|
|
|
|
+
|
|
|
|
|
+ /* Fill in the rest of the `struct so_list' node. */
|
|
|
|
|
+ catch_errors (solib_map_sections, solib,
|
|
|
|
|
+ "Error while mapping shared library sections:\n",
|
|
|
|
|
+ RETURN_MASK_ALL);
|
|
|
|
|
+
|
|
|
|
|
+ /* If requested, add the shared object's sections to the TARGET's
|
|
|
|
|
+ section table. Do this immediately after mapping the object so
|
|
|
|
|
+ that later nodes in the list can query this object, as is needed
|
|
|
|
|
+ in solib-osf.c. */
|
|
|
|
|
+ if (target)
|
|
|
|
|
+ {
|
|
|
|
|
+ int count = (solib->sections_end - solib->sections);
|
|
|
|
|
+ if (count > 0)
|
|
|
|
|
+ {
|
|
|
|
|
+ int space = target_resize_to_sections (target, count);
|
|
|
|
|
+ if (debug_solib)
|
|
|
|
|
+ fprintf_unfiltered (gdb_stdlog,
|
|
|
|
|
+ "add_to_target_sections: add %s to to_sections\n",
|
|
|
|
|
+ solib->so_original_name);
|
|
|
|
|
+ memcpy (target->to_sections + space,
|
|
|
|
|
+ solib->sections,
|
|
|
|
|
+ count * sizeof (solib->sections[0]));
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
|
|
|
|
|
/* GLOBAL FUNCTION
|
|
|
|
|
|
2006-07-11 06:33:02 +00:00
|
|
|
|
@@ -978,4 +1023,12 @@ This takes precedence over the environme
|
|
|
|
|
reload_shared_libraries,
|
|
|
|
|
show_solib_search_path,
|
|
|
|
|
&setlist, &showlist);
|
2005-01-11 00:15:58 +00:00
|
|
|
|
+
|
2006-07-11 06:33:02 +00:00
|
|
|
|
+ add_setshow_boolean_cmd ("solib", no_class, &debug_solib,
|
|
|
|
|
+ _("\
|
|
|
|
|
+Set debugging of GNU/Linux shlib module.\n"), _("\
|
|
|
|
|
+Show debugging status of GNU/Linux shlib module.\n"), _("\
|
|
|
|
|
+Enables printf debugging output of GNU/Linux shlib module.\n"),
|
|
|
|
|
+ NULL, NULL,
|
|
|
|
|
+ &setdebuglist, &showdebuglist);
|
2005-01-11 00:15:58 +00:00
|
|
|
|
}
|
2006-07-11 06:33:02 +00:00
|
|
|
|
Index: gdb-6.5/gdb/elfread.c
|
|
|
|
|
===================================================================
|
|
|
|
|
--- gdb-6.5.orig/gdb/elfread.c 2006-07-11 02:47:06.000000000 -0300
|
|
|
|
|
+++ gdb-6.5/gdb/elfread.c 2006-07-11 02:47:11.000000000 -0300
|
|
|
|
|
@@ -556,7 +556,7 @@ elf_symfile_read (struct objfile *objfil
|
2005-01-11 00:15:58 +00:00
|
|
|
|
/* If we are reinitializing, or if we have never loaded syms yet,
|
|
|
|
|
set table to empty. MAINLINE is cleared so that *_read_psymtab
|
|
|
|
|
functions do not all also re-initialize the psymbol table. */
|
|
|
|
|
- if (mainline)
|
|
|
|
|
+ if (mainline == 1)
|
|
|
|
|
{
|
|
|
|
|
init_psymbol_list (objfile, 0);
|
|
|
|
|
mainline = 0;
|