mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-03-07 13:39:43 +08:00
This commit is the result of the following actions: - Running gdb/copyright.py to update all of the copyright headers to include 2024, - Manually updating a few files the copyright.py script told me to update, these files had copyright headers embedded within the file, - Regenerating gdbsupport/Makefile.in to refresh it's copyright date, - Using grep to find other files that still mentioned 2023. If these files were updated last year from 2022 to 2023 then I've updated them this year to 2024. I'm sure I've probably missed some dates. Feel free to fix them up as you spot them.
230 lines
5.7 KiB
C
230 lines
5.7 KiB
C
/* DWARF DIEs
|
|
|
|
Copyright (C) 1994-2024 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 "dwarf2/die.h"
|
|
#include "dwarf2/stringify.h"
|
|
|
|
/* See die.h. */
|
|
|
|
struct die_info *
|
|
die_info::allocate (struct obstack *obstack, int num_attrs)
|
|
{
|
|
size_t size = sizeof (struct die_info);
|
|
|
|
if (num_attrs > 1)
|
|
size += (num_attrs - 1) * sizeof (struct attribute);
|
|
|
|
struct die_info *die = (struct die_info *) obstack_alloc (obstack, size);
|
|
memset (die, 0, size);
|
|
return die;
|
|
}
|
|
|
|
/* See die.h. */
|
|
|
|
hashval_t
|
|
die_info::hash (const void *item)
|
|
{
|
|
const struct die_info *die = (const struct die_info *) item;
|
|
|
|
return to_underlying (die->sect_off);
|
|
}
|
|
|
|
/* See die.h. */
|
|
|
|
int
|
|
die_info::eq (const void *item_lhs, const void *item_rhs)
|
|
{
|
|
const struct die_info *die_lhs = (const struct die_info *) item_lhs;
|
|
const struct die_info *die_rhs = (const struct die_info *) item_rhs;
|
|
|
|
return die_lhs->sect_off == die_rhs->sect_off;
|
|
}
|
|
|
|
static void
|
|
dump_die_shallow (struct ui_file *f, int indent, struct die_info *die)
|
|
{
|
|
unsigned int i;
|
|
|
|
gdb_printf (f, "%*sDie: %s (abbrev %d, offset %s)\n",
|
|
indent, "",
|
|
dwarf_tag_name (die->tag), die->abbrev,
|
|
sect_offset_str (die->sect_off));
|
|
|
|
if (die->parent != NULL)
|
|
gdb_printf (f, "%*s parent at offset: %s\n",
|
|
indent, "",
|
|
sect_offset_str (die->parent->sect_off));
|
|
|
|
gdb_printf (f, "%*s has children: %s\n",
|
|
indent, "",
|
|
dwarf_bool_name (die->child != NULL));
|
|
|
|
gdb_printf (f, "%*s attributes:\n", indent, "");
|
|
|
|
for (i = 0; i < die->num_attrs; ++i)
|
|
{
|
|
gdb_printf (f, "%*s %s (%s) ",
|
|
indent, "",
|
|
dwarf_attr_name (die->attrs[i].name),
|
|
dwarf_form_name (die->attrs[i].form));
|
|
|
|
switch (die->attrs[i].form)
|
|
{
|
|
case DW_FORM_addr:
|
|
case DW_FORM_addrx:
|
|
case DW_FORM_GNU_addr_index:
|
|
gdb_printf (f, "address: ");
|
|
gdb_puts (hex_string ((CORE_ADDR) die->attrs[i].as_address ()), f);
|
|
break;
|
|
case DW_FORM_block2:
|
|
case DW_FORM_block4:
|
|
case DW_FORM_block:
|
|
case DW_FORM_block1:
|
|
gdb_printf (f, "block: size %s",
|
|
pulongest (die->attrs[i].as_block ()->size));
|
|
break;
|
|
case DW_FORM_exprloc:
|
|
gdb_printf (f, "expression: size %s",
|
|
pulongest (die->attrs[i].as_block ()->size));
|
|
break;
|
|
case DW_FORM_data16:
|
|
gdb_printf (f, "constant of 16 bytes");
|
|
break;
|
|
case DW_FORM_ref_addr:
|
|
gdb_printf (f, "ref address: ");
|
|
gdb_puts (hex_string (die->attrs[i].as_unsigned ()), f);
|
|
break;
|
|
case DW_FORM_GNU_ref_alt:
|
|
gdb_printf (f, "alt ref address: ");
|
|
gdb_puts (hex_string (die->attrs[i].as_unsigned ()), f);
|
|
break;
|
|
case DW_FORM_ref1:
|
|
case DW_FORM_ref2:
|
|
case DW_FORM_ref4:
|
|
case DW_FORM_ref8:
|
|
case DW_FORM_ref_udata:
|
|
gdb_printf (f, "constant ref: 0x%lx (adjusted)",
|
|
(long) (die->attrs[i].as_unsigned ()));
|
|
break;
|
|
case DW_FORM_data1:
|
|
case DW_FORM_data2:
|
|
case DW_FORM_data4:
|
|
case DW_FORM_data8:
|
|
case DW_FORM_udata:
|
|
gdb_printf (f, "constant: %s",
|
|
pulongest (die->attrs[i].as_unsigned ()));
|
|
break;
|
|
case DW_FORM_sec_offset:
|
|
gdb_printf (f, "section offset: %s",
|
|
pulongest (die->attrs[i].as_unsigned ()));
|
|
break;
|
|
case DW_FORM_ref_sig8:
|
|
gdb_printf (f, "signature: %s",
|
|
hex_string (die->attrs[i].as_signature ()));
|
|
break;
|
|
case DW_FORM_string:
|
|
case DW_FORM_strp:
|
|
case DW_FORM_line_strp:
|
|
case DW_FORM_strx:
|
|
case DW_FORM_GNU_str_index:
|
|
case DW_FORM_GNU_strp_alt:
|
|
gdb_printf (f, "string: \"%s\" (%s canonicalized)",
|
|
die->attrs[i].as_string ()
|
|
? die->attrs[i].as_string () : "",
|
|
die->attrs[i].canonical_string_p () ? "is" : "not");
|
|
break;
|
|
case DW_FORM_flag:
|
|
if (die->attrs[i].as_boolean ())
|
|
gdb_printf (f, "flag: TRUE");
|
|
else
|
|
gdb_printf (f, "flag: FALSE");
|
|
break;
|
|
case DW_FORM_flag_present:
|
|
gdb_printf (f, "flag: TRUE");
|
|
break;
|
|
case DW_FORM_indirect:
|
|
/* The reader will have reduced the indirect form to
|
|
the "base form" so this form should not occur. */
|
|
gdb_printf (f,
|
|
"unexpected attribute form: DW_FORM_indirect");
|
|
break;
|
|
case DW_FORM_sdata:
|
|
case DW_FORM_implicit_const:
|
|
gdb_printf (f, "constant: %s",
|
|
plongest (die->attrs[i].as_signed ()));
|
|
break;
|
|
default:
|
|
gdb_printf (f, "unsupported attribute form: %d.",
|
|
die->attrs[i].form);
|
|
break;
|
|
}
|
|
gdb_printf (f, "\n");
|
|
}
|
|
}
|
|
|
|
static void
|
|
dump_die_1 (struct ui_file *f, int level, int max_level, struct die_info *die)
|
|
{
|
|
int indent = level * 4;
|
|
|
|
gdb_assert (die != NULL);
|
|
|
|
if (level >= max_level)
|
|
return;
|
|
|
|
dump_die_shallow (f, indent, die);
|
|
|
|
if (die->child != NULL)
|
|
{
|
|
gdb_printf (f, "%*s Children:", indent, "");
|
|
if (level + 1 < max_level)
|
|
{
|
|
gdb_printf (f, "\n");
|
|
dump_die_1 (f, level + 1, max_level, die->child);
|
|
}
|
|
else
|
|
{
|
|
gdb_printf (f,
|
|
" [not printed, max nesting level reached]\n");
|
|
}
|
|
}
|
|
|
|
if (die->sibling != NULL && level > 0)
|
|
{
|
|
dump_die_1 (f, level, max_level, die->sibling);
|
|
}
|
|
}
|
|
|
|
/* See die.h. */
|
|
|
|
void
|
|
die_info::dump (int max_level)
|
|
{
|
|
dump_die_1 (gdb_stdlog, 0, max_level, this);
|
|
}
|
|
|
|
/* See die.h. */
|
|
|
|
void
|
|
die_info::error_dump ()
|
|
{
|
|
dump_die_shallow (gdb_stderr, 0, this);
|
|
}
|