diff options
author | Jeff Law <law@redhat.com> | 1994-01-18 04:04:36 +0000 |
---|---|---|
committer | Jeff Law <law@redhat.com> | 1994-01-18 04:04:36 +0000 |
commit | c7f3b703a337991dd1f32b0df0651aefc319b798 (patch) | |
tree | 4240eece64487249bed8c623817554a6c5ae6fec /gdb/paread.c | |
parent | 5c38fcee3f85a29611b1f57910be963a21b27795 (diff) | |
download | gdb-c7f3b703a337991dd1f32b0df0651aefc319b798.zip gdb-c7f3b703a337991dd1f32b0df0651aefc319b798.tar.gz gdb-c7f3b703a337991dd1f32b0df0651aefc319b798.tar.bz2 |
* config/pa/tm-hppa.h (unwind_table_entry): Use one of the
reserved fields to hold a stub unwind entry type. Fix typo.
(stub_unwind_entry): New structure for raw stub unwind entries.
(stub_unwind_types): The types of stubs we may encounter.
(UNWIND_ENTRY_SIZE, STUB_UNWIND_ENTRY_SIZE): New defines.
* hppa-tdep.c (rp_saved): Use additional information provided
by linker stub unwind descriptors.
(frameless_function_invocation): Likewise.
(frame_chain_valid): Likewise.
* paread.c (compare_unwind_entries): New function for sorting
unwind table entries.
(read_unwind_info): Rewrite to remove dependency on host endianness.
Read in data from the $UNWIND_END$ subspace which contains linker
stub unwind descriptors. Merge that data into the basic unwind
table.
Diffstat (limited to 'gdb/paread.c')
-rw-r--r-- | gdb/paread.c | 155 |
1 files changed, 144 insertions, 11 deletions
diff --git a/gdb/paread.c b/gdb/paread.c index 7c24fdd..efc8b2b 100644 --- a/gdb/paread.c +++ b/gdb/paread.c @@ -47,6 +47,10 @@ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ static void pa_symfile_init PARAMS ((struct objfile *)); +static int +compare_unwind_entries PARAMS ((struct unwind_table_entry *, + struct unwind_table_entry *)); + static void pa_new_init PARAMS ((struct objfile *)); @@ -232,6 +236,23 @@ pa_symtab_read (abfd, addr, objfile) install_minimal_symbols (objfile); } +/* Compare the start address for two unwind entries returning 1 if + the first address is larger than the second, -1 if the second is + larger than the first, and zero if they are equal. */ + +static int +compare_unwind_entries (a, b) + struct unwind_table_entry *a; + struct unwind_table_entry *b; +{ + if (a->region_start > b->region_start) + return 1; + else if (a->region_start < b->region_start) + return -1; + else + return 0; +} + /* Read in the backtrace information stored in the `$UNWIND_START$' section of the object file. This info is used mainly by find_unwind_entry() to find out the stack frame size and frame pointer used by procedures. We put @@ -242,7 +263,9 @@ static void read_unwind_info (objfile) struct objfile *objfile; { - asection *unwind_sec; + asection *unwind_sec, *stub_unwind_sec; + unsigned unwind_size, stub_unwind_size, total_size; + unsigned index, unwind_entries, stub_entries, total_entries; struct obj_unwind_info *ui; ui = obstack_alloc (&objfile->psymbol_obstack, @@ -252,22 +275,132 @@ read_unwind_info (objfile) ui->cache = NULL; ui->last = -1; - unwind_sec = bfd_get_section_by_name (objfile->obfd, - "$UNWIND_START$"); + /* Get hooks to both unwind sections. */ + unwind_sec = bfd_get_section_by_name (objfile->obfd, "$UNWIND_START$"); + stub_unwind_sec = bfd_get_section_by_name (objfile->obfd, "$UNWIND_END$"); + + /* Get sizes and unwind counts for both sections. */ if (unwind_sec) { - int size; - int i, *ip; + unwind_size = bfd_section_size (objfile->obfd, unwind_sec); + unwind_entries = unwind_size / UNWIND_ENTRY_SIZE; + } + else + { + unwind_size = 0; + unwind_entries = 0; + } + + if (stub_unwind_sec) + { + stub_unwind_size = bfd_section_size (objfile->obfd, stub_unwind_sec); + stub_entries = stub_unwind_size / STUB_UNWIND_ENTRY_SIZE; + } + else + { + stub_unwind_size = 0; + stub_entries = 0; + } + + /* Compute total number of stubs. */ + total_entries = unwind_entries + stub_entries; + total_size = total_entries * sizeof (struct unwind_table_entry); + + /* Allocate memory for the unwind table. */ + ui->table = obstack_alloc (&objfile->psymbol_obstack, total_size); + ui->last = total_entries + 1; + + /* We will read the unwind entries into temporary memory, then + fill in the actual unwind table. */ + if (unwind_size > 0) + { + unsigned long tmp; + unsigned i; + char *buf = alloca (unwind_size); + + bfd_get_section_contents (objfile->obfd, unwind_sec, buf, 0, unwind_size); + + /* Now internalize the information being careful to handle host/target + endian issues. */ + for (i = 0; i < unwind_entries; i++) + { + ui->table[i].region_start = bfd_get_32 (objfile->obfd, + (bfd_byte *)buf); + buf += 4; + ui->table[i].region_end = bfd_get_32 (objfile->obfd, (bfd_byte *)buf); + buf += 4; + tmp = bfd_get_32 (objfile->obfd, (bfd_byte *)buf); + buf += 4; + ui->table[i].Cannot_unwind = (tmp >> 31) & 0x1;; + ui->table[i].Millicode = (tmp >> 30) & 0x1; + ui->table[i].Millicode_save_sr0 = (tmp >> 29) & 0x1; + ui->table[i].Region_description = (tmp >> 27) & 0x3; + ui->table[i].reserved1 = (tmp >> 26) & 0x1; + ui->table[i].Entry_SR = (tmp >> 25) & 0x1; + ui->table[i].Entry_FR = (tmp >> 21) & 0xf; + ui->table[i].Entry_GR = (tmp >> 16) & 0x1f; + ui->table[i].Args_stored = (tmp >> 15) & 0x1; + ui->table[i].Variable_Frame = (tmp >> 14) & 0x1; + ui->table[i].Separate_Package_Body = (tmp >> 13) & 0x1; + ui->table[i].Frame_Extension_Millicode = (tmp >> 12 ) & 0x1; + ui->table[i].Stack_Overflow_Check = (tmp >> 11) & 0x1; + ui->table[i].Two_Instruction_SP_Increment = (tmp >> 10) & 0x1; + ui->table[i].Ada_Region = (tmp >> 9) & 0x1; + ui->table[i].reserved2 = (tmp >> 5) & 0xf; + ui->table[i].Save_SP = (tmp >> 4) & 0x1; + ui->table[i].Save_RP = (tmp >> 3) & 0x1; + ui->table[i].Save_MRP_in_frame = (tmp >> 2) & 0x1; + ui->table[i].extn_ptr_defined = (tmp >> 1) & 0x1; + ui->table[i].Cleanup_defined = tmp & 0x1; + tmp = bfd_get_32 (objfile->obfd, (bfd_byte *)buf); + buf += 4; + ui->table[i].MPE_XL_interrupt_marker = (tmp >> 31) & 0x1; + ui->table[i].HP_UX_interrupt_marker = (tmp >> 30) & 0x1; + ui->table[i].Large_frame = (tmp >> 29) & 0x1; + ui->table[i].reserved4 = (tmp >> 27) & 0x3; + ui->table[i].Total_frame_size = tmp & 0x7ffffff; + } + + } + + if (stub_unwind_size > 0) + { + unsigned int i; + char *buf = alloca (stub_unwind_size); - size = bfd_section_size (objfile->obfd, unwind_sec); - ui->table = obstack_alloc (&objfile->psymbol_obstack, size); - ui->last = size / sizeof (struct unwind_table_entry) - 1; + /* Read in the stub unwind entries. */ + bfd_get_section_contents (objfile->obfd, stub_unwind_sec, buf, + 0, stub_unwind_size); - bfd_get_section_contents (objfile->obfd, unwind_sec, ui->table, - 0, size); + /* Now convert them into regular unwind entries. */ + index = unwind_entries; + for (i = 0; i < stub_entries; i++, index++) + { + /* Clear out the next unwind entry. */ + memset (&ui->table[index], 0, sizeof (struct unwind_table_entry)); + + /* Convert offset & size into region_start and region_end. + Stuff away the stub type into "reserved" fields. */ + ui->table[index].region_start = bfd_get_32 (objfile->obfd, + (bfd_byte *) buf); + buf += 4; + ui->table[index].stub_type = bfd_get_8 (objfile->obfd, + (bfd_byte *) buf); + buf += 2; + ui->table[index].region_end + = ui->table[index].region_start + 4 * bfd_get_16 (objfile->obfd, + (bfd_byte *) buf); + buf += 2; + } - OBJ_UNWIND_INFO (objfile) = ui; } + + /* Unwind table needs to be kept sorted. */ + qsort (ui->table, total_entries, sizeof (struct unwind_table_entry), + compare_unwind_entries); + + /* Keep a pointer to the unwind information. */ + OBJ_UNWIND_INFO (objfile) = ui; } /* Scan and build partial symbols for a symbol file. |