compilation-bundle/dwarf-compilation.base/contrib/libdwarf/dwarfdump/print_die.c

5724 lines
206 KiB
C

/*
Copyright (C) 2000-2006 Silicon Graphics, Inc. All Rights Reserved.
Portions Copyright 2007-2010 Sun Microsystems, Inc. All rights reserved.
Portions Copyright 2009-2012 SN Systems Ltd. All rights reserved.
Portions Copyright 2007-2017 David Anderson. All rights reserved.
This program is free software; you can redistribute it and/or modify it
under the terms of version 2 of the GNU General Public License as
published by the Free Software Foundation.
This program is distributed in the hope that it would be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
Further, this software is distributed without any warranty that it is
free of the rightful claim of any third person regarding infringement
or the like. Any license provided herein, whether implied or
otherwise, applies only to this software file. Patent licenses, if
any, provided herein do not apply to combinations of this program with
other software, or any other product whatsoever.
You should have received a copy of the GNU General Public License along
with this program; if not, write the Free Software Foundation, Inc., 51
Franklin Street - Fifth Floor, Boston MA 02110-1301, USA.
*/
/* The address of the Free Software Foundation is
Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Boston, MA 02110-1301, USA.
SGI has moved from the Crittenden Lane address. */
#include "globals.h"
#include "naming.h"
#include "esb.h" /* For flexible string buffer. */
#include "print_frames.h" /* for get_string_from_locs() . */
#include "macrocheck.h"
#include "helpertree.h"
#include "tag_common.h"
/* Traverse a DIE and attributes to check self references */
static boolean traverse_one_die(Dwarf_Debug dbg,
Dwarf_Attribute attrib,
Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool Dwarf_Bool,
char **srcfiles,
Dwarf_Signed cnt, int die_indent_level);
static boolean traverse_attribute(Dwarf_Debug dbg,
Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool is_info,
Dwarf_Half attr, Dwarf_Attribute attr_in,
boolean print_information,
char **srcfiles, Dwarf_Signed cnt,
int die_indent_level);
static void print_die_and_children_internal(Dwarf_Debug dbg,
Dwarf_Die in_die_in,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool is_info,
char **srcfiles, Dwarf_Signed cnt);
static int print_one_die_section(Dwarf_Debug dbg,Dwarf_Bool is_info,
Dwarf_Error *pod_err);
/* Is this a PU has been invalidated by the SN Systems linker? */
#define IsInvalidCode(low,high) ((low == elf_max_address) || (low == 0 && high == 0))
#ifdef HAVE_USAGE_TAG_ATTR
/* Record TAGs usage */
static unsigned int tag_usage[DW_TAG_last] = {0};
#endif /* HAVE_USAGE_TAG_ATTR */
static int get_form_values(Dwarf_Debug dbg,Dwarf_Attribute attrib,
Dwarf_Half * theform, Dwarf_Half * directform);
static void show_form_itself(int show_form,int verbose,
int theform, int directform, struct esb_s * str_out);
static void print_exprloc_content(Dwarf_Debug dbg,Dwarf_Die die, Dwarf_Attribute attrib,
int showhextoo, struct esb_s *esbp);
static boolean print_attribute(Dwarf_Debug dbg, Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Half attr,
Dwarf_Attribute actual_addr,
boolean print_information,
int die_indent_level, char **srcfiles,
Dwarf_Signed cnt);
static void get_location_list(Dwarf_Debug dbg, Dwarf_Die die,
Dwarf_Attribute attr, struct esb_s *esbp);
static int legal_tag_attr_combination(Dwarf_Half tag, Dwarf_Half attr);
static int legal_tag_tree_combination(Dwarf_Half parent_tag,
Dwarf_Half child_tag);
static int _dwarf_print_one_expr_op(Dwarf_Debug dbg,
Dwarf_Loc* expr,
Dwarf_Locdesc_c exprc,
int index, Dwarf_Addr baseaddr,struct esb_s *string_out);
static int formxdata_print_value(Dwarf_Debug dbg,
Dwarf_Die die,Dwarf_Attribute attrib,
Dwarf_Half theform,
struct esb_s *esbp, Dwarf_Error * err, Dwarf_Bool hex_format);
static void bracket_hex(const char *s1, Dwarf_Unsigned v,
const char *s2, struct esb_s * esbp);
static void formx_unsigned(Dwarf_Unsigned u, struct esb_s *esbp,
Dwarf_Bool hex_format);
static void formx_data16(Dwarf_Form_Data16 * u, struct esb_s *esbp,
Dwarf_Bool hex_format);
static void formx_signed(Dwarf_Signed s, struct esb_s *esbp);
static int pd_dwarf_names_print_on_error = 1;
static int die_stack_indent_level = 0;
static boolean local_symbols_already_began = FALSE;
typedef const char *(*encoding_type_func) (unsigned,int doprintingonerr);
Dwarf_Off fde_offset_for_cu_low = DW_DLV_BADOFFSET;
Dwarf_Off fde_offset_for_cu_high = DW_DLV_BADOFFSET;
/* Indicators to record a pair [low,high], these
are used in printing DIEs to accumulate the high
and low pc across attributes and to record the pair
as soon as both are known. Probably would be better to
use variables as arguments to
print_attribute(). */
static Dwarf_Addr lowAddr = 0;
static Dwarf_Addr highAddr = 0;
static Dwarf_Bool bSawLow = FALSE;
static Dwarf_Bool bSawHigh = FALSE;
/* The following too is related to high and low pc
attributes of a function. It's misnamed, it really means
'yes, we have high and low pc' if it is TRUE. Defaulting to TRUE
seems bogus. */
static Dwarf_Bool in_valid_code = TRUE;
struct operation_descr_s {
int op_code;
int op_count;
const char * op_1type;
};
struct operation_descr_s opdesc[]= {
{DW_OP_addr,1,"addr" },
{DW_OP_deref,0,"" },
{DW_OP_const1u,1,"1u" },
{DW_OP_const1s,1,"1s" },
{DW_OP_const2u,1,"2u" },
{DW_OP_const2s,1,"2s" },
{DW_OP_const4u,1,"4u" },
{DW_OP_const4s,1,"4s" },
{DW_OP_const8u,1,"8u" },
{DW_OP_const8s,1,"8s" },
{DW_OP_constu,1,"uleb" },
{DW_OP_consts,1,"sleb" },
{DW_OP_dup,0,""},
{DW_OP_drop,0,""},
{DW_OP_over,0,""},
{DW_OP_pick,1,"1u"},
{DW_OP_swap,0,""},
{DW_OP_rot,0,""},
{DW_OP_xderef,0,""},
{DW_OP_abs,0,""},
{DW_OP_and,0,""},
{DW_OP_div,0,""},
{DW_OP_minus,0,""},
{DW_OP_mod,0,""},
{DW_OP_mul,0,""},
{DW_OP_neg,0,""},
{DW_OP_not,0,""},
{DW_OP_or,0,""},
{DW_OP_plus,0,""},
{DW_OP_plus_uconst,1,"uleb"},
{DW_OP_shl,0,""},
{DW_OP_shr,0,""},
{DW_OP_shra,0,""},
{DW_OP_xor,0,""},
{DW_OP_skip,1,"2s"},
{DW_OP_bra,1,"2s"},
{DW_OP_eq,0,""},
{DW_OP_ge,0,""},
{DW_OP_gt,0,""},
{DW_OP_le,0,""},
{DW_OP_lt,0,""},
{DW_OP_ne,0,""},
/* lit0 thru reg31 handled specially, no operands */
/* breg0 thru breg31 handled specially, 1 operand */
{DW_OP_regx,1,"uleb"},
{DW_OP_fbreg,1,"sleb"},
{DW_OP_bregx,2,"uleb"},
{DW_OP_piece,1,"uleb"},
{DW_OP_deref_size,1,"1u"},
{DW_OP_xderef_size,1,"1u"},
{DW_OP_nop,0,""},
{DW_OP_push_object_address,0,""},
{DW_OP_call2,1,"2u"},
{DW_OP_call4,1,"4u"},
{DW_OP_call_ref,1,"off"},
{DW_OP_form_tls_address,0,""},
{DW_OP_call_frame_cfa,0,""},
{DW_OP_bit_piece,2,"uleb"},
{DW_OP_implicit_value,2,"u"},
{DW_OP_stack_value,0,""},
{DW_OP_GNU_uninit,0,""},
{DW_OP_GNU_encoded_addr,1,"addr"},
{DW_OP_implicit_pointer,2,"addr" }, /* DWARF5 */
{DW_OP_GNU_implicit_pointer,2,"addr" },
{DW_OP_entry_value,2,"val" }, /* DWARF5 */
{DW_OP_GNU_entry_value,2,"val" },
{DW_OP_const_type,3,"uleb" }, /* DWARF5 */
{DW_OP_GNU_const_type,3,"uleb" },
{DW_OP_regval_type,2,"uleb" }, /* DWARF5 */
{DW_OP_GNU_regval_type,2,"uleb" },
{DW_OP_deref_type,1,"val" }, /* DWARF5 */
{DW_OP_GNU_deref_type,1,"val" },
{DW_OP_convert,1,"uleb" }, /* DWARF5 */
{DW_OP_GNU_convert,1,"uleb" },
{DW_OP_reinterpret,1,"uleb" }, /* DWARF5 */
{DW_OP_GNU_reinterpret,1,"uleb" },
{DW_OP_GNU_parameter_ref,1,"val" },
{DW_OP_GNU_const_index,1,"val" },
{DW_OP_GNU_push_tls_address,0,"" },
{DW_OP_addrx,1,"uleb" }, /* DWARF5 */
{DW_OP_GNU_addr_index,1,"val" },
{DW_OP_constx,1,"u" }, /* DWARF5 */
{DW_OP_GNU_const_index,1,"u" },
{DW_OP_GNU_parameter_ref,1,"u" },
{DW_OP_xderef,0,"" }, /* DWARF5 */
{DW_OP_xderef_type,2,"1u" }, /* DWARF5 */
/* terminator */
{0,0,""}
};
struct die_stack_data_s {
Dwarf_Die die_;
/* sibling_die_globaloffset_ is set while processing the DIE.
We do not know the sibling global offset
when we create the stack entry.
If the sibling attribute absent we never know. */
Dwarf_Off sibling_die_globaloffset_;
/* We may need is_info here too. */
Dwarf_Off cu_die_offset_; /* global offset. */
boolean already_printed_;
};
struct die_stack_data_s empty_stack_entry;
#define DIE_STACK_SIZE 800
static struct die_stack_data_s die_stack[DIE_STACK_SIZE];
#define SET_DIE_STACK_ENTRY(i,x,o) { die_stack[i].die_ = x; \
die_stack[i].cu_die_offset_ = o; \
die_stack[i].sibling_die_globaloffset_ = 0; \
die_stack[i].already_printed_ = FALSE; }
#define EMPTY_DIE_STACK_ENTRY(i) { die_stack[i] = empty_stack_entry; }
#define SET_DIE_STACK_SIBLING(x) { \
die_stack[die_stack_indent_level].sibling_die_globaloffset_ = x; }
/* The first non-zero sibling offset we can find
is what we want to return. The lowest sibling
offset in the stack. Or 0 if we have none known.
*/
static Dwarf_Off
get_die_stack_sibling()
{
int i = die_stack_indent_level;
for( ; i >=0 ; --i)
{
Dwarf_Off v = die_stack[i].sibling_die_globaloffset_;
if (v) {
return v;
}
}
return 0;
}
/* Higher stack level numbers must have a smaller sibling
offset than lower or else the sibling offsets are wrong.
Stack entries with sibling_die_globaloffset_ 0 must be
ignored in this, it just means there was no sibling
attribute at that level.
*/
static void
validate_die_stack_siblings(Dwarf_Debug dbg)
{
int i = die_stack_indent_level;
Dwarf_Off innersiboffset = 0;
char small_buf[200];
for( ; i >=0 ; --i)
{
Dwarf_Off v = die_stack[i].sibling_die_globaloffset_;
if (v) {
innersiboffset = v;
break;
}
}
if(!innersiboffset) {
/* no sibling values to check. */
return;
}
for(--i ; i >= 0 ; --i)
{
/* outersiboffset is an outer sibling offset. */
Dwarf_Off outersiboffset = die_stack[i].sibling_die_globaloffset_;
if (outersiboffset ) {
if (outersiboffset < innersiboffset) {
Dwarf_Error ouerr = 0;
snprintf(small_buf, sizeof(small_buf),
"Die stack sibling error, outer global offset "
"0x%" DW_PR_XZEROS DW_PR_DUx
" less than inner global offset "
"0x%" DW_PR_XZEROS DW_PR_DUx
", the DIE tree is erroneous.",
outersiboffset,
innersiboffset);
print_error(dbg,small_buf, DW_DLV_OK, ouerr);
}
/* We only need check one level with an offset
at each entry. */
break;
}
}
return;
}
static int
print_as_info_or_cu()
{
return (glflags.gf_info_flag || glflags.gf_types_flag
|| glflags.gf_cu_name_flag);
}
#if 0
/* Only used for debugging. */
static void
dump_die_offsets(Dwarf_Debug dbg, Dwarf_Die die,
const char *msg)
{
Dwarf_Error dderr = 0;
Dwarf_Off goff = 0;
Dwarf_Off loff = 0;
Dwarf_Half tag = 0;
int res = 0;
res = dwarf_die_offsets(die,&goff, &loff,&dderr);
DROP_ERROR_INSTANCE(dbg,res,dderr);
res = dwarf_tag(die, &tag, &dderr);
DROP_ERROR_INSTANCE(dbg,res,dderr);
printf("debugonly: Die tag 0x%x GOFF 0x%llx Loff 0x%llx %s\n",
tag,goff,loff,msg);
}
#endif
static Dwarf_Bool
form_refers_local_info(Dwarf_Half form)
{
if (form == DW_FORM_GNU_ref_alt ||
form == DW_FORM_GNU_strp_alt ||
form == DW_FORM_strp_sup ) {
/* These do not refer to the current
section and cannot be checked
as if they did. */
return FALSE;
}
return TRUE;
}
/* process each compilation unit in .debug_info */
void
print_infos(Dwarf_Debug dbg,Dwarf_Bool is_info)
{
int nres = 0;
Dwarf_Error pi_err = 0;
if (is_info) {
nres = print_one_die_section(dbg,TRUE,&pi_err);
if (nres == DW_DLV_ERROR) {
char * errmsg = dwarf_errmsg(pi_err);
Dwarf_Unsigned myerr = dwarf_errno(pi_err);
fprintf(stderr, "%s ERROR: %s: %s (%lu)\n",
program_name, "attempting to print .debug_info",
errmsg, (unsigned long) myerr);
fprintf(stderr, "attempting to continue.\n");
}
return;
}
/* !is_info */
nres = print_one_die_section(dbg,FALSE,&pi_err);
if (nres == DW_DLV_ERROR) {
char * errmsg = dwarf_errmsg(pi_err);
Dwarf_Unsigned myerr = dwarf_errno(pi_err);
fprintf(stderr, "%s ERROR: %s: %s (%lu)\n",
program_name, "attempting to print .debug_types",
errmsg, (unsigned long) myerr);
fprintf(stderr, "attempting to continue.\n");
}
}
static void
print_debug_fission_header(struct Dwarf_Debug_Fission_Per_CU_s *fsd)
{
const char * fissionsec = ".debug_cu_index";
unsigned i = 0;
struct esb_s hash_str;
if (!fsd || !fsd->pcu_type) {
/* No fission data. */
return;
}
esb_constructor(&hash_str);
printf("\n");
if (!strcmp(fsd->pcu_type,"tu")) {
fissionsec = ".debug_tu_index";
}
printf(" %-19s = %s\n","Fission section",fissionsec);
printf(" %-19s = 0x%" DW_PR_XZEROS DW_PR_DUx "\n","Fission index ",
fsd->pcu_index);
format_sig8_string(&fsd->pcu_hash,&hash_str);
printf(" %-19s = %s\n","Fission hash",esb_get_string(&hash_str));
/* 0 is always unused. Skip it. */
esb_destructor(&hash_str);
printf(" %-19s = %s\n","Fission entries","offset size DW_SECTn");
for( i = 1; i < DW_FISSION_SECT_COUNT; ++i) {
const char *nstring = 0;
Dwarf_Unsigned off = 0;
Dwarf_Unsigned size = fsd->pcu_size[i];
int res = 0;
if (size == 0) {
continue;
}
res = dwarf_get_SECT_name(i,&nstring);
if (res != DW_DLV_OK) {
nstring = "Unknown SECT";
}
off = fsd->pcu_offset[i];
printf(" %-19s = 0x%" DW_PR_XZEROS DW_PR_DUx " 0x%"
DW_PR_XZEROS DW_PR_DUx " %2d\n",
nstring,
off,size,i);
}
}
static void
print_cu_hdr_cudie(UNUSEDARG Dwarf_Debug dbg,
UNUSEDARG Dwarf_Die cudie,
Dwarf_Unsigned overall_offset,
Dwarf_Unsigned offset )
{
struct Dwarf_Debug_Fission_Per_CU_s fission_data;
if (dense) {
printf("\n");
return;
}
memset(&fission_data,0,sizeof(fission_data));
printf("\nCOMPILE_UNIT<header overall offset = 0x%"
DW_PR_XZEROS DW_PR_DUx ">",
(Dwarf_Unsigned)(overall_offset - offset));
#if 0
if (verbose) {
int fission_data_result = 0;
fission_data_result = dwarf_get_debugfission_for_die(cudie,
&fission_data,&err);
if (fission_data_result == DW_DLV_ERROR) {
print_error(dbg,"Failure looking for Debug Fission data",
fission_data_result, err);
}
print_debug_fission_header(&fission_data);
}
#endif
printf(":\n");
}
static void
print_cu_hdr_std(Dwarf_Unsigned cu_header_length,
Dwarf_Unsigned abbrev_offset,
Dwarf_Half version_stamp,
Dwarf_Half address_size,
/* offset_size is often called length_size in libdwarf. */
Dwarf_Half offset_size,
int debug_fission_res,
Dwarf_Half cu_type,
struct Dwarf_Debug_Fission_Per_CU_s * fsd)
{
int res = 0;
const char *utname = 0;
res = dwarf_get_UT_name(cu_type,&utname);
if (res != DW_DLV_OK) {
utname = "ERROR";
}
if (dense) {
printf("<%s>", "cu_header");
printf(" %s<0x%" DW_PR_XZEROS DW_PR_DUx
">", "cu_header_length",
cu_header_length);
printf(" %s<0x%04x>", "version_stamp",
version_stamp);
printf(" %s<0x%" DW_PR_XZEROS DW_PR_DUx
">", "abbrev_offset", abbrev_offset);
printf(" %s<0x%02x>", "address_size",
address_size);
printf(" %s<0x%02x>", "offset_size",
offset_size);
printf(" %s<0x%02x %s>", "cu_type",
cu_type,utname);
if (debug_fission_res == DW_DLV_OK) {
struct esb_s hash_str;
unsigned i = 0;
esb_constructor(&hash_str);
format_sig8_string(&fsd->pcu_hash,&hash_str);
printf(" %s<0x%" DW_PR_XZEROS DW_PR_DUx ">", "fissionindex",
fsd->pcu_index);
printf(" %s<%s>", "fissionhash",
esb_get_string(&hash_str));
esb_destructor(&hash_str);
for( i = 1; i < DW_FISSION_SECT_COUNT; ++i) {
const char *nstring = 0;
Dwarf_Unsigned off = 0;
Dwarf_Unsigned size = fsd->pcu_size[i];
int fires = 0;
if (size == 0) {
continue;
}
fires = dwarf_get_SECT_name(i,&nstring);
if (fires != DW_DLV_OK) {
nstring = "UnknownDW_SECT";
}
off = fsd->pcu_offset[i];
printf(" %s< 0x%" DW_PR_XZEROS DW_PR_DUx " 0x%"
DW_PR_XZEROS DW_PR_DUx ">",
nstring,
off,size);
}
}
} else {
printf("\nCU_HEADER:\n");
printf(" %-16s = 0x%" DW_PR_XZEROS DW_PR_DUx
" %" DW_PR_DUu
"\n", "cu_header_length",
cu_header_length,
cu_header_length);
printf(" %-16s = 0x%04x %u\n", "version_stamp",
version_stamp,version_stamp);
printf(" %-16s = 0x%" DW_PR_XZEROS DW_PR_DUx
" %" DW_PR_DUu
"\n", "abbrev_offset",
abbrev_offset,
abbrev_offset);
printf(" %-16s = 0x%02x %u\n", "address_size",
address_size,address_size);
printf(" %-16s = 0x%02x %u\n", "offset_size",
offset_size,offset_size);
printf(" %-16s = 0x%02x %s\n", "cu_type",
cu_type,utname);
if (debug_fission_res == DW_DLV_OK) {
print_debug_fission_header(fsd);
}
}
}
static void
print_cu_hdr_signature(Dwarf_Sig8 *signature,Dwarf_Unsigned typeoffset)
{
if (dense) {
struct esb_s sig8str;
esb_constructor(&sig8str);
format_sig8_string(signature,&sig8str);
printf(" %s<%s>", "signature",esb_get_string(&sig8str));
printf(" %s<0x%" DW_PR_XZEROS DW_PR_DUx ">",
"typeoffset", typeoffset);
esb_destructor(&sig8str);
} else {
struct esb_s sig8str;
esb_constructor(&sig8str);
format_sig8_string(signature,&sig8str);
printf(" %-16s = %s\n", "signature",esb_get_string(&sig8str));
printf(" %-16s = 0x%" DW_PR_XZEROS DW_PR_DUx " %" DW_PR_DUu "\n",
"typeoffset",
typeoffset,typeoffset);
esb_destructor(&sig8str);
}
}
static int
get_macinfo_offset(Dwarf_Debug dbg,
Dwarf_Die cu_die,
Dwarf_Unsigned *offset,
Dwarf_Error *macerr)
{
Dwarf_Attribute attrib= 0;
int vres = 0;
int ares = 0;
ares = dwarf_attr(cu_die, DW_AT_macro_info, &attrib, macerr);
if (ares == DW_DLV_ERROR) {
print_error(dbg, "dwarf_attr on DW_AT_macro_info", ares,*macerr);
} else if (ares == DW_DLV_NO_ENTRY) {
return ares;
}
vres = dwarf_global_formref(attrib,offset,macerr);
if (vres == DW_DLV_ERROR) {
dwarf_dealloc(dbg,attrib,DW_DLA_ATTR);
print_error(dbg, "dwarf_global_formref on DW_AT_macro_info",
vres, *macerr);
} else if (vres == DW_DLV_NO_ENTRY) {
}
dwarf_dealloc(dbg,attrib,DW_DLA_ATTR);
return vres;
}
/* */
static int
print_one_die_section(Dwarf_Debug dbg,Dwarf_Bool is_info,
Dwarf_Error *pod_err)
{
Dwarf_Unsigned cu_header_length = 0;
Dwarf_Unsigned abbrev_offset = 0;
Dwarf_Half version_stamp = 0;
Dwarf_Half address_size = 0;
Dwarf_Half extension_size = 0;
Dwarf_Half length_size = 0;
Dwarf_Sig8 signature;
Dwarf_Unsigned typeoffset = 0;
Dwarf_Unsigned next_cu_offset = 0;
unsigned loop_count = 0;
int nres = DW_DLV_OK;
int cu_count = 0;
char * cu_short_name = NULL;
char * cu_long_name = NULL;
const char * section_name = 0;
int res = 0;
Dwarf_Off dieprint_cu_goffset = 0;
current_section_id = is_info?DEBUG_INFO:DEBUG_TYPES;
res = dwarf_get_die_section_name(dbg, is_info,
&section_name,pod_err);
if (res == DW_DLV_NO_ENTRY) {
/* The section is absent. Print nothing for now FIXME
unless is_info (just to match previous practice) FIXME . */
if (!is_info) {
return DW_DLV_NO_ENTRY;
}
}
if (res != DW_DLV_OK) {
if(!section_name || !strlen(section_name)) {
if (is_info) {
section_name = ".debug_info";
} else {
section_name = ".debug_types";
}
}
}
if (print_as_info_or_cu() && glflags.gf_do_print_dwarf) {
printf("\n%s\n",sanitized(section_name));
}
/* Loop until it fails. */
for (;;++loop_count) {
int sres = DW_DLV_OK;
Dwarf_Die cu_die = 0;
struct Dwarf_Debug_Fission_Per_CU_s fission_data;
int fission_data_result = 0;
Dwarf_Half cu_type = 0;
memset(&fission_data,0,sizeof(fission_data));
nres = dwarf_next_cu_header_d(dbg,
is_info,
&cu_header_length, &version_stamp,
&abbrev_offset, &address_size,
&length_size,&extension_size,
&signature, &typeoffset,
&next_cu_offset,
&cu_type, pod_err);
if (nres == DW_DLV_NO_ENTRY) {
return nres;
}
if (nres != DW_DLV_OK) {
return nres;
}
if (cu_count >= break_after_n_units) {
printf("Break at %d\n",cu_count);
dieprint_cu_goffset = 0;
break;
}
/* Regardless of any options used, get basic
information about the current CU: producer, name */
sres = dwarf_siblingof_b(dbg, NULL,is_info, &cu_die, pod_err);
if (sres != DW_DLV_OK) {
dieprint_cu_goffset = 0;
print_error(dbg, "siblingof cu header", sres, *pod_err);
}
/* Get the CU offset for easy error reporting */
dwarf_die_offsets(cu_die,&DIE_overall_offset,&DIE_offset,pod_err);
DIE_CU_overall_offset = DIE_overall_offset;
DIE_CU_offset = DIE_offset;
dieprint_cu_goffset = DIE_overall_offset;
if (glflags.gf_cu_name_flag) {
if (should_skip_this_cu(dbg,cu_die)) {
dwarf_dealloc(dbg, cu_die, DW_DLA_DIE);
cu_die = 0;
++cu_count;
dieprint_cu_goffset = next_cu_offset;
continue;
}
}
{
/* Get producer name for this CU and update compiler list */
struct esb_s producername;
esb_constructor(&producername);
get_producer_name(dbg,cu_die,
dieprint_cu_goffset,&producername);
update_compiler_target(esb_get_string(&producername));
esb_destructor(&producername);
}
/* Once the compiler table has been updated, see
if we need to generate the list of CU compiled
by all the producers contained in the elf file */
if (glflags.gf_producer_children_flag) {
get_cu_name(dbg,cu_die,
dieprint_cu_goffset,&cu_short_name,&cu_long_name);
/* Add CU name to current compiler entry */
add_cu_name_compiler_target(cu_long_name);
}
/* If the current compiler is not requested by the
user, then move to the next CU */
if (!checking_this_compiler()) {
dwarf_dealloc(dbg, cu_die, DW_DLA_DIE);
++cu_count;
dieprint_cu_goffset = next_cu_offset;
cu_die = 0;
continue;
}
fission_data_result = dwarf_get_debugfission_for_die(cu_die,
&fission_data,pod_err);
if (fission_data_result == DW_DLV_ERROR) {
print_error(dbg, "Failure looking for Debug Fission data",
fission_data_result, *pod_err);
}
if(fission_data_result == DW_DLV_OK) {
/* In a .dwp file some checks get all sorts
of spurious errors. */
glflags.gf_suppress_checking_on_dwp = TRUE;
glflags.gf_check_ranges = FALSE;
glflags.gf_check_aranges = FALSE;
glflags.gf_check_decl_file = FALSE;
glflags.gf_check_lines = FALSE;
glflags.gf_check_pubname_attr = FALSE;
glflags.gf_check_fdes = FALSE;
}
/* We have not seen the compile unit yet, reset these
error-reporting globals. */
seen_CU = FALSE;
need_CU_name = TRUE;
need_CU_base_address = TRUE;
need_CU_high_address = TRUE;
/* Some prerelease gcc versions used ranges but seemingly
assumed the lack of a base address in the CU was
defined to be a zero base.
Assuming a base address (and low and high) is sensible. */
CU_base_address = 0;
CU_high_address = 0;
CU_low_address = 0;
/* Release the 'cu_die' created by the call
to 'dwarf_siblingof' at the top of the main loop. */
dwarf_dealloc(dbg, cu_die, DW_DLA_DIE);
cu_die = 0; /* For debugging, stale die should be NULL. */
if ((glflags.gf_info_flag || glflags.gf_types_flag) &&
glflags.gf_do_print_dwarf) {
if (verbose) {
print_cu_hdr_std(cu_header_length,abbrev_offset,
version_stamp,address_size,length_size,
fission_data_result,cu_type,&fission_data);
if (cu_type == DW_UT_type) {
print_cu_hdr_signature(&signature,typeoffset);
}
if (dense) {
printf("\n");
}
} else {
if (cu_type == DW_UT_type) {
if (dense) {
printf("<%s>", "cu_header");
} else {
printf("\nCU_HEADER:\n");
}
print_cu_hdr_signature(&signature,typeoffset);
if (dense) {
printf("\n");
}
}
}
}
/* Get abbreviation info for this CU */
get_abbrev_array_info(dbg,abbrev_offset);
/* Process a single compilation unit in .debug_info or
.debug_types. */
sres = dwarf_siblingof_b(dbg, NULL,is_info, &cu_die, pod_err);
if (sres == DW_DLV_OK) {
if (print_as_info_or_cu() || glflags.gf_search_is_on) {
Dwarf_Signed cnt = 0;
char **srcfiles = 0;
Dwarf_Error srcerr = 0;
int srcf = dwarf_srcfiles(cu_die,
&srcfiles, &cnt, &srcerr);
if (srcf == DW_DLV_ERROR) {
print_error_and_continue(dbg, "dwarf_srcfiles",
srcf,srcerr);
dwarf_dealloc(dbg,srcerr,DW_DLA_ERROR);
srcerr = 0;
srcfiles = 0;
cnt = 0;
} /*DW_DLV_NO_ENTRY generally means there
there is no dW_AT_stmt_list attribute.
and we do not want to print anything
about statements in that case */
/* Get the CU offset for easy error reporting */
dwarf_die_offsets(cu_die,&DIE_overall_offset,&DIE_offset,pod_err);
DIE_CU_overall_offset = DIE_overall_offset;
DIE_CU_offset = DIE_offset;
dieprint_cu_goffset = DIE_overall_offset;
print_die_and_children(dbg, cu_die,
dieprint_cu_goffset,is_info, srcfiles, cnt);
if (srcf == DW_DLV_OK) {
int si = 0;
for (si = 0; si < cnt; ++si) {
dwarf_dealloc(dbg, srcfiles[si], DW_DLA_STRING);
}
dwarf_dealloc(dbg, srcfiles, DW_DLA_LIST);
}
}
/* Dump Ranges Information */
if (dump_ranges_info) {
PrintBucketGroup(pRangesInfo,TRUE);
}
/* Check the range array if in checl mode */
if ( glflags.gf_check_ranges) {
check_range_array_info(dbg);
}
/* Traverse the line section if in check mode
or if line-printing requested */
if (glflags.gf_line_flag || glflags.gf_check_decl_file) {
int oldsection = current_section_id;
print_line_numbers_this_cu(dbg, cu_die);
current_section_id = oldsection;
}
if (glflags.gf_macro_flag || glflags.gf_check_macros) {
Dwarf_Bool in_import_list = FALSE;
Dwarf_Unsigned import_offset = 0;
int oldsection = current_section_id;
print_macros_5style_this_cu(dbg, cu_die,
in_import_list,import_offset);
in_import_list = TRUE;
while(get_next_unprinted_macro_offset(&macro_check_tree,
&import_offset) == DW_DLV_OK) {
print_macros_5style_this_cu(dbg, cu_die,
in_import_list,import_offset);
}
current_section_id = oldsection;
}
if (glflags.gf_macinfo_flag || glflags.gf_check_macros) {
int mres = 0;
int oldsection = current_section_id;
Dwarf_Unsigned offset = 0;
mres = get_macinfo_offset(dbg,cu_die,&offset,pod_err);
if (mres == DW_DLV_NO_ENTRY) {
/* By far the most likely result. */
}else if (mres == DW_DLV_ERROR) {
print_error(dbg, "get_macinfo_offset", mres,*pod_err);
} else {
print_macinfo_by_offset(dbg,offset);
current_section_id = oldsection;
}
}
dwarf_dealloc(dbg, cu_die, DW_DLA_DIE);
cu_die = 0;
} else if (sres == DW_DLV_NO_ENTRY) {
/* Do nothing I guess. */
} else {
print_error(dbg, "Regetting cu_die", sres, *pod_err);
}
++cu_count;
dieprint_cu_goffset = next_cu_offset;
}
dieprint_cu_goffset = 0;
return nres;
}
static void
print_a_die_stack(Dwarf_Debug dbg,char **srcfiles,Dwarf_Signed cnt,int lev)
{
boolean print_information = TRUE;
boolean ignore_die_stack = FALSE;
print_one_die(dbg,
die_stack[lev].die_,
die_stack[lev].cu_die_offset_,
print_information,lev,srcfiles,cnt,
ignore_die_stack);
}
extern void
print_die_and_children(Dwarf_Debug dbg,
Dwarf_Die in_die_in,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool is_info,
char **srcfiles, Dwarf_Signed cnt)
{
print_die_and_children_internal(dbg,
in_die_in, dieprint_cu_goffset,
is_info,srcfiles,cnt);
}
static void
print_die_stack(Dwarf_Debug dbg,char **srcfiles,Dwarf_Signed cnt)
{
int lev = 0;
boolean print_information = TRUE;
boolean ignore_die_stack = FALSE;
for (lev = 0; lev <= die_stack_indent_level; ++lev)
{
print_one_die(dbg,die_stack[lev].die_,
die_stack[lev].cu_die_offset_,
print_information,
lev,srcfiles,cnt,
ignore_die_stack);
}
}
/* recursively follow the die tree */
static void
print_die_and_children_internal(Dwarf_Debug dbg,
Dwarf_Die in_die_in,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool is_info,
char **srcfiles, Dwarf_Signed cnt)
{
Dwarf_Die child = 0;
Dwarf_Die sibling = 0;
Dwarf_Error dacerr = 0;
int tres = 0;
int cdres = 0;
Dwarf_Die in_die = in_die_in;
for (;;) {
/* Get the CU offset for easy error reporting */
dwarf_die_offsets(in_die,&DIE_overall_offset,&DIE_offset,&dacerr);
SET_DIE_STACK_ENTRY(die_stack_indent_level,in_die,
dieprint_cu_goffset);
if ( glflags.gf_check_tag_tree ||
glflags.gf_print_usage_tag_attr) {
DWARF_CHECK_COUNT(tag_tree_result,1);
if (die_stack_indent_level == 0) {
Dwarf_Half tag = 0;
tres = dwarf_tag(in_die, &tag, &dacerr);
if (tres != DW_DLV_OK) {
DWARF_CHECK_ERROR(tag_tree_result,
"Tag-tree root tag unavailable: "
"is not DW_TAG_compile_unit");
} else if (tag == DW_TAG_compile_unit) {
/* OK */
} else if (tag == DW_TAG_partial_unit) {
/* OK */
} else if (tag == DW_TAG_type_unit) {
/* OK */
} else {
DWARF_CHECK_ERROR(tag_tree_result,
"tag-tree root is not DW_TAG_compile_unit "
"or DW_TAG_partial_unit or DW_TAG_type_unit");
}
} else {
Dwarf_Half tag_parent = 0;
Dwarf_Half tag_child = 0;
Dwarf_Error dacerr2 = 0;
int pres = 0;
int cres = 0;
const char *ctagname = "<child tag invalid>";
const char *ptagname = "<parent tag invalid>";
pres = dwarf_tag(die_stack[die_stack_indent_level - 1].die_,
&tag_parent, &dacerr);
cres = dwarf_tag(in_die, &tag_child, &dacerr2);
if (pres != DW_DLV_OK)
tag_parent = 0;
if (cres != DW_DLV_OK)
tag_child = 0;
DROP_ERROR_INSTANCE(dbg,pres,dacerr);
DROP_ERROR_INSTANCE(dbg,cres,dacerr2);
/* Check for specific compiler */
if (checking_this_compiler()) {
/* Process specific TAGs. */
tag_specific_checks_setup(tag_child,die_stack_indent_level);
if (cres != DW_DLV_OK || pres != DW_DLV_OK) {
if (cres == DW_DLV_OK) {
ctagname = get_TAG_name(tag_child,
pd_dwarf_names_print_on_error);
}
if (pres == DW_DLV_OK) {
ptagname = get_TAG_name(tag_parent,
pd_dwarf_names_print_on_error);
}
DWARF_CHECK_ERROR3(tag_tree_result,ptagname,
ctagname,
"Tag-tree relation is not standard..");
} else if (legal_tag_tree_combination(tag_parent,
tag_child)) {
/* OK */
} else {
/* Report errors only if tag-tree check is on */
if (glflags.gf_check_tag_tree) {
DWARF_CHECK_ERROR3(tag_tree_result,
get_TAG_name(tag_parent,
pd_dwarf_names_print_on_error),
get_TAG_name(tag_child,
pd_dwarf_names_print_on_error),
"tag-tree relation is not standard.");
}
}
}
}
}
if (glflags.gf_record_dwarf_error &&
glflags.gf_check_verbose_mode) {
glflags.gf_record_dwarf_error = FALSE;
}
/* Here do pre-descent processing of the die. */
{
boolean retry_print_on_match = FALSE;
boolean ignore_die_stack = FALSE;
retry_print_on_match = print_one_die(dbg, in_die,
dieprint_cu_goffset,
print_as_info_or_cu(),
die_stack_indent_level, srcfiles, cnt,ignore_die_stack);
validate_die_stack_siblings(dbg);
if (!print_as_info_or_cu() && retry_print_on_match) {
if (glflags.gf_display_parent_tree) {
print_die_stack(dbg,srcfiles,cnt);
} else {
if (glflags.gf_display_children_tree) {
print_a_die_stack(dbg,srcfiles,cnt,die_stack_indent_level);
}
}
if (glflags.gf_display_children_tree) {
glflags.gf_stop_indent_level = die_stack_indent_level;
glflags.gf_info_flag = TRUE;
glflags.gf_types_flag = TRUE;
}
}
}
cdres = dwarf_child(in_die, &child, &dacerr);
/* Check for specific compiler */
if (glflags.gf_check_abbreviations &&
checking_this_compiler()) {
Dwarf_Half ab_has_child;
Dwarf_Bool bError = FALSE;
Dwarf_Half tag = 0;
tres = dwarf_die_abbrev_children_flag(in_die,&ab_has_child);
if (tres == DW_DLV_OK) {
DWARF_CHECK_COUNT(abbreviations_result,1);
tres = dwarf_tag(in_die, &tag, &dacerr);
if (tres == DW_DLV_OK) {
switch (tag) {
case DW_TAG_array_type:
case DW_TAG_class_type:
case DW_TAG_compile_unit:
case DW_TAG_type_unit:
case DW_TAG_partial_unit:
case DW_TAG_enumeration_type:
case DW_TAG_lexical_block:
case DW_TAG_namespace:
case DW_TAG_structure_type:
case DW_TAG_subprogram:
case DW_TAG_subroutine_type:
case DW_TAG_union_type:
case DW_TAG_entry_point:
case DW_TAG_inlined_subroutine:
break;
default:
bError = (cdres == DW_DLV_OK && !ab_has_child) ||
(cdres == DW_DLV_NO_ENTRY && ab_has_child);
if (bError) {
DWARF_CHECK_ERROR(abbreviations_result,
"check 'dw_children' flag combination.");
}
break;
}
}
}
}
/* child first: we are doing depth-first walk */
if (cdres == DW_DLV_OK) {
/* If the global offset of the (first) child is
<= the parent DW_AT_sibling global-offset-value
then the compiler has made a mistake, and
the DIE tree is corrupt. */
Dwarf_Off child_overall_offset = 0;
int cores = dwarf_dieoffset(child, &child_overall_offset, &dacerr);
if (cores == DW_DLV_OK) {
char small_buf[200];
Dwarf_Off parent_sib_val = get_die_stack_sibling();
if (parent_sib_val &&
(parent_sib_val <= child_overall_offset )) {
snprintf(small_buf,sizeof(small_buf),
"A parent DW_AT_sibling of "
"0x%" DW_PR_XZEROS DW_PR_DUx
" points %s the first child "
"0x%" DW_PR_XZEROS DW_PR_DUx
" so the die tree is corrupt "
"(showing section, not CU, offsets). ",
parent_sib_val,
(parent_sib_val == child_overall_offset)?"at":"before",
child_overall_offset);
print_error(dbg,small_buf,DW_DLV_OK,dacerr);
}
}
die_stack_indent_level++;
SET_DIE_STACK_ENTRY(die_stack_indent_level,0,dieprint_cu_goffset);
if (die_stack_indent_level >= DIE_STACK_SIZE ) {
print_error(dbg,
"ERROR: compiled in DIE_STACK_SIZE limit exceeded",
DW_DLV_OK,dacerr);
}
print_die_and_children_internal(dbg, child,
dieprint_cu_goffset,
is_info, srcfiles, cnt);
EMPTY_DIE_STACK_ENTRY(die_stack_indent_level);
die_stack_indent_level--;
if (die_stack_indent_level == 0) {
local_symbols_already_began = FALSE;
}
dwarf_dealloc(dbg, child, DW_DLA_DIE);
child = 0;
} else if (cdres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_child", cdres, dacerr);
}
/* Stop the display of all children */
if (glflags.gf_display_children_tree &&
(glflags.gf_info_flag || glflags.gf_types_flag) &&
glflags.gf_stop_indent_level == die_stack_indent_level) {
glflags.gf_info_flag = FALSE;
glflags.gf_types_flag = FALSE;
}
cdres = dwarf_siblingof_b(dbg, in_die,is_info,
&sibling, &dacerr);
if (cdres == DW_DLV_OK) {
/* print_die_and_children(dbg, sibling, srcfiles, cnt);
We loop around to actually print this, rather than
recursing. Recursing is horribly wasteful of stack
space. */
} else if (cdres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_siblingof", cdres, dacerr);
}
/* If we have a sibling, verify that its offset
is next to the last processed DIE;
An incorrect sibling chain is a nasty bug. */
if (cdres == DW_DLV_OK && sibling && glflags.gf_check_di_gaps &&
checking_this_compiler()) {
Dwarf_Off glb_off;
DWARF_CHECK_COUNT(di_gaps_result,1);
if (dwarf_validate_die_sibling(sibling,&glb_off) == DW_DLV_ERROR) {
static char msg[128];
Dwarf_Off sib_off;
dwarf_dieoffset(sibling,&sib_off,&dacerr);
sprintf(msg,
"GSIB = 0x%" DW_PR_XZEROS DW_PR_DUx
" GOFF = 0x%" DW_PR_XZEROS DW_PR_DUx
" Gap = %" DW_PR_DUu " bytes",
sib_off,glb_off,sib_off-glb_off);
DWARF_CHECK_ERROR2(di_gaps_result,
"Incorrect sibling chain",msg);
}
}
/* Here do any post-descent (ie post-dwarf_child) processing of
the in_die. */
EMPTY_DIE_STACK_ENTRY(die_stack_indent_level);
if (in_die != in_die_in) {
/* Dealloc our in_die, but not the argument die, it belongs
to our caller. Whether the siblingof call worked or not. */
dwarf_dealloc(dbg, in_die, DW_DLA_DIE);
in_die = 0;
}
if (cdres == DW_DLV_OK) {
/* Set to process the sibling, loop again. */
in_die = sibling;
} else {
/* We are done, no more siblings at this level. */
break;
}
} /* end for loop on siblings */
return;
}
/* Print one die on error and verbose or non check mode */
#define PRINTING_DIES (glflags.gf_do_print_dwarf || (glflags.gf_record_dwarf_error && glflags.gf_check_verbose_mode))
/* If print_information is FALSE, check the TAG and if it is a CU die
print the information anyway. */
boolean
print_one_die(Dwarf_Debug dbg, Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
boolean print_information,
int die_indent_level,
char **srcfiles, Dwarf_Signed cnt,
boolean ignore_die_stack)
{
Dwarf_Signed i = 0;
Dwarf_Signed j = 0;
Dwarf_Off offset = 0;
Dwarf_Off overall_offset = 0;
const char * tagname = 0;
Dwarf_Half tag = 0;
Dwarf_Signed atcnt = 0;
Dwarf_Attribute *atlist = 0;
int tres = 0;
int ores = 0;
int atres = 0;
int abbrev_code = dwarf_die_abbrev_code(die);
boolean attribute_matched = FALSE;
Dwarf_Error podie_err = 0;
/* Print using indentation
< 1><0x000854ff GOFF=0x00546047> DW_TAG_pointer_type -> 34
< 1><0x000854ff> DW_TAG_pointer_type -> 18
DW_TAG_pointer_type -> 2
*/
/* Attribute indent. */
int nColumn = glflags.gf_show_global_offsets ? 34 : 18;
if (glflags.gf_check_abbreviations && checking_this_compiler()) {
validate_abbrev_code(dbg,abbrev_code);
}
if (!ignore_die_stack && die_stack[die_indent_level].already_printed_) {
/* FALSE seems like a safe return. */
return FALSE;
}
/* Reset indentation column if no offsets */
if (!glflags.gf_display_offsets) {
nColumn = 2;
}
tres = dwarf_tag(die, &tag, &podie_err);
if (tres != DW_DLV_OK) {
print_error(dbg, "accessing tag of die!", tres, podie_err);
}
tagname = get_TAG_name(tag,pd_dwarf_names_print_on_error);
#ifdef HAVE_USAGE_TAG_ATTR
/* Record usage of TAGs */
if ( glflags.gf_print_usage_tag_attr && tag < DW_TAG_last) {
++tag_usage[tag];
}
#endif /* HAVE_USAGE_TAG_ATTR */
tag_specific_checks_setup(tag,die_indent_level);
ores = dwarf_dieoffset(die, &overall_offset, &podie_err);
if (ores != DW_DLV_OK) {
print_error(dbg, "dwarf_dieoffset", ores, podie_err);
}
ores = dwarf_die_CU_offset(die, &offset, &podie_err);
if (ores != DW_DLV_OK) {
print_error(dbg, "dwarf_die_CU_offset", ores, podie_err);
}
if (dump_visited_info && glflags.gf_check_self_references) {
printf("<%2d><0x%" DW_PR_XZEROS DW_PR_DUx
" GOFF=0x%" DW_PR_XZEROS DW_PR_DUx "> ",
die_indent_level, (Dwarf_Unsigned)offset,
(Dwarf_Unsigned)overall_offset);
printf("%*s%s\n",die_indent_level * 2 + 2," ",tagname);
}
/* Print the die */
if (PRINTING_DIES && print_information) {
if (!ignore_die_stack) {
die_stack[die_indent_level].already_printed_ = TRUE;
}
if (die_indent_level == 0) {
print_cu_hdr_cudie(dbg,die, overall_offset, offset);
} else if (local_symbols_already_began == FALSE &&
die_indent_level == 1 && !dense) {
printf("\nLOCAL_SYMBOLS:\n");
local_symbols_already_began = TRUE;
}
/* Print just the Tags and Attributes */
if (!glflags.gf_display_offsets) {
/* Print using indentation */
printf("%*s%s\n",die_stack_indent_level * 2 + 2," ",tagname);
} else {
if (dense) {
if (glflags.gf_show_global_offsets) {
if (die_indent_level == 0) {
printf("<%d><0x%" DW_PR_DUx "+0x%" DW_PR_DUx " GOFF=0x%"
DW_PR_DUx ">", die_indent_level,
(Dwarf_Unsigned)(overall_offset - offset),
(Dwarf_Unsigned)offset,
(Dwarf_Unsigned)overall_offset);
} else {
printf("<%d><0x%" DW_PR_DUx " GOFF=0x%" DW_PR_DUx ">",
die_indent_level,
(Dwarf_Unsigned)offset,
(Dwarf_Unsigned)overall_offset);
}
} else {
if (die_indent_level == 0) {
printf("<%d><0x%" DW_PR_DUx "+0x%" DW_PR_DUx ">",
die_indent_level,
(Dwarf_Unsigned)(overall_offset - offset),
(Dwarf_Unsigned)offset);
} else {
printf("<%d><0x%" DW_PR_DUx ">", die_indent_level,
(Dwarf_Unsigned)offset);
}
}
printf("<%s>",tagname);
if (verbose) {
Dwarf_Off agoff = 0;
Dwarf_Unsigned acount = 0;
printf(" <abbrev %d",abbrev_code);
if (glflags.gf_show_global_offsets) {
int agres = 0;
agres = dwarf_die_abbrev_global_offset(die,
&agoff, &acount,&podie_err);
if(agres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_die_abbrev_global_offset",
agres, podie_err);
} else if (agres == DW_DLV_NO_ENTRY) {
print_error(dbg,
"dwarf_die_abbrev_global_offset no entry?",
agres, podie_err);
} else {
printf(" ABGOFF = 0x%" DW_PR_XZEROS DW_PR_DUx
" count = 0x%" DW_PR_XZEROS DW_PR_DUx,
agoff, acount);
}
}
printf(">");
}
} else {
if (glflags.gf_show_global_offsets) {
printf("<%2d><0x%" DW_PR_XZEROS DW_PR_DUx
" GOFF=0x%" DW_PR_XZEROS DW_PR_DUx ">",
die_indent_level, (Dwarf_Unsigned)offset,
(Dwarf_Unsigned)overall_offset);
} else {
printf("<%2d><0x%" DW_PR_XZEROS DW_PR_DUx ">",
die_indent_level,
(Dwarf_Unsigned)offset);
}
/* Print using indentation */
printf("%*s%s",die_indent_level * 2 + 2," ",tagname);
if (verbose) {
Dwarf_Off agoff = 0;
Dwarf_Unsigned acount = 0;
printf(" <abbrev %d",abbrev_code);
if (glflags.gf_show_global_offsets) {
int agres = 0;
agres = dwarf_die_abbrev_global_offset(die,
&agoff, &acount,&podie_err);
if(agres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_die_abbrev_global_offset",
agres, podie_err);
} else if (agres == DW_DLV_NO_ENTRY) {
print_error(dbg,
"dwarf_die_abbrev_global_offset no entry?",
agres, podie_err);
} else {
printf(" ABGOFF = 0x%" DW_PR_XZEROS DW_PR_DUx
" count = 0x%" DW_PR_XZEROS DW_PR_DUx,
agoff, acount);
}
}
printf(">");
}
fputs("\n",stdout);
}
}
}
atres = dwarf_attrlist(die, &atlist, &atcnt, &podie_err);
if (atres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_attrlist", atres, podie_err);
} else if (atres == DW_DLV_NO_ENTRY) {
/* indicates there are no attrs. It is not an error. */
atcnt = 0;
}
/* Reset any loose references to low or high PC */
bSawLow = FALSE;
bSawHigh = FALSE;
/* Get the offset for easy error reporting: This is not the CU die. */
dwarf_die_offsets(die,&DIE_overall_offset,&DIE_offset,&podie_err);
for (i = 0; i < atcnt; i++) {
Dwarf_Half attr;
int ares;
ares = dwarf_whatattr(atlist[i], &attr, &podie_err);
if (ares == DW_DLV_OK) {
/* Check duplicated attributes; use brute force as the number of
attributes is quite small; the problem was detected with the
LLVM toolchain, generating more than 12 repeated attributes */
if (glflags.gf_check_duplicated_attributes) {
Dwarf_Half attr_next;
DWARF_CHECK_COUNT(duplicated_attributes_result,1);
for (j = i + 1; j < atcnt; ++j) {
ares = dwarf_whatattr(atlist[j], &attr_next, &podie_err);
if (ares == DW_DLV_OK) {
if (attr == attr_next) {
DWARF_CHECK_ERROR2(duplicated_attributes_result,
"Duplicated attribute ",
get_AT_name(attr,pd_dwarf_names_print_on_error));
}
} else {
print_error(dbg, "dwarf_whatattr entry missing",
ares, podie_err);
}
}
}
/* Print using indentation */
if (!dense && PRINTING_DIES && print_information) {
printf("%*s",die_indent_level * 2 + 2 + nColumn," ");
}
{
boolean attr_match = print_attribute(dbg, die,
dieprint_cu_goffset,
attr,
atlist[i],
print_information, die_indent_level, srcfiles, cnt);
if (print_information == FALSE && attr_match) {
attribute_matched = TRUE;
}
}
if (glflags.gf_record_dwarf_error &&
glflags.gf_check_verbose_mode) {
glflags.gf_record_dwarf_error = FALSE;
}
} else {
print_error(dbg, "dwarf_whatattr entry missing", ares, podie_err);
}
}
for (i = 0; i < atcnt; i++) {
dwarf_dealloc(dbg, atlist[i], DW_DLA_ATTR);
}
if (atres == DW_DLV_OK) {
dwarf_dealloc(dbg, atlist, DW_DLA_LIST);
}
if (PRINTING_DIES && dense && print_information) {
printf("\n");
}
return attribute_matched;
}
/* Encodings have undefined signedness. Accept either
signedness. The values are integer-like (they are defined
in the DWARF specification), so the
form the compiler uses (as long as it is
a constant value) is a non-issue.
The numbers need not be small (in spite of the
function name), but the result should be an integer.
If string_out is non-NULL, construct a string output, either
an error message or the name of the encoding.
The function pointer passed in is to code generated
by a script at dwarfdump build time. The code for
the val_as_string function is generated
from dwarf.h. See <build dir>/dwarf_names.c
The known_signed bool is set true(nonzero) or false (zero)
and *both* uval_out and sval_out are set to the value,
though of course uval_out cannot represent a signed
value properly and sval_out cannot represent all unsigned
values properly.
If string_out is non-NULL then attr_name and val_as_string
must also be non-NULL. */
static int
get_small_encoding_integer_and_name(Dwarf_Debug dbg,
Dwarf_Attribute attrib,
Dwarf_Unsigned * uval_out,
const char *attr_name,
struct esb_s* string_out,
encoding_type_func val_as_string,
Dwarf_Error * seierr,
int show_form)
{
Dwarf_Unsigned uval = 0;
char buf[100]; /* The strings are small. */
int vres = dwarf_formudata(attrib, &uval, seierr);
if (vres != DW_DLV_OK) {
Dwarf_Signed sval = 0;
if(vres == DW_DLV_ERROR) {
dwarf_dealloc(dbg,*seierr, DW_DLV_ERROR);
*seierr = 0;
}
vres = dwarf_formsdata(attrib, &sval, seierr);
if (vres != DW_DLV_OK) {
vres = dwarf_global_formref(attrib,&uval,seierr);
if (vres != DW_DLV_OK) {
if (string_out != 0) {
snprintf(buf, sizeof(buf),
"%s has a bad form.", attr_name);
esb_append(string_out,buf);
}
return vres;
}
*uval_out = uval;
} else {
uval = (Dwarf_Unsigned) sval;
*uval_out = uval;
}
} else {
*uval_out = uval;
}
if (string_out) {
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
struct esb_s fstring;
esb_constructor(&fstring);
get_form_values(dbg,attrib,&theform,&directform);
esb_append(&fstring, val_as_string((Dwarf_Half) uval,
pd_dwarf_names_print_on_error));
show_form_itself(show_form, verbose, theform, directform,&fstring);
esb_append(string_out,esb_get_string(&fstring));
esb_destructor(&fstring);
}
return DW_DLV_OK;
}
/* We need a 32-bit signed number here, but there's no portable
way of getting that. So use __uint32_t instead. It's supplied
in a reliable way by the autoconf infrastructure. */
static void
get_FLAG_BLOCK_string(Dwarf_Debug dbg, Dwarf_Attribute attrib,
struct esb_s*esbp)
{
int fres = 0;
Dwarf_Block *tempb = 0;
__uint32_t * array = 0;
Dwarf_Unsigned array_len = 0;
__uint32_t * array_ptr;
Dwarf_Unsigned array_remain = 0;
char linebuf[100];
Dwarf_Error fblkerr = 0;
/* first get compressed block data */
fres = dwarf_formblock (attrib,&tempb, &fblkerr);
if (fres != DW_DLV_OK) {
print_error(dbg,"DW_FORM_blockn cannot get block\n",fres,fblkerr);
return;
}
/* uncompress block into int array */
array = dwarf_uncompress_integer_block(dbg,
1, /* 'true' (meaning signed ints)*/
32, /* bits per unit */
(void *)tempb->bl_data,
tempb->bl_len,
&array_len, /* len of out array */
&fblkerr);
if (array == (void*) DW_DLV_BADOFFSET) {
print_error(dbg,"DW_AT_SUN_func_offsets cannot uncompress data\n",0,fblkerr);
return;
}
if (array_len == 0) {
print_error(dbg,"DW_AT_SUN_func_offsets has no data\n",0,fblkerr);
return;
}
/* fill in string buffer */
array_remain = array_len;
array_ptr = array;
while (array_remain > 8) {
/* Print a full line */
/* If you touch this string, update the magic number 8 in
the += and -= below! */
snprintf(linebuf, sizeof(linebuf),
"\n 0x%08x 0x%08x 0x%08x 0x%08x 0x%08x 0x%08x 0x%08x 0x%08x",
array_ptr[0], array_ptr[1],
array_ptr[2], array_ptr[3],
array_ptr[4], array_ptr[5],
array_ptr[6], array_ptr[7]);
array_ptr += 8;
array_remain -= 8;
esb_append(esbp, linebuf);
}
/* now do the last line */
if (array_remain > 0) {
esb_append(esbp, "\n ");
while (array_remain > 0) {
snprintf(linebuf, sizeof(linebuf), " 0x%08x", *array_ptr);
array_remain--;
array_ptr++;
esb_append(esbp, linebuf);
}
}
/* free array buffer */
dwarf_dealloc_uncompressed_block(dbg, array);
}
static const char *
get_rangelist_type_descr(Dwarf_Ranges *r)
{
switch (r->dwr_type) {
case DW_RANGES_ENTRY: return "range entry";
case DW_RANGES_ADDRESS_SELECTION: return "addr selection";
case DW_RANGES_END: return "range end";
}
/* Impossible. */
return "Unknown";
}
void
print_ranges_list_to_extra(Dwarf_Debug dbg,
Dwarf_Unsigned off,
Dwarf_Ranges *rangeset,
Dwarf_Signed rangecount,
Dwarf_Unsigned bytecount,
struct esb_s *stringbuf)
{
int res = 0;
char tmp[200];
const char * sec_name = 0;
Dwarf_Signed i = 0;
Dwarf_Error err =0;
res = dwarf_get_ranges_section_name(dbg,&sec_name,&err);
if(res != DW_DLV_OK || !sec_name || !strlen(sec_name)) {
sec_name = ".debug_ranges";
}
if (dense) {
snprintf(tmp,sizeof(tmp),
"< ranges: %" DW_PR_DSd " ranges at %s offset %"
DW_PR_DUu " (0x%" DW_PR_XZEROS DW_PR_DUx ") "
"(%" DW_PR_DUu " bytes)>",
rangecount,
sec_name,
off,
off,
bytecount);
esb_append(stringbuf,tmp);
} else {
snprintf(tmp,sizeof(tmp),
"\t\tranges: %" DW_PR_DSd " at %s offset %"
DW_PR_DUu " (0x%" DW_PR_XZEROS DW_PR_DUx ") "
"(%" DW_PR_DUu " bytes)\n",
rangecount,
sec_name,
off,
off,
bytecount);
esb_append(stringbuf,tmp);
}
for (i = 0; i < rangecount; ++i) {
Dwarf_Ranges * r = rangeset +i;
const char *type = get_rangelist_type_descr(r);
if (dense) {
snprintf(tmp,sizeof(tmp),
"<[%2" DW_PR_DSd
"] %s 0x%" DW_PR_XZEROS DW_PR_DUx
" 0x%" DW_PR_XZEROS DW_PR_DUx ">",
(Dwarf_Signed)i,
type,
(Dwarf_Unsigned)r->dwr_addr1,
(Dwarf_Unsigned)r->dwr_addr2);
} else {
snprintf(tmp,sizeof(tmp),
"\t\t\t[%2" DW_PR_DSd
"] %-14s 0x%" DW_PR_XZEROS DW_PR_DUx
" 0x%" DW_PR_XZEROS DW_PR_DUx "\n",
(Dwarf_Signed)i,
type,
(Dwarf_Unsigned)r->dwr_addr1,
(Dwarf_Unsigned)r->dwr_addr2);
}
esb_append(stringbuf,tmp);
}
}
static void
do_dump_visited_info(int level, Dwarf_Off loff,Dwarf_Off goff,
Dwarf_Off cu_die_goff,
const char *atname, const char *valname)
{
printf("<%2d><0x%" DW_PR_XZEROS DW_PR_DUx
" GOFF=0x%" DW_PR_XZEROS DW_PR_DUx
" CU-GOFF=0x%" DW_PR_XZEROS DW_PR_DUx
"> ",
level, loff, goff,cu_die_goff);
printf("%*s%s -> %s\n",level * 2 + 2,
" ",atname,valname);
}
/* DW_FORM_data16 should not apply here. */
static boolean
is_location_form(int form)
{
if (form == DW_FORM_block1 ||
form == DW_FORM_block2 ||
form == DW_FORM_block4 ||
form == DW_FORM_block ||
form == DW_FORM_data4 ||
form == DW_FORM_data8 ||
form == DW_FORM_sec_offset) {
return TRUE;
}
return FALSE;
}
static void
show_attr_form_error(Dwarf_Debug dbg,unsigned attr,
unsigned form,
struct esb_s *out)
{
const char *n = 0;
int res = 0;
char buf[30];
Dwarf_Error formerr = 0;
esb_append(out,"ERROR: Attribute ");
snprintf(buf,sizeof(buf),"%u",attr);
esb_append(out,buf);
esb_append(out," (");
res = dwarf_get_AT_name(attr,&n);
if (res != DW_DLV_OK) {
n = "UknownAttribute";
}
esb_append(out,n);
esb_append(out,") ");
esb_append(out," has form ");
snprintf(buf,sizeof(buf),"%u",form);
esb_append(out,buf);
esb_append(out," (");
res = dwarf_get_FORM_name(form,&n);
if (res != DW_DLV_OK) {
n = "UknownForm";
}
esb_append(out,n);
esb_append(out,"), a form which is not appropriate");
print_error_and_continue(dbg,esb_get_string(out), DW_DLV_OK,formerr);
}
/* Traverse an attribute and following any reference
in order to detect self references to DIES (loop). */
static boolean
traverse_attribute(Dwarf_Debug dbg, Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool is_info,
Dwarf_Half attr,
Dwarf_Attribute attr_in,
UNUSEDARG boolean print_information,
char **srcfiles, Dwarf_Signed cnt,
int die_indent_level)
{
Dwarf_Attribute attrib = 0;
const char * atname = 0;
int tres = 0;
Dwarf_Half tag = 0;
boolean circular_reference = FALSE;
struct esb_s valname;
Dwarf_Error err = 0;
esb_constructor(&valname);
is_info = dwarf_get_die_infotypes_flag(die);
atname = get_AT_name(attr,pd_dwarf_names_print_on_error);
/* The following gets the real attribute, even in the face of an
incorrect doubling, or worse, of attributes. */
attrib = attr_in;
/* Do not get attr via dwarf_attr: if there are (erroneously)
multiple of an attr in a DIE, dwarf_attr will not get the
second, erroneous one and dwarfdump will print the first one
multiple times. Oops. */
tres = dwarf_tag(die, &tag, &err);
if (tres == DW_DLV_ERROR) {
tag = 0;
} else if (tres == DW_DLV_NO_ENTRY) {
tag = 0;
} else {
/* ok */
}
switch (attr) {
case DW_AT_specification:
case DW_AT_abstract_origin:
case DW_AT_type: {
int res = 0;
Dwarf_Off die_goff = 0;
Dwarf_Off ref_goff = 0;
Dwarf_Die ref_die = 0;
struct esb_s specificationstr;
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
get_form_values(dbg,attrib,&theform,&directform);
if (!form_refers_local_info(theform)) {
break;
}
esb_constructor(&specificationstr);
++die_indent_level;
get_attr_value(dbg, tag, die, dieprint_cu_goffset,
attrib, srcfiles, cnt,
&specificationstr, show_form_used,verbose);
esb_append(&valname, esb_get_string(&specificationstr));
esb_destructor(&specificationstr);
/* Get the global offset for reference */
res = dwarf_global_formref(attrib, &ref_goff, &err);
if (res != DW_DLV_OK) {
int dwerrno = dwarf_errno(err);
if (dwerrno == DW_DLE_REF_SIG8_NOT_HANDLED ) {
/* No need to stop, ref_sig8 refers out of
the current section. */
break;
} else {
print_error(dbg, "dwarf_global_formref fails in traversal",
res, err);
}
}
/* Gives global offset in section. */
res = dwarf_dieoffset(die, &die_goff, &err);
if (res != DW_DLV_OK) {
int dwerrno = dwarf_errno(err);
if (dwerrno == DW_DLE_REF_SIG8_NOT_HANDLED ) {
/* No need to stop, ref_sig8 refers out of
the current section. */
break;
} else {
print_error(dbg, "dwarf_dieoffset fails in traversal", res, err);
}
}
/* Follow reference chain, looking for self references */
res = dwarf_offdie_b(dbg,ref_goff,is_info,&ref_die,&err);
if (res == DW_DLV_OK) {
Dwarf_Off target_die_cu_goff = 0;
if (dump_visited_info) {
Dwarf_Off die_loff = 0;
res = dwarf_die_CU_offset(die, &die_loff, &err);
DROP_ERROR_INSTANCE(dbg,res,err);
do_dump_visited_info(die_indent_level,die_loff,die_goff,
dieprint_cu_goffset,
atname,esb_get_string(&valname));
}
++die_indent_level;
res =dwarf_CU_dieoffset_given_die(ref_die,
&target_die_cu_goff, &err);
if (res != DW_DLV_OK) {
print_error(dbg, "dwarf_dieoffset() accessing cu_goff die!",
res, err);
}
circular_reference = traverse_one_die(dbg,attrib,ref_die,
target_die_cu_goff,
is_info,
srcfiles,cnt,die_indent_level);
DeleteKeyInBucketGroup(pVisitedInfo,ref_goff);
dwarf_dealloc(dbg,ref_die,DW_DLA_DIE);
--die_indent_level;
ref_die = 0;
}
}
break;
} /* End switch. */
esb_destructor(&valname);
return circular_reference;
}
/* Traverse one DIE in order to detect self references to DIES.
This fails to deal with changing CUs via global
references so srcfiles and cnt
are sometimes bogus. FIXME
*/
static boolean
traverse_one_die(Dwarf_Debug dbg,
Dwarf_Attribute attrib,
Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Bool is_info,
char **srcfiles, Dwarf_Signed cnt, int die_indent_level)
{
Dwarf_Half tag = 0;
Dwarf_Off overall_offset = 0;
Dwarf_Signed atcnt = 0;
int res = 0;
boolean circular_reference = FALSE;
boolean print_information = FALSE;
Dwarf_Error err = 0;
res = dwarf_tag(die, &tag, &err);
if (res != DW_DLV_OK) {
print_error(dbg, "accessing tag of die!", res, err);
}
res = dwarf_dieoffset(die, &overall_offset, &err);
if (res != DW_DLV_OK) {
print_error(dbg, "dwarf_dieoffset", res, err);
}
if (dump_visited_info) {
Dwarf_Off offset = 0;
const char * tagname = 0;
res = dwarf_die_CU_offset(die, &offset, &err);
if (res != DW_DLV_OK) {
print_error(dbg, "dwarf_die_CU_offset", res, err);
}
tagname = get_TAG_name(tag,pd_dwarf_names_print_on_error);
do_dump_visited_info(die_indent_level,offset,overall_offset,
dieprint_cu_goffset,
tagname,"");
}
DWARF_CHECK_COUNT(self_references_result,1);
if (FindKeyInBucketGroup(pVisitedInfo,overall_offset)) {
char * localvaln = NULL;
Dwarf_Half attr = 0;
struct esb_s bucketgroupstr;
const char *atname = NULL;
esb_constructor(&bucketgroupstr);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,
attrib, srcfiles,
cnt, &bucketgroupstr, show_form_used,verbose);
localvaln = esb_get_string(&bucketgroupstr);
dwarf_whatattr(attrib, &attr, &err);
atname = get_AT_name(attr,pd_dwarf_names_print_on_error);
/* We have a self reference */
DWARF_CHECK_ERROR3(self_references_result,
"Invalid self reference to DIE: ",atname,localvaln);
circular_reference = TRUE;
esb_destructor(&bucketgroupstr);
} else {
Dwarf_Signed i = 0;
Dwarf_Attribute *atlist = 0;
/* Add current DIE */
AddEntryIntoBucketGroup(pVisitedInfo,overall_offset,
0,0,0,NULL,FALSE);
res = dwarf_attrlist(die, &atlist, &atcnt, &err);
if (res == DW_DLV_ERROR) {
print_error(dbg, "dwarf_attrlist", res, err);
} else if (res == DW_DLV_NO_ENTRY) {
/* indicates there are no attrs. It is not an error. */
atcnt = 0;
}
for (i = 0; i < atcnt; i++) {
Dwarf_Half attr;
int ares;
ares = dwarf_whatattr(atlist[i], &attr, &err);
if (ares == DW_DLV_OK) {
circular_reference = traverse_attribute(dbg, die,
dieprint_cu_goffset,
is_info,
attr,
atlist[i],
print_information, srcfiles, cnt,
die_indent_level);
} else {
print_error(dbg, "dwarf_whatattr entry missing",
ares, err);
}
}
for (i = 0; i < atcnt; i++) {
dwarf_dealloc(dbg, atlist[i], DW_DLA_ATTR);
}
if (res == DW_DLV_OK) {
dwarf_dealloc(dbg, atlist, DW_DLA_LIST);
}
/* Delete current DIE */
DeleteKeyInBucketGroup(pVisitedInfo,overall_offset);
}
return circular_reference;
}
/* Extracted this from print_attribute()
to get tolerable indents.
In other words to make it readable.
It uses global data fields excessively, but so does
print_attribute().
The majority of the code here is checking for
compiler errors. */
static void
print_range_attribute(Dwarf_Debug dbg,
Dwarf_Die die,
Dwarf_Half attr,
Dwarf_Attribute attr_in,
Dwarf_Half theform,
int pra_dwarf_names_print_on_error,
boolean print_information,
int *append_extra_string,
struct esb_s *esb_extrap)
{
Dwarf_Error raerr = 0;
Dwarf_Unsigned original_off = 0;
int fres = 0;
fres = dwarf_global_formref(attr_in, &original_off, &raerr);
if (fres == DW_DLV_OK) {
Dwarf_Ranges *rangeset = 0;
Dwarf_Signed rangecount = 0;
Dwarf_Unsigned bytecount = 0;
/* If this is a dwp the ranges will be
missing or reported from a tied file.
For now we add the ranges to dbg, not tiedbg
as we do not mention tieddbg here.
May need a new interface. FIXME? */
int rres = dwarf_get_ranges_a(dbg,original_off,
die,
&rangeset,
&rangecount,&bytecount,&raerr);
if (rres == DW_DLV_OK) {
/* Ignore ranges inside a stripped function */
if (!glflags.gf_suppress_checking_on_dwp &&
glflags.gf_check_ranges &&
in_valid_code && checking_this_compiler()) {
/* Record the offset, as the ranges check will be done at
the end of the compilation unit; this approach solves
the issue of DWARF4 generating values for the high pc
as offsets relative to the low pc and the compilation
unit having DW_AT_ranges attribute. */
Dwarf_Off die_glb_offset = 0;
Dwarf_Off die_off = 0;
dwarf_die_offsets(die,&die_glb_offset,&die_off,&raerr);
record_range_array_info_entry(die_glb_offset,original_off);
}
if (print_information) {
*append_extra_string = 1;
print_ranges_list_to_extra(dbg,original_off,
rangeset,rangecount,bytecount,
esb_extrap);
}
dwarf_ranges_dealloc(dbg,rangeset,rangecount);
} else if (rres == DW_DLV_ERROR) {
if ( glflags.gf_suppress_checking_on_dwp) {
/* Ignore checks */
} else if ( glflags.gf_do_print_dwarf) {
printf("\ndwarf_get_ranges() "
"cannot find DW_AT_ranges at offset 0x%"
DW_PR_XZEROS DW_PR_DUx
" (0x%" DW_PR_XZEROS DW_PR_DUx ").",
original_off,
original_off);
} else {
DWARF_CHECK_COUNT(ranges_result,1);
DWARF_CHECK_ERROR2(ranges_result,
get_AT_name(attr,
pra_dwarf_names_print_on_error),
" cannot find DW_AT_ranges at offset");
}
} else {
/* NO ENTRY */
if ( glflags.gf_suppress_checking_on_dwp) {
/* Ignore checks */
} else if ( glflags.gf_do_print_dwarf) {
printf("\ndwarf_get_ranges() "
"finds no DW_AT_ranges at offset 0x%"
DW_PR_XZEROS DW_PR_DUx
" (%" DW_PR_DUu ").",
original_off,
original_off);
} else {
DWARF_CHECK_COUNT(ranges_result,1);
DWARF_CHECK_ERROR2(ranges_result,
get_AT_name(attr,
pra_dwarf_names_print_on_error),
" fails to find DW_AT_ranges at offset");
}
}
} else {
if (glflags.gf_do_print_dwarf) {
struct esb_s local;
char tmp[100];
esb_constructor(&local);
snprintf(tmp,sizeof(tmp)," attr 0x%x form 0x%x ",
(unsigned)attr,(unsigned)theform);
esb_append(&local,
" fails to find DW_AT_ranges offset");
esb_append(&local,tmp);
printf(" %s ",esb_get_string(&local));
esb_destructor(&local);
} else {
DWARF_CHECK_COUNT(ranges_result,1);
DWARF_CHECK_ERROR2(ranges_result,
get_AT_name(attr,
pra_dwarf_names_print_on_error),
" fails to find DW_AT_ranges offset");
}
}
}
/* A DW_AT_name in a CU DIE will likely have dots
and be entirely sensible. So lets
not call things a possible error when they are not.
Some assemblers allow '.' in an identifier too.
We should check for that, but we don't yet.
We should check the compiler before checking
for 'altabi.' too (FIXME).
This is a heuristic, not all that reliable.
Return 0 if it is a vaguely standard identifier.
Else return 1, meaning 'it might be a file name
or have '.' in it quite sensibly.'
If we don't do the TAG check we might report "t.c"
as a questionable DW_AT_name. Which would be silly.
*/
static int
dot_ok_in_identifier(int tag,
UNUSEDARG Dwarf_Die die,
const char *val)
{
if (strncmp(val,"altabi.",7)) {
/* Ignore the names of the form 'altabi.name',
which apply to one specific compiler. */
return 1;
}
if (tag == DW_TAG_compile_unit || tag == DW_TAG_partial_unit ||
tag == DW_TAG_imported_unit || tag == DW_TAG_type_unit) {
return 1;
}
return 0;
}
static void
trim_quotes(const char *val,struct esb_s *es)
{
if (val[0] == '"') {
size_t l = strlen(val);
if (l > 2 && val[l-1] == '"') {
esb_appendn(es,val+1,l-2);
return;
}
}
esb_append(es,val);
}
static int
have_a_search_match(const char *valname,const char *atname)
{
/* valname may have had quotes inserted, but search_match_text
will not. So we need to use a new copy, not valname here.
*/
struct esb_s esb_match;
char *s2;
esb_constructor(&esb_match);
trim_quotes(valname,&esb_match);
s2 = esb_get_string(&esb_match);
if (search_match_text ) {
if (!strcmp(s2,search_match_text) ||
!strcmp(atname,search_match_text)) {
esb_destructor(&esb_match);
return TRUE;
}
}
if (search_any_text) {
if (is_strstrnocase(s2,search_any_text) ||
is_strstrnocase(atname,search_any_text)) {
esb_destructor(&esb_match);
return TRUE;
}
}
#ifdef HAVE_REGEX
if (search_regex_text) {
if (!regexec(&search_re,s2,0,NULL,0) ||
!regexec(&search_re,atname,0,NULL,0)) {
esb_destructor(&esb_match);
return TRUE;
}
}
#endif
esb_destructor(&esb_match);
return FALSE;
}
/* Use our local die_stack to try to determine
signedness of the DW_AT_discr_list
LEB numbers. Returns -1 if we know
it is signed. Returns 1 if we know it is
unsigned. Returns 0 if we really do not know. */
static int
determine_discr_signedness(Dwarf_Debug dbg)
{
Dwarf_Die parent = 0;
Dwarf_Half tag = 0;
int tres = 0;
Dwarf_Error descrerr = 0;
if (die_stack_indent_level < 1) {
/* We have no idea. */
return 0;
}
parent = die_stack[die_stack_indent_level -1].die_;
if (!parent) {
/* We have no idea. */
return 0;
}
tres = dwarf_tag(parent, &tag, &descrerr);
if (tres != DW_DLV_OK) {
if(tres == DW_DLV_ERROR) {
dwarf_dealloc(dbg, descrerr, DW_DLA_ERROR);
descrerr =0;
}
return 0;
}
if (tag != DW_TAG_variant_part) {
return 0;
}
/* Expect DW_AT_discr or DW_AT_type here, and if
DW_AT_discr, that might have the DW_AT_type. */
/* FIXME: Initially lets just punt, say unsigned. */
return 1;
}
static void
checksignv(
struct esb_s *strout,
const char *title,
Dwarf_Signed sv,
Dwarf_Unsigned uv)
{
char tmpstrb[40];
/* The test and output are not entirely meaningful, but
it can be useful for readers of dwarfdump output. */
if (uv == (Dwarf_Unsigned)sv) {
/* Nothing to do here. */
return;
}
esb_append(strout," <");
esb_append(strout,title);
esb_append(strout," ");
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DSd,sv);
esb_append(strout,tmpstrb);
esb_append(strout,":");
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DUu,uv);
esb_append(strout,tmpstrb);
esb_append(strout,">");
}
static void
append_discr_array_vals(Dwarf_Debug dbg,
Dwarf_Dsc_Head h,
Dwarf_Unsigned arraycount,
int isunsigned,
struct esb_s *strout,
Dwarf_Error*paerr)
{
char tmpstrb[100];
Dwarf_Unsigned u = 0;
if (isunsigned == 0) {
esb_append(strout,
"<discriminant list signedness unknown>");
}
snprintf(tmpstrb,sizeof(tmpstrb),
"\n discr list array len: "
"%" DW_PR_DUu
"\n",
arraycount);
esb_append(strout,tmpstrb);
for(u = 0; u < arraycount; u++) {
int u2res = 0;
Dwarf_Half dtype = 0;
Dwarf_Signed slow = 0;
Dwarf_Signed shigh = 0;
Dwarf_Unsigned ulow = 0;
Dwarf_Unsigned uhigh = 0;
const char *dsc_name = "";
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DUu,u);
u2res = dwarf_discr_entry_u(h,u,
&dtype,&ulow,&uhigh,paerr);
if (u2res == DW_DLV_ERROR) {
print_error(dbg,
"DW_AT_discr_list entry access fail\n",
u2res, *paerr);
}
u2res = dwarf_discr_entry_s(h,u,
&dtype,&slow,&shigh,paerr);
if (u2res == DW_DLV_ERROR) {
print_error(dbg,
"DW_AT_discr_list entry access fail\n",
u2res, *paerr);
}
if (u2res == DW_DLV_NO_ENTRY) {
esb_append(strout,"\n "
"discr index missing! ");
esb_append(strout,tmpstrb);
break;
}
esb_append(strout," ");
esb_append(strout,tmpstrb);
esb_append(strout,": ");
#if 0
snprintf(tmpstrb,sizeof(tmpstrb),
"type=%u ",dtype);
#endif
dsc_name = get_DSC_name(dtype,pd_dwarf_names_print_on_error);
esb_append(strout,dsc_name);
esb_append(strout," ");
if (!dtype) {
if (isunsigned < 0) {
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DSd,slow);
esb_append(strout,tmpstrb);
checksignv(strout,"as signed:unsigned",slow,ulow);
} else {
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DUu,ulow);
esb_append(strout,tmpstrb);
checksignv(strout,"as signed:unsigned",slow,ulow);
}
} else {
if (isunsigned < 0) {
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DSd,slow);
esb_append(strout,tmpstrb);
checksignv(strout,"as signed:unsigned",slow,ulow);
} else {
snprintf(tmpstrb,sizeof(tmpstrb),
"%" DW_PR_DUu,ulow);
esb_append(strout,tmpstrb);
checksignv(strout,"as signed:unsigned",slow,ulow);
}
if (isunsigned < 0) {
snprintf(tmpstrb,sizeof(tmpstrb),
", %" DW_PR_DSd,shigh);
esb_append(strout,tmpstrb);
checksignv(strout,"as signed:unsigned",shigh,uhigh);
} else {
snprintf(tmpstrb,sizeof(tmpstrb),
", %" DW_PR_DUu,uhigh);
esb_append(strout,tmpstrb);
checksignv(strout,"as signed:unsigned",shigh,uhigh);
}
}
esb_append(strout,"\n");
}
}
/* Only two types of CU can have highpc or lowpc. */
static boolean
tag_type_is_addressable_cu(int tag)
{
if (tag == DW_TAG_compile_unit) {
return TRUE;
}
if (tag == DW_TAG_partial_unit) {
return TRUE;
}
return FALSE;
}
static boolean
print_attribute(Dwarf_Debug dbg, Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Half attr,
Dwarf_Attribute attr_in,
boolean print_information,
int die_indent_level,
char **srcfiles, Dwarf_Signed cnt)
{
Dwarf_Attribute attrib = 0;
Dwarf_Unsigned uval = 0;
const char * atname = 0;
struct esb_s valname;
struct esb_s esb_extra;
int tres = 0;
Dwarf_Half tag = 0;
int append_extra_string = 0;
boolean found_search_attr = FALSE;
boolean bTextFound = FALSE;
Dwarf_Bool is_info = FALSE;
Dwarf_Addr elf_max_address = 0;
Dwarf_Error paerr = 0;
esb_constructor(&esb_extra);
esb_constructor(&valname);
is_info = dwarf_get_die_infotypes_flag(die);
atname = get_AT_name(attr,pd_dwarf_names_print_on_error);
get_address_size_and_max(dbg,0,&elf_max_address,&paerr);
/* The following gets the real attribute, even in the face of an
incorrect doubling, or worse, of attributes. */
attrib = attr_in;
/* Do not get attr via dwarf_attr: if there are (erroneously)
multiple of an attr in a DIE, dwarf_attr will not get the
second, erroneous one and dwarfdump will print the first one
multiple times. Oops. */
tres = dwarf_tag(die, &tag, &paerr);
if (tres == DW_DLV_ERROR) {
tag = 0;
} else if (tres == DW_DLV_NO_ENTRY) {
tag = 0;
} else {
/* ok */
}
if ((glflags.gf_check_attr_tag || glflags.gf_print_usage_tag_attr)&&
checking_this_compiler()) {
const char *tagname = "<tag invalid>";
DWARF_CHECK_COUNT(attr_tag_result,1);
if (tres == DW_DLV_ERROR) {
DWARF_CHECK_ERROR3(attr_tag_result,tagname,
get_AT_name(attr,pd_dwarf_names_print_on_error),
"check the tag-attr combination, dwarf_tag failed.");
} else if (tres == DW_DLV_NO_ENTRY) {
DWARF_CHECK_ERROR3(attr_tag_result,tagname,
get_AT_name(attr,pd_dwarf_names_print_on_error),
"check the tag-attr combination, dwarf_tag NO ENTRY?.");
} else if (legal_tag_attr_combination(tag, attr)) {
/* OK */
} else {
/* Report errors only if tag-attr check is on */
if (glflags.gf_check_attr_tag) {
tagname = get_TAG_name(tag,pd_dwarf_names_print_on_error);
tag_specific_checks_setup(tag,die_stack_indent_level);
DWARF_CHECK_ERROR3(attr_tag_result,tagname,
get_AT_name(attr,pd_dwarf_names_print_on_error),
"check the tag-attr combination");
}
}
}
switch (attr) {
case DW_AT_language:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_language", &valname,
get_LANG_name, &paerr,
show_form_used);
break;
case DW_AT_accessibility:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_accessibility",
&valname, get_ACCESS_name,
&paerr,
show_form_used);
break;
case DW_AT_visibility:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_visibility",
&valname, get_VIS_name,
&paerr,
show_form_used);
break;
case DW_AT_virtuality:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_virtuality",
&valname,
get_VIRTUALITY_name, &paerr,
show_form_used);
break;
case DW_AT_identifier_case:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_identifier",
&valname, get_ID_name,
&paerr,
show_form_used);
break;
case DW_AT_inline:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_inline", &valname,
get_INL_name, &paerr,
show_form_used);
break;
case DW_AT_encoding:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_encoding", &valname,
get_ATE_name, &paerr,
show_form_used);
break;
case DW_AT_ordering:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_ordering", &valname,
get_ORD_name, &paerr,
show_form_used);
break;
case DW_AT_calling_convention:
get_small_encoding_integer_and_name(dbg, attrib, &uval,
"DW_AT_calling_convention",
&valname, get_CC_name,
&paerr,
show_form_used);
break;
case DW_AT_discr_list: { /* DWARF2 */
/* This has one of the block forms.
It should be in a DW_TAG_variant.
Up to September 2016 it was treated as integer or name
here, which was quite wrong. */
enum Dwarf_Form_Class fc = DW_FORM_CLASS_UNKNOWN;
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
Dwarf_Half version = 0;
Dwarf_Half offset_size = 0;
int wres = 0;
get_form_values(dbg,attrib,&theform,&directform);
wres = dwarf_get_version_of_die(die,&version,&offset_size);
if (wres != DW_DLV_OK) {
print_error(dbg,"ERROR: Cannot get DIE context version number",
DW_DLV_OK,paerr);
break;
}
fc = dwarf_get_form_class(version,attr,offset_size,theform);
if (fc == DW_FORM_CLASS_BLOCK) {
int fres = 0;
Dwarf_Block *tempb = 0;
/* the block is a series of entries each of one
of these formats:
DW_DSC_label caselabel
DW_DSC_range lowvalue highvalue
The values are all LEB. Signed or unsigned
depending on the DW_TAG_variant_part owning
the DW_TAG_variant. The DW_TAG_variant_part
will have a DW_AT_type or a DW_AT_discr
and that attribute will reveal the signedness of all
the leb values.
As a practical matter DW_DSC_label/DW_DSC_range
value (zero or one, so far)
can safely be read as ULEB or SLEB
and one gets a valid value whereas
the caselabel, lowvalue,highvalue must be
decoded with the proper sign. the high level
(dwarfdump in this case) is the agent that
should determine the proper signedness. */
fres = dwarf_formblock(attrib, &tempb, &paerr);
if (fres == DW_DLV_OK) {
struct esb_s bformstr;
int isunsigned = 0; /* Meaning unknown */
Dwarf_Dsc_Head h = 0;
Dwarf_Unsigned arraycount = 0;
int sres = 0;
esb_constructor(&bformstr);
show_form_itself(show_form_used,verbose, theform,
directform,&bformstr);
isunsigned = determine_discr_signedness(dbg);
esb_empty_string(&valname);
sres = dwarf_discr_list(dbg,
(Dwarf_Small *)tempb->bl_data,
tempb->bl_len,
&h,&arraycount,&paerr);
if (sres == DW_DLV_NO_ENTRY) {
esb_append(&bformstr,"<empty discriminant list>");
break;
}
if (sres == DW_DLV_ERROR) {
print_error(dbg, "DW_AT_discr_list access fail\n",
sres, paerr);
}
append_discr_array_vals(dbg,h,arraycount,
isunsigned,&bformstr,&paerr);
if (verbose > 1) {
unsigned u = 0;
char tmpstrb[100];
snprintf(tmpstrb,sizeof(tmpstrb),
"\n block byte len:"
"0x%" DW_PR_XZEROS DW_PR_DUx
"\n ",
tempb->bl_len);
esb_append(&bformstr, tmpstrb);
for (u = 0; u < tempb->bl_len; u++) {
snprintf(tmpstrb, sizeof(tmpstrb), "%02x ",
*(u + (unsigned char *)tempb->bl_data));
esb_append(&bformstr, tmpstrb);
}
}
esb_append(&valname, esb_get_string(&bformstr));
dwarf_dealloc(dbg,h,DW_DLA_DSC_HEAD);
dwarf_dealloc(dbg, tempb, DW_DLA_BLOCK);
esb_destructor(&bformstr);
tempb = 0;
} else {
print_error(dbg, "DW_FORM_blockn cannot get block\n", fres,
paerr);
}
} else {
print_error(dbg, "DW_AT_discr_list is not form BLOCK\n",
DW_DLV_OK, paerr);
}
}
break;
case DW_AT_data_member_location:
{
/* Value is a constant or a location
description or location list.
If a constant, it could be signed or
unsigned. Telling whether a constant
or a reference is nontrivial
since DW_FORM_data{4,8}
could be either in DWARF{2,3} */
enum Dwarf_Form_Class fc = DW_FORM_CLASS_UNKNOWN;
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
Dwarf_Half version = 0;
Dwarf_Half offset_size = 0;
int wres = 0;
get_form_values(dbg,attrib,&theform,&directform);
wres = dwarf_get_version_of_die(die,&version,&offset_size);
if (wres != DW_DLV_OK) {
print_error(dbg,"ERROR: Cannot get DIE context version number",
DW_DLV_OK,paerr);
break;
}
fc = dwarf_get_form_class(version,attr,offset_size,theform);
if (fc == DW_FORM_CLASS_CONSTANT) {
struct esb_s classconstantstr;
esb_constructor(&classconstantstr);
/* Makes no sense to look at type of our DIE
to determine how to print the constant. */
wres = formxdata_print_value(dbg,NULL,attrib,
theform,
&classconstantstr,
&paerr, FALSE);
show_form_itself(show_form_used,verbose, theform,
directform,&classconstantstr);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&classconstantstr));
esb_destructor(&classconstantstr);
if (wres == DW_DLV_OK){
/* String appended already. */
break;
} else if (wres == DW_DLV_NO_ENTRY) {
print_error(dbg,"Cannot get DW_AT_data_member_location, how can it be NO_ENTRY? ",wres,paerr);
break;
} else {
print_error(dbg,"Cannot get DW_AT_data_member_location ",wres,paerr);
break;
}
}
/* FALL THRU, this is a
a location description, or a reference
to one, or a mistake. */
}
/* FALL THRU to location description */
case DW_AT_location:
case DW_AT_vtable_elem_location:
case DW_AT_string_length:
case DW_AT_return_addr:
case DW_AT_use_location:
case DW_AT_static_link:
case DW_AT_frame_base:
{
/* The value is a location description
or location list. */
struct esb_s framebasestr;
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
esb_constructor(&framebasestr);
get_form_values(dbg,attrib,&theform,&directform);
if (is_location_form(theform)) {
get_location_list(dbg, die, attrib, &framebasestr);
show_form_itself(show_form_used,verbose,
theform, directform,&framebasestr);
} else if (theform == DW_FORM_exprloc) {
int showhextoo = 1;
print_exprloc_content(dbg,die,attrib,showhextoo,&framebasestr);
} else {
show_attr_form_error(dbg,attr,theform,&framebasestr);
}
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&framebasestr));
esb_destructor(&framebasestr);
}
break;
case DW_AT_SUN_func_offsets:
{
/* value is a location description or location list */
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
struct esb_s funcformstr;
esb_constructor(&funcformstr);
get_form_values(dbg,attrib,&theform,&directform);
get_FLAG_BLOCK_string(dbg, attrib,&funcformstr);
show_form_itself(show_form_used,verbose, theform,
directform,&funcformstr);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&funcformstr));
esb_destructor(&funcformstr);
}
break;
case DW_AT_SUN_cf_kind:
{
Dwarf_Half kind = 0;
Dwarf_Unsigned tempud = 0;
Dwarf_Error cferr = 0;
int wres = 0;
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
struct esb_s cfkindstr;
esb_constructor(&cfkindstr);
get_form_values(dbg,attrib,&theform,&directform);
wres = dwarf_formudata (attrib,&tempud, &cferr);
if (wres == DW_DLV_OK) {
kind = tempud;
esb_append(&cfkindstr,
get_ATCF_name(kind,pd_dwarf_names_print_on_error));
} else if (wres == DW_DLV_NO_ENTRY) {
esb_append(&cfkindstr, "?");
} else {
print_error(dbg,"Cannot get formudata....",wres,cferr);
esb_append(&cfkindstr, "??");
}
show_form_itself(show_form_used,verbose, theform,
directform,&cfkindstr);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&cfkindstr));
esb_destructor(&cfkindstr);
}
break;
case DW_AT_upper_bound:
{
Dwarf_Half theform;
int rv;
struct esb_s upperboundstr;
esb_constructor(&upperboundstr);
rv = dwarf_whatform(attrib,&theform,&paerr);
/* depending on the form and the attribute, process the form */
if (rv == DW_DLV_ERROR) {
print_error(dbg, "dwarf_whatform Cannot find attr form",
rv, paerr);
} else if (rv == DW_DLV_NO_ENTRY) {
esb_destructor(&upperboundstr);
break;
}
switch (theform) {
case DW_FORM_block1: {
Dwarf_Half btheform = 0;
Dwarf_Half directform = 0;
get_form_values(dbg,attrib,&btheform,&directform);
get_location_list(dbg, die, attrib, &upperboundstr);
show_form_itself(show_form_used,verbose, btheform,
directform,&upperboundstr);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&upperboundstr));
}
break;
default:
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,
attrib, srcfiles, cnt, &upperboundstr,
show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&upperboundstr));
break;
}
esb_destructor(&upperboundstr);
break;
}
case DW_AT_low_pc:
case DW_AT_high_pc:
{
Dwarf_Half theform;
int rv;
/* For DWARF4, the high_pc offset from the low_pc */
Dwarf_Unsigned highpcOff = 0;
Dwarf_Bool offsetDetected = FALSE;
struct esb_s highpcstr;
esb_constructor(&highpcstr);
rv = dwarf_whatform(attrib,&theform,&paerr);
/* Depending on the form and the attribute,
process the form. */
if (rv == DW_DLV_ERROR) {
print_error(dbg, "dwarf_whatform cannot Find attr form",
rv, paerr);
} else if (rv == DW_DLV_NO_ENTRY) {
break;
}
if (theform != DW_FORM_addr &&
theform != DW_FORM_GNU_addr_index &&
theform != DW_FORM_addrx) {
/* New in DWARF4: other forms
(of class constant) are not an address
but are instead offset from pc.
One could test for DWARF4 here before adding
this string, but that seems unnecessary as this
could not happen with DWARF3 or earlier.
A normal consumer would have to add this value to
DW_AT_low_pc to get a true pc. */
esb_append(&highpcstr,"<offset-from-lowpc>");
/* Update the high_pc value if we are checking the ranges */
if ( glflags.gf_check_ranges && attr == DW_AT_high_pc) {
/* Get the offset value */
int show_form_here = 0;
int res = get_small_encoding_integer_and_name(dbg,
attrib,
&highpcOff,
/* attrname */ (const char *) NULL,
/* err_string */ ( struct esb_s *) NULL,
(encoding_type_func) 0,
&paerr,show_form_here);
if (res != DW_DLV_OK) {
print_error(dbg, "get_small_encoding_integer_and_name",
res, paerr);
}
offsetDetected = TRUE;
}
}
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,
attrib, srcfiles, cnt,
&highpcstr,show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&highpcstr));
esb_destructor(&highpcstr);
/* Update base and high addresses for CU */
if (seen_CU && (need_CU_base_address || need_CU_high_address)) {
/* Update base address for CU */
if (attr == DW_AT_low_pc) {
if (need_CU_base_address &&
tag_type_is_addressable_cu(tag)) {
int res = dwarf_formaddr(attrib, &CU_base_address,
&paerr);
DROP_ERROR_INSTANCE(dbg,res,paerr);
if (res == DW_DLV_OK) {
need_CU_base_address = FALSE;
CU_low_address = CU_base_address;
}
} else if (!CU_low_address) {
/* We take the first non-zero address
as meaningful. Even if no such in CU DIE. */
int res = dwarf_formaddr(attrib, &CU_low_address,
&paerr);
DROP_ERROR_INSTANCE(dbg,res,paerr);
if (res == DW_DLV_OK) {
/* Stop looking for base. Bogus, but
there is none available, so stop. */
need_CU_base_address = FALSE;
}
}
}
/* Update high address for CU */
if (attr == DW_AT_high_pc) {
if (need_CU_high_address ) {
/* This is bogus in that it accepts the first
high address in the CU, from any TAG */
int res = dwarf_formaddr(attrib, &CU_high_address,
&paerr);
DROP_ERROR_INSTANCE(dbg,res,paerr);
if (res == DW_DLV_OK) {
need_CU_high_address = FALSE;
}
}
}
}
/* Record the low and high addresses as we have them */
/* For DWARF4 allow the high_pc value as an offset */
if ((glflags.gf_check_decl_file ||
glflags.gf_check_ranges ||
glflags.gf_check_locations) &&
((theform == DW_FORM_addr ||
theform == DW_FORM_GNU_addr_index ||
theform == DW_FORM_addrx) || offsetDetected)) {
int res = 0;
Dwarf_Addr addr = 0;
/* Calculate the real high_pc value */
if (offsetDetected && seen_PU_base_address) {
addr = lowAddr + highpcOff;
res = DW_DLV_OK;
} else {
res = dwarf_formaddr(attrib, &addr, &paerr);
DROP_ERROR_INSTANCE(dbg,res,paerr);
}
if(res == DW_DLV_OK) {
if (attr == DW_AT_low_pc) {
lowAddr = addr;
bSawLow = TRUE;
/* Record the base address of the last seen PU
to be used when checking line information */
if (seen_PU && !seen_PU_base_address) {
seen_PU_base_address = TRUE;
PU_base_address = addr;
}
} else { /* DW_AT_high_pc */
highAddr = addr;
bSawHigh = TRUE;
/* Record the high address of the last seen PU
to be used when checking line information */
if (seen_PU && !seen_PU_high_address) {
seen_PU_high_address = TRUE;
PU_high_address = addr;
}
}
}
/* We have now both low_pc and high_pc values */
if (bSawLow && bSawHigh) {
/* We need to decide if this PU is
valid, as the SN Linker marks a stripped
function by setting lowpc to -1;
also for discarded comdat, both lowpc
and highpc are zero */
if (need_PU_valid_code) {
need_PU_valid_code = FALSE;
/* To ignore a PU as invalid code,
only consider the lowpc and
highpc values associated with the
DW_TAG_subprogram; other
instances of lowpc and highpc,
must be ignore (lexical blocks) */
in_valid_code = TRUE;
if (IsInvalidCode(lowAddr,highAddr) &&
tag == DW_TAG_subprogram) {
in_valid_code = FALSE;
}
}
/* We have a low_pc/high_pc pair;
check if they are valid */
if (in_valid_code) {
DWARF_CHECK_COUNT(ranges_result,1);
if (lowAddr != elf_max_address &&
lowAddr > highAddr) {
DWARF_CHECK_ERROR(ranges_result,
".debug_info: Incorrect values "
"for low_pc/high_pc");
if (glflags.gf_check_verbose_mode &&
PRINTING_UNIQUE) {
printf("Low = 0x%" DW_PR_XZEROS DW_PR_DUx
", High = 0x%" DW_PR_XZEROS DW_PR_DUx "\n",
lowAddr,highAddr);
}
}
if (glflags.gf_check_decl_file ||
glflags.gf_check_ranges ||
glflags.gf_check_locations) {
AddEntryIntoBucketGroup(pRangesInfo,0,
lowAddr,
lowAddr,highAddr,NULL,FALSE);
}
}
bSawLow = FALSE;
bSawHigh = FALSE;
}
}
}
break;
case DW_AT_ranges:
{
Dwarf_Half theform = 0;
int rv;
struct esb_s rangesstr;
esb_constructor(&rangesstr);
rv = dwarf_whatform(attrib,&theform,&paerr);
if (rv == DW_DLV_ERROR) {
print_error(dbg, "dwarf_whatform cannot find Attr Form",
rv, paerr);
} else if (rv == DW_DLV_NO_ENTRY) {
esb_destructor(&rangesstr);
break;
}
esb_empty_string(&rangesstr);
get_attr_value(dbg, tag,die,
dieprint_cu_goffset,attrib, srcfiles, cnt, &rangesstr,
show_form_used,verbose);
print_range_attribute(dbg, die, attr,attr_in, theform,
pd_dwarf_names_print_on_error,print_information,
&append_extra_string,
&esb_extra);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&rangesstr));
esb_destructor(&rangesstr);
}
break;
case DW_AT_MIPS_linkage_name:
{
struct esb_s linkagenamestr;
esb_constructor(&linkagenamestr);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset, attrib, srcfiles,
cnt, &linkagenamestr, show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&linkagenamestr));
esb_destructor(&linkagenamestr);
if ( glflags.gf_check_locations || glflags.gf_check_ranges) {
int local_show_form = 0;
int local_verbose = 0;
const char *name = 0;
struct esb_s lesb;
esb_constructor(&lesb);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&lesb, local_show_form,local_verbose);
/* Look for specific name forms, attempting to
notice and report 'odd' identifiers. */
name = esb_get_string(&lesb);
safe_strcpy(PU_name,sizeof(PU_name),name,strlen(name));
esb_destructor(&lesb);
}
}
break;
case DW_AT_name:
case DW_AT_GNU_template_name:
{
struct esb_s templatenamestr;
esb_constructor(&templatenamestr);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&templatenamestr, show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&templatenamestr));
esb_destructor(&templatenamestr);
if ( glflags.gf_check_names && checking_this_compiler()) {
int local_show_form = FALSE;
int local_verbose = 0;
struct esb_s lesb;
const char *name = 0;
esb_constructor(&lesb);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&lesb, local_show_form,local_verbose);
/* Look for specific name forms, attempting to
notice and report 'odd' identifiers. */
name = esb_get_string(&lesb);
DWARF_CHECK_COUNT(names_result,1);
if (!strcmp("\"(null)\"",name)) {
DWARF_CHECK_ERROR(names_result,
"string attribute is \"(null)\".");
} else {
if (!dot_ok_in_identifier(tag,die,name)
&& !need_CU_name && strchr(name,'.')) {
/* This is a suggestion there 'might' be
a surprising name, not a guarantee of an
error. */
DWARF_CHECK_ERROR(names_result,
"string attribute is invalid.");
}
}
esb_destructor(&lesb);
}
}
/* If we are in checking mode and we do not have a PU name */
if (( glflags.gf_check_locations || glflags.gf_check_ranges) &&
seen_PU && !PU_name[0]) {
int local_show_form = FALSE;
int local_verbose = 0;
const char *name = 0;
struct esb_s lesb;
esb_constructor(&lesb);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&lesb, local_show_form,local_verbose);
name = esb_get_string(&lesb);
safe_strcpy(PU_name,sizeof(PU_name),name,strlen(name));
esb_destructor(&lesb);
}
/* If we are processing the compile unit, record the name */
if (seen_CU && need_CU_name) {
/* Lets not get the form name included. */
struct esb_s lesb;
int local_show_form_used = FALSE;
int local_verbose = 0;
esb_constructor(&lesb);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&lesb, local_show_form_used,local_verbose);
safe_strcpy(CU_name,sizeof(CU_name),
esb_get_string(&lesb),esb_string_len(&lesb));
need_CU_name = FALSE;
esb_destructor(&lesb);
}
break;
case DW_AT_producer:
{
struct esb_s lesb;
esb_constructor(&lesb);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&lesb, show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&lesb));
esb_destructor(&lesb);
/* If we are in checking mode, identify the compiler */
if ( glflags.gf_do_check_dwarf || glflags.gf_search_is_on) {
/* Do not use show-form here! We just want the producer name, not
the form name. */
int show_form_local = FALSE;
int local_verbose = 0;
struct esb_s local_e;
esb_constructor(&local_e);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt,
&local_e, show_form_local,local_verbose);
/* Check if this compiler version is a target */
update_compiler_target(esb_get_string(&local_e));
esb_destructor(&local_e);
}
}
break;
/* When dealing with linkonce symbols, the low_pc and high_pc
are associated with a specific symbol; SNC always generate a name with
DW_AT_MIPS_linkage_name; GCC does not; instead gcc generates
DW_AT_abstract_origin or DW_AT_specification; in that case we have to
traverse this attribute in order to get the name for the linkonce */
case DW_AT_specification:
case DW_AT_abstract_origin:
case DW_AT_type:
{
struct esb_s lesb;
esb_constructor(&lesb);
get_attr_value(dbg, tag, die,
dieprint_cu_goffset,attrib, srcfiles, cnt, &lesb,
show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&lesb));
esb_destructor(&lesb);
if (glflags.gf_check_forward_decl ||
glflags.gf_check_self_references ||
glflags.gf_search_is_on) {
Dwarf_Off die_goff = 0;
Dwarf_Off ref_goff = 0;
int res = 0;
int suppress_check = 0;
Dwarf_Half theform = 0;
Dwarf_Half directform = 0;
get_form_values(dbg,attrib,&theform,&directform);
res = dwarf_global_formref(attrib, &ref_goff, &paerr);
if (res == DW_DLV_ERROR) {
int myerr = dwarf_errno(paerr);
if (myerr == DW_DLE_REF_SIG8_NOT_HANDLED) {
/* DW_DLE_REF_SIG8_NOT_HANDLED */
/* No offset available, it makes little sense
to delve into this sort of reference unless
we think a graph of self-refs *across*
type-units is possible. Hmm. FIXME? */
suppress_check = 1 ;
DWARF_CHECK_COUNT(self_references_result,1);
DWARF_CHECK_ERROR(self_references_result,
"DW_AT_ref_sig8 not handled so "
"self references not fully checked");
dwarf_dealloc(dbg,paerr,DW_DLA_ERROR);
paerr = 0;
} else {
print_error(dbg, "dwarf_die_CU_offsetD", res, paerr);
}
} else if (res == DW_DLV_NO_ENTRY) {
print_error(dbg, "dwarf_die_CU_offsetD (NO ENTRY)", res, paerr);
}
res = dwarf_dieoffset(die, &die_goff, &paerr);
if (res != DW_DLV_OK) {
print_error(dbg, "ref formwith no ref?!", res, paerr);
}
if (!suppress_check &&
glflags.gf_check_self_references &&
form_refers_local_info(theform) ) {
Dwarf_Die ref_die = 0;
ResetBucketGroup(pVisitedInfo);
AddEntryIntoBucketGroup(pVisitedInfo,die_goff,0,0,0,NULL,FALSE);
/* Follow reference chain, looking for self references */
res = dwarf_offdie_b(dbg,ref_goff,is_info,&ref_die,&paerr);
if (res == DW_DLV_OK) {
Dwarf_Off ref_die_cu_goff = 0;
Dwarf_Off die_loff = 0; /* CU-relative. */
if (dump_visited_info) {
res = dwarf_die_CU_offset(die, &die_loff, &paerr);
DROP_ERROR_INSTANCE(dbg,res,paerr);
do_dump_visited_info(die_indent_level,
die_loff,die_goff,
dieprint_cu_goffset,
atname,esb_get_string(&valname));
}
++die_indent_level;
res =dwarf_CU_dieoffset_given_die(ref_die,
&ref_die_cu_goff, &paerr);
/* Check above call return status? FIXME */
if (res != DW_DLV_OK) {
print_error(dbg,"dwarf_CU_die_dieoffset_given_die()"
" accessing cu_goff die!",
res, paerr);
}
traverse_one_die(dbg,attrib,ref_die,
ref_die_cu_goff,
is_info,srcfiles,cnt,die_indent_level);
dwarf_dealloc(dbg,ref_die,DW_DLA_DIE);
ref_die = 0;
--die_indent_level;
}
DeleteKeyInBucketGroup(pVisitedInfo,die_goff);
}
if (!suppress_check && glflags.gf_check_forward_decl) {
if (attr == DW_AT_specification) {
/* Check the DW_AT_specification does not make forward
references to DIEs.
DWARF4 specifications, section 2.13.2,
but really they are legal,
this test is probably wrong. */
DWARF_CHECK_COUNT(forward_decl_result,1);
if (ref_goff > die_goff) {
DWARF_CHECK_ERROR2(forward_decl_result,
"Invalid forward reference to DIE: ",
esb_get_string(&valname));
}
}
}
/* When doing search, if the attribute is DW_AT_specification or
DW_AT_abstract_origin, get any name associated with the DIE
referenced in the offset. The 2 more typical cases are:
Member functions, where 2 DIES are generated:
DIE for the declaration and DIE for the definition
and connected via the DW_AT_specification.
Inlined functions, where 2 DIES are generated:
DIE for the concrete instance and DIE for the abstract
instance and connected via the DW_AT_abstract_origin.
*/
if ( glflags.gf_search_is_on && (attr == DW_AT_specification ||
attr == DW_AT_abstract_origin)) {
Dwarf_Die ref_die = 0;
/* Follow reference chain, looking for the DIE name */
res = dwarf_offdie_b(dbg,ref_goff,is_info,&ref_die,&paerr);
if (res == DW_DLV_OK) {
/* Get the DIE name */
char *name = 0;
res = dwarf_diename(ref_die,&name,&paerr);
if (res == DW_DLV_OK) {
esb_empty_string(&valname);
esb_append(&valname,name);
}
/* Release the allocated DIE */
dwarf_dealloc(dbg,ref_die,DW_DLA_DIE);
}
}
}
/* If we are in checking mode and we do not have a PU name */
if (( glflags.gf_check_locations || glflags.gf_check_ranges) &&
seen_PU && !PU_name[0]) {
if (tag == DW_TAG_subprogram) {
/* This gets the DW_AT_name if this DIE has one. */
Dwarf_Addr low_pc = 0;
static char proc_name[BUFSIZ];
proc_name[0] = 0;
get_proc_name(dbg,die,low_pc,proc_name,BUFSIZ,/*pcMap=*/0);
if (proc_name[0]) {
safe_strcpy(PU_name,sizeof(PU_name),proc_name,
strlen(proc_name));
}
}
}
}
break;
default:
{
struct esb_s lesb;
esb_constructor(&lesb);
get_attr_value(dbg, tag,die,
dieprint_cu_goffset,attrib, srcfiles, cnt, &lesb,
show_form_used,verbose);
esb_empty_string(&valname);
esb_append(&valname, esb_get_string(&lesb));
esb_destructor(&lesb);
}
break;
}
if (!print_information) {
if (have_a_search_match(esb_get_string(&valname),atname)) {
/* Count occurrence of text */
++search_occurrences;
if ( glflags.gf_search_wide_format) {
found_search_attr = TRUE;
} else {
PRINT_CU_INFO();
bTextFound = TRUE;
}
}
}
if ((PRINTING_UNIQUE && PRINTING_DIES && print_information) || bTextFound) {
/* Print just the Tags and Attributes */
if (!glflags.gf_display_offsets) {
printf("%-28s\n",atname);
} else {
if (dense) {
printf(" %s<%s>", atname, esb_get_string(&valname));
if (append_extra_string) {
char *v = esb_get_string(&esb_extra);
printf("%s", v);
}
} else {
printf("%-28s", atname);
if (strlen(atname) >= 28) {
printf(" ");
}
printf("%s\n", sanitized(esb_get_string(&valname)));
if (append_extra_string) {
char *v = esb_get_string(&esb_extra);
printf("%s", sanitized(v));
}
}
}
bTextFound = FALSE;
}
esb_destructor(&valname);
esb_destructor(&esb_extra);
return found_search_attr;
}
void
dwarfdump_print_one_locdesc(Dwarf_Debug dbg,
Dwarf_Locdesc * llbuf, /* Non-zero for old interface. */
Dwarf_Locdesc_c locdesc, /* Non-zero for 2015 interface. */
UNUSEDARG Dwarf_Unsigned llent, /* Which desc we have . */
Dwarf_Unsigned entrycount, /* How many location operators (DW_OP)? */
Dwarf_Addr baseaddr,
struct esb_s *string_out)
{
Dwarf_Half no_of_ops = 0;
unsigned i = 0;
if(llbuf) {
Dwarf_Locdesc *locd = 0;
locd = llbuf;
no_of_ops = llbuf->ld_cents;
for (i = 0; i < no_of_ops; i++) {
Dwarf_Loc * op = &locd->ld_s[i];
int res = _dwarf_print_one_expr_op(dbg,op,NULL,i,
baseaddr,string_out);
if (res == DW_DLV_ERROR) {
return;
}
}
return;
}
/* ASSERT: locs != NULL */
no_of_ops = entrycount;
for (i = 0; i < no_of_ops; i++) {
int res = 0;
res = _dwarf_print_one_expr_op(dbg,NULL,locdesc,i,
baseaddr,string_out);
if (res == DW_DLV_ERROR) {
return;
}
}
}
static int
op_has_no_operands(int op)
{
unsigned i = 0;
if (op >= DW_OP_lit0 && op <= DW_OP_reg31) {
return TRUE;
}
for (; ; ++i) {
struct operation_descr_s *odp = opdesc+i;
if (odp->op_code == 0) {
break;
}
if (odp->op_code != op) {
continue;
}
if (odp->op_count == 0) {
return TRUE;
}
return FALSE;
}
return FALSE;
}
static void
show_contents(struct esb_s *string_out,
unsigned int length,const unsigned char * bp)
{
unsigned int i = 0;
char small_buf[20];
if(!length) {
return;
}
esb_append(string_out," contents 0x");
for (; i < length; ++i,++bp) {
/* Do not use DW_PR_DUx here,
the value *bp is a const unsigned char. */
snprintf(small_buf, sizeof(small_buf),
"%02x", *bp);
esb_append(string_out,small_buf);
}
}
int
_dwarf_print_one_expr_op(Dwarf_Debug dbg,
Dwarf_Loc* expr,
Dwarf_Locdesc_c exprc,
int index,
UNUSEDARG Dwarf_Addr baseaddr,
struct esb_s *string_out)
{
/* local_space_needed is intended to be 'more than big enough'
for a short group of loclist entries. */
char small_buf[100];
Dwarf_Small op = 0;
Dwarf_Unsigned opd1 = 0;
Dwarf_Unsigned opd2 = 0;
Dwarf_Unsigned opd3 = 0;
Dwarf_Unsigned offsetforbranch = 0;
const char * op_name = 0;
Dwarf_Error onexerr = 0;
if (index > 0) {
esb_append(string_out, " ");
}
if (expr) {
/* DWARF 2,3,4 style */
op = expr->lr_atom;
opd1 = expr->lr_number;
opd2 = expr->lr_number2;
} else {
/* DWARF 2,3,4 and DWARF5 style */
int res = dwarf_get_location_op_value_c(exprc,
index, &op,&opd1,&opd2,&opd3,&offsetforbranch,
&onexerr);
if (res != DW_DLV_OK) {
print_error(dbg,
"dwarf_get_location_op_value_c unexpected value!",
DW_DLV_OK, onexerr);
return DW_DLV_ERROR;
}
}
op_name = get_OP_name(op,pd_dwarf_names_print_on_error);
esb_append(string_out, op_name);
if (op_has_no_operands(op)) {
/* Nothing to add. */
} else if (op >= DW_OP_breg0 && op <= DW_OP_breg31) {
snprintf(small_buf, sizeof(small_buf),
"%+" DW_PR_DSd , (Dwarf_Signed) opd1);
esb_append(string_out, small_buf);
} else {
switch (op) {
case DW_OP_addr:
bracket_hex(" ",opd1,"",string_out);
break;
case DW_OP_const1s:
case DW_OP_const2s:
case DW_OP_const4s:
case DW_OP_const8s:
case DW_OP_consts:
case DW_OP_skip:
case DW_OP_bra:
case DW_OP_fbreg:
esb_append(string_out," ");
formx_signed(opd1,string_out);
break;
case DW_OP_GNU_addr_index: /* unsigned val */
case DW_OP_addrx: /* DWARF5: unsigned val */
case DW_OP_GNU_const_index:
case DW_OP_constx: /* DWARF5: unsigned val */
case DW_OP_const1u:
case DW_OP_const2u:
case DW_OP_const4u:
case DW_OP_const8u:
case DW_OP_constu:
case DW_OP_pick:
case DW_OP_plus_uconst:
case DW_OP_regx:
case DW_OP_piece:
case DW_OP_deref_size:
case DW_OP_xderef_size:
snprintf(small_buf, sizeof(small_buf),
" %" DW_PR_DUu , opd1);
esb_append(string_out, small_buf);
break;
case DW_OP_bregx:
bracket_hex(" ",opd1,"",string_out);
esb_append(string_out,"+");
formx_signed(opd2,string_out);
break;
case DW_OP_call2:
bracket_hex(" ",opd1,"",string_out);
break;
case DW_OP_call4:
bracket_hex(" ",opd1,"",string_out);
break;
case DW_OP_call_ref:
bracket_hex(" ",opd1,"",string_out);
break;
case DW_OP_bit_piece:
bracket_hex(" ",opd1,"",string_out);
bracket_hex(" offset ",opd2,"",string_out);
break;
case DW_OP_implicit_value:
{
#define IMPLICIT_VALUE_PRINT_MAX 12
unsigned int print_len = 0;
bracket_hex(" ",opd1,"",string_out);
/* The other operand is a block of opd1 bytes. */
/* FIXME */
print_len = opd1;
if (print_len > IMPLICIT_VALUE_PRINT_MAX) {
print_len = IMPLICIT_VALUE_PRINT_MAX;
}
#undef IMPLICIT_VALUE_PRINT_MAX
{
const unsigned char *bp = 0;
/* This is a really ugly cast, a way
to implement DW_OP_implicit value in
this libdwarf context. */
bp = (const unsigned char *) opd2;
show_contents(string_out,print_len,bp);
}
}
break;
/* We do not know what the operands, if any, are. */
case DW_OP_HP_unknown:
case DW_OP_HP_is_value:
case DW_OP_HP_fltconst4:
case DW_OP_HP_fltconst8:
case DW_OP_HP_mod_range:
case DW_OP_HP_unmod_range:
case DW_OP_HP_tls:
case DW_OP_INTEL_bit_piece:
break;
case DW_OP_stack_value: /* DWARF4 */
break;
case DW_OP_GNU_uninit: /* DW_OP_APPLE_uninit */
/* No operands. */
break;
case DW_OP_GNU_encoded_addr:
bracket_hex(" ",opd1,"",string_out);
break;
case DW_OP_implicit_pointer: /* DWARF5 */
case DW_OP_GNU_implicit_pointer:
bracket_hex(" ",opd1,"",string_out);
esb_append(string_out, " ");
formx_signed(opd2,string_out);
break;
case DW_OP_entry_value: /* DWARF5 */
case DW_OP_GNU_entry_value: {
const unsigned char *bp = 0;
unsigned int length = 0;
length = opd1;
bracket_hex(" ",opd1,"",string_out);
bp = (Dwarf_Small *) opd2;
if (!bp) {
esb_append(string_out,
"ERROR: Null databyte pointer DW_OP_entry_value ");
} else {
show_contents(string_out,length,bp);
}
}
break;
case DW_OP_const_type: /* DWARF5 */
case DW_OP_GNU_const_type:
{
const unsigned char *bp = 0;
unsigned int length = 0;
bracket_hex(" ",opd1,"",string_out);
length = opd2;
esb_append(string_out," const length: ");
snprintf(small_buf, sizeof(small_buf),
"%u" , length);
esb_append(string_out, small_buf);
/* Now point to the data bytes of the const. */
bp = (Dwarf_Small *) opd3;
if (!bp) {
esb_append(string_out,
"ERROR: Null databyte pointer DW_OP_const_type ");
} else {
show_contents(string_out,length,bp);
}
}
break;
case DW_OP_regval_type: /* DWARF5 */
case DW_OP_GNU_regval_type: {
snprintf(small_buf, sizeof(small_buf),
" 0x%" DW_PR_DUx , opd1);
esb_append(string_out, small_buf);
bracket_hex(" ",opd2,"",string_out);
}
break;
case DW_OP_deref_type: /* DWARF5 */
case DW_OP_GNU_deref_type: {
snprintf(small_buf, sizeof(small_buf),
" 0x%02" DW_PR_DUx , opd1);
esb_append(string_out, small_buf);
bracket_hex(" ",opd2,"",string_out);
}
break;
case DW_OP_convert: /* DWARF5 */
case DW_OP_GNU_convert:
case DW_OP_reinterpret: /* DWARF5 */
case DW_OP_GNU_reinterpret:
case DW_OP_GNU_parameter_ref:
snprintf(small_buf, sizeof(small_buf),
" 0x%02" DW_PR_DUx , opd1);
esb_append(string_out, small_buf);
break;
default:
{
snprintf(small_buf, sizeof(small_buf),
" dwarf_op unknown 0x%x", (unsigned)op);
esb_append(string_out,small_buf);
}
break;
}
}
return DW_DLV_OK;
}
static void
loc_error_check(UNUSEDARG Dwarf_Debug dbg,
Dwarf_Addr lopcfinal,
Dwarf_Addr lopc,
Dwarf_Addr hipcfinal,
Dwarf_Addr hipc,
Dwarf_Unsigned offset,
Dwarf_Addr base_address,
Dwarf_Bool *bError)
{
DWARF_CHECK_COUNT(locations_result,1);
/* Check the low_pc and high_pc are within
a valid range in the .text section */
if (IsValidInBucketGroup(pRangesInfo,lopcfinal) &&
IsValidInBucketGroup(pRangesInfo,hipcfinal)) {
/* Valid values; do nothing */
} else {
/* At this point may be we are dealing with
a linkonce symbol */
if (IsValidInLinkonce(pLinkonceInfo,PU_name,
lopcfinal,hipcfinal)) {
/* Valid values; do nothing */
} else {
*bError = TRUE;
DWARF_CHECK_ERROR(locations_result,
".debug_loc: Address outside a "
"valid .text range");
if ( glflags.gf_check_verbose_mode && PRINTING_UNIQUE) {
printf(
"Offset = 0x%" DW_PR_XZEROS DW_PR_DUx
", Base = 0x%" DW_PR_XZEROS DW_PR_DUx ", "
"Low = 0x%" DW_PR_XZEROS DW_PR_DUx
" (0x%" DW_PR_XZEROS DW_PR_DUx
"), High = 0x%" DW_PR_XZEROS DW_PR_DUx
" (0x%" DW_PR_XZEROS DW_PR_DUx ")\n",
offset,base_address,lopcfinal,
lopc,
hipcfinal,
hipc);
}
}
}
}
static const char *
adexplain(Dwarf_Unsigned liberr,
const char * alterr)
{
if (liberr == DW_DLE_MISSING_NEEDED_DEBUG_ADDR_SECTION) {
return "no-tied-debug-addr-available";
}
return alterr;
}
/* Fill buffer with location lists
Buffer esbp expands as needed.
*/
/*ARGSUSED*/ static void
get_location_list(Dwarf_Debug dbg,
Dwarf_Die die,
Dwarf_Attribute attr,
struct esb_s *esbp)
{
Dwarf_Locdesc *llbuf = 0;
Dwarf_Locdesc **llbufarray = 0; /* Only for older interface. */
Dwarf_Unsigned no_of_elements;
Dwarf_Loc_Head_c loclist_head = 0; /* 2015 loclist interface */
Dwarf_Error llerr = 0;
Dwarf_Unsigned i = 0;
int lres = 0;
unsigned llent = 0;
/* Base address used to update entries in .debug_loc.
CU_base_address is a global. Terrible way to
pass in this value. FIXME. See also CU_low_address
as base address is special for address ranges */
Dwarf_Addr base_address = CU_base_address;
Dwarf_Addr lopc = 0;
Dwarf_Addr hipc = 0;
Dwarf_Bool bError = FALSE;
Dwarf_Small lle_value = 0; /* DWARF5 */
Dwarf_Small loclist_source = 0;
/* This is the section offset of the expression, not
the location description prefix. */
Dwarf_Unsigned section_offset = 0;
Dwarf_Half elf_address_size = 0;
Dwarf_Addr elf_max_address = 0;
/* old and new interfaces differ on signedness. */
Dwarf_Signed locentry_count = 0;
Dwarf_Unsigned ulocentry_count = 0;
Dwarf_Bool checking = FALSE;
if (!glflags.gf_use_old_dwarf_loclist) {
lres = dwarf_get_loclist_c(attr,&loclist_head,
&no_of_elements,&llerr);
if (lres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_get_loclist_c", lres, llerr);
} else if (lres == DW_DLV_NO_ENTRY) {
return;
}
} else {
Dwarf_Signed sno = 0;
lres = dwarf_loclist_n(attr, &llbufarray, &sno, &llerr);
if (lres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_loclist", lres, llerr);
} else if (lres == DW_DLV_NO_ENTRY) {
return;
}
no_of_elements = sno;
}
get_address_size_and_max(dbg,&elf_address_size,&elf_max_address,&llerr);
for (llent = 0; llent < no_of_elements; ++llent) {
char small_buf[150];
Dwarf_Unsigned locdesc_offset = 0;
Dwarf_Locdesc_c locentry = 0; /* 2015 */
Dwarf_Addr lopcfinal = 0;
Dwarf_Addr hipcfinal = 0;
if (!glflags.gf_use_old_dwarf_loclist) {
lres = dwarf_get_locdesc_entry_c(loclist_head,
llent,
&lle_value,
&lopc, &hipc,
&ulocentry_count,
&locentry,
&loclist_source,
&section_offset,
&locdesc_offset,
&llerr);
if (lres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_get_loclist_entry_c", lres, llerr);
} else if (lres == DW_DLV_NO_ENTRY) {
return;
}
locentry_count = ulocentry_count;
} else {
llbuf = llbufarray[llent];
lopc = llbuf->ld_lopc;
hipc = llbuf->ld_hipc;
loclist_source = llbuf->ld_from_loclist;
section_offset = llbuf->ld_section_offset;
locdesc_offset = section_offset -
sizeof(Dwarf_Half) - 2 * elf_address_size;
locentry_count = llbuf->ld_cents;
ulocentry_count = locentry_count;
if (lopc == elf_max_address) {
lle_value = DW_LLEX_base_address_selection_entry;
} else if (lopc== 0 && hipc == 0) {
lle_value = DW_LLEX_end_of_list_entry;
} else {
lle_value = DW_LLEX_offset_pair_entry;
}
}
if (!dense && loclist_source) {
if (llent == 0) {
if (loclist_source == 1) {
snprintf(small_buf, sizeof(small_buf),
"<loclist at offset 0x%"
DW_PR_XZEROS DW_PR_DUx
" with %ld entries follows>",
locdesc_offset,
(long) no_of_elements);
} else {
/* ASSERT: loclist_source == 2 */
snprintf(small_buf, sizeof(small_buf),
"<dwo loclist at offset 0x%"
DW_PR_XZEROS DW_PR_DUx
" with %ld entries follows>",
locdesc_offset,
(long) no_of_elements);
}
esb_append(esbp, small_buf);
}
esb_append(esbp, "\n\t\t\t");
snprintf(small_buf, sizeof(small_buf), "[%2d]", llent);
esb_append(esbp, small_buf);
}
/* If we have a location list refering to the .debug_loc
Check for specific compiler we are validating. */
if ( glflags.gf_check_locations && in_valid_code &&
loclist_source && checking_this_compiler()) {
checking = TRUE;
}
/* When dwarf_debug_addr_index_to_addr() fails
it is probably DW_DLE_MISSING_NEEDED_DEBUG_ADDR_SECTION 257
(because no TIED file supplied)
but we don't distinguish that from other errors here. */
if(loclist_source || checking) {
/* Simplifies to use the DWARF5 DW_LLE as the test.*/
if (lle_value == DW_LLEX_base_address_selection_entry) {
/* (0xffffffff,addr), use specific address
(current PU address) */
Dwarf_Addr realaddr = 0;
if (loclist_source == 2) {
/* hipc is index of a slot in .debug_addr section.
which contains base_address. */
int res = dwarf_debug_addr_index_to_addr(die,
hipc,&realaddr,&llerr);
if(res == DW_DLV_OK) {
base_address = realaddr;
} else if(res == DW_DLV_ERROR) {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" %s>",hipc,
adexplain(dwarf_errno(llerr),
"base-address-unavailable"));
esb_append(esbp,small_buf);
base_address = 0;
} else {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" no-entry finding index >",hipc);
esb_append(esbp,small_buf);
/* Cannot find .debug_addr */
base_address = 0;
}
snprintf(small_buf,sizeof(small_buf),
"<index to debug_addr : 0x%"
DW_PR_XZEROS DW_PR_DUx
" new base address 0x%"
DW_PR_XZEROS DW_PR_DUx
">",
hipc,base_address);
esb_append(esbp,small_buf);
} else {
base_address = hipc;
snprintf(small_buf,sizeof(small_buf),
"<new base address 0x%"
DW_PR_XZEROS DW_PR_DUx
">",
base_address);
esb_append(esbp,small_buf);
}
} else if (lle_value == DW_LLEX_end_of_list_entry) {
/* Nothing to do. */
esb_append(esbp,"<end-of-list>");
} else if (lle_value == DW_LLEX_start_length_entry) {
int foundaddr = FALSE;
if (loclist_source == 2) {
Dwarf_Addr realaddr = 0;
Dwarf_Addr slotindex = lopc;
/* start (lopc) is index of a slot
in .debug_addr section. */
int res = dwarf_debug_addr_index_to_addr(die,
lopc,&realaddr,&llerr);
if(res == DW_DLV_OK) {
lopc = realaddr;
foundaddr = TRUE;
} else if(res == DW_DLV_ERROR) {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" %s>",lopc,
adexplain(dwarf_errno(llerr),
"start-address-unavailable"));
esb_append(esbp,small_buf);
} else {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" no-entry finding start index >",lopc);
esb_append(esbp,small_buf);
/* Cannot find .debug_addr */
lopc = 0;
}
snprintf(small_buf,sizeof(small_buf),
"<start-length index to debug_addr : 0x%"
DW_PR_XZEROS DW_PR_DUx
" addr 0x%"
DW_PR_XZEROS DW_PR_DUx
" length 0x%"
DW_PR_XZEROS DW_PR_DUx
"> ",
slotindex,realaddr,hipc);
esb_append(esbp,small_buf);
} else {
esb_append(esbp,"<Impossible start-length entry>");
/* Impossible */
lopc = 0;
}
lopcfinal = lopc;
hipcfinal = lopcfinal + hipc;
if (checking && foundaddr) {
loc_error_check(dbg,lopcfinal, lopc,
hipcfinal, hipc, locdesc_offset, base_address,
&bError);
}
} else if (lle_value == DW_LLEX_offset_pair_entry) {
/* Same for both loclist_source. */
lopcfinal = lopc + base_address;
hipcfinal = hipc + base_address;
snprintf(small_buf,sizeof(small_buf),
"< offset pair low-off : 0x%"
DW_PR_XZEROS DW_PR_DUx
" addr 0x%"
DW_PR_XZEROS DW_PR_DUx
" high-off 0x%"
DW_PR_XZEROS DW_PR_DUx
" addr 0x%"
DW_PR_XZEROS DW_PR_DUx
">",
lopc,lopcfinal,hipc,hipcfinal);
esb_append(esbp,small_buf);
if(checking) {
loc_error_check(dbg,lopcfinal, lopc,
hipcfinal, hipc, locdesc_offset, base_address,
&bError);
}
} else if (lle_value == DW_LLEX_start_end_entry) {
int foundaddr = FALSE;
/* These are NOT relative to base_address */
if (loclist_source == 2) {
/* indices in .debug_addr of start and end
addresses. */
Dwarf_Addr reallo = 0;
Dwarf_Addr realhi = 0;
/* start is index of a slot in .debug_addr section. */
int res = dwarf_debug_addr_index_to_addr(die,
lopc,&reallo,&llerr);
if(res == DW_DLV_OK) {
lopcfinal = reallo;
foundaddr = TRUE;
} else if(res == DW_DLV_ERROR) {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" %s>",lopc,
adexplain(dwarf_errno(llerr),
"start-address-unavailable"));
esb_append(esbp,small_buf);
} else {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" error finding start index >",lopc);
esb_append(esbp,small_buf);
/* Cannot find .debug_addr */
lopcfinal = 0;
}
res = dwarf_debug_addr_index_to_addr(die,
hipc,&realhi,&llerr);
if(res == DW_DLV_OK) {
hipcfinal = realhi;
} else if(res == DW_DLV_ERROR) {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" %s>",hipc,
adexplain(dwarf_errno(llerr),
"end-address-unavailable"));
esb_append(esbp,small_buf);
foundaddr = FALSE;
} else {
snprintf(small_buf,sizeof(small_buf),
"<debug_addr index 0x%"
DW_PR_XZEROS DW_PR_DUx
" problem-finding-end-address >",hipc);
esb_append(esbp,small_buf);
/* Cannot find .debug_addr */
hipcfinal = 0;
foundaddr = FALSE;
}
snprintf(small_buf,sizeof(small_buf),
"< start-end low-index : 0x%"
DW_PR_XZEROS DW_PR_DUx
" addr 0x%"
DW_PR_XZEROS DW_PR_DUx
" high-index 0x%"
DW_PR_XZEROS DW_PR_DUx
" addr 0x%"
DW_PR_XZEROS DW_PR_DUx
">",
lopc,lopcfinal,hipc,hipcfinal);
esb_append(esbp,small_buf);
} else {
esb_append(esbp,"<Impossible start-end entry>");
/* Impossible */
}
if (checking && foundaddr) {
loc_error_check(dbg,lopcfinal, lopc,
hipcfinal, hipc, locdesc_offset, 0,
&bError);
}
} else {
snprintf(small_buf,sizeof(small_buf),
"Unexpected LLEX code 0x%x, ERROR",lle_value);
print_error(dbg, small_buf, DW_DLV_OK, llerr);
}
if (glflags.gf_display_offsets && verbose) {
char *secname = ".debug_info";
if(loclist_source == 1) {
secname = ".debug_loc";
} else if (loclist_source == 2) {
secname = ".debug_loc.dwo";
} else if (loclist_source) {
secname = "<unknown location entry code. ERROR.>";
}
snprintf(small_buf, sizeof(small_buf),
"<from %s offset 0x%" DW_PR_XZEROS DW_PR_DUx ">",
secname,
locdesc_offset);
esb_append(esbp, small_buf);
}
}
dwarfdump_print_one_locdesc(dbg,
/* Either llbuf or locentry non-zero.
Not both. */
llbuf,
locentry,
llent, /* Which loc desc this is */
locentry_count, /* How many ops in this loc desc */
base_address,
esbp);
}
if (bError && glflags.gf_check_verbose_mode && PRINTING_UNIQUE) {
printf("\n");
}
if (!glflags.gf_use_old_dwarf_loclist) {
dwarf_loc_head_c_dealloc(loclist_head);
} else {
for (i = 0; i < no_of_elements; ++i) {
dwarf_dealloc(dbg, llbufarray[i]->ld_s, DW_DLA_LOC_BLOCK);
dwarf_dealloc(dbg, llbufarray[i], DW_DLA_LOCDESC);
}
dwarf_dealloc(dbg, llbufarray, DW_DLA_LIST);
}
}
/* New October 2017.
The 'decimal' representation here is questionable.
*/
static void
formx_data16(Dwarf_Form_Data16 * u,
struct esb_s *esbp, Dwarf_Bool hex_format)
{
char small_buf[20];
unsigned i = 0;
for( ; i < sizeof(Dwarf_Form_Data16); ++i){
esb_append(esbp, "0x");
if (hex_format) {
snprintf(small_buf, sizeof(small_buf),"%02x ",
u->fd_data[i]);
} else {
snprintf(small_buf, sizeof(small_buf),"%02d ",
u->fd_data[i]);
}
esb_append(esbp, small_buf);
}
}
static void
formx_unsigned(Dwarf_Unsigned u, struct esb_s *esbp, Dwarf_Bool hex_format)
{
char small_buf[40];
if (hex_format) {
snprintf(small_buf, sizeof(small_buf),"0x%" DW_PR_XZEROS DW_PR_DUx , u);
} else {
snprintf(small_buf, sizeof(small_buf),
"%" DW_PR_DUu , u);
}
esb_append(esbp, small_buf);
}
static void
formx_signed(Dwarf_Signed s, struct esb_s *esbp)
{
char small_buf[40];
snprintf(small_buf, sizeof(small_buf),
"%" DW_PR_DSd ,s);
esb_append(esbp, small_buf);
}
static void
formx_unsigned_and_signed_if_neg(Dwarf_Unsigned tempud,
Dwarf_Signed tempd,
const char *leader,Dwarf_Bool hex_format,struct esb_s*esbp)
{
formx_unsigned(tempud,esbp,hex_format);
if(tempd < 0) {
esb_append(esbp,leader);
formx_signed(tempd,esbp);
esb_append(esbp,")");
}
}
/* If the DIE DW_AT_type exists and is directly known signed/unsigned
return -1 for signed 1 for unsigned.
Otherwise return 0 meaning 'no information'.
So we only need to a messy lookup once per type-die offset */
static int
check_for_type_unsigned(Dwarf_Debug dbg,
Dwarf_Die die,
UNUSEDARG struct esb_s *esbp)
{
Dwarf_Bool is_info = 0;
struct Helpertree_Base_s * helperbase = 0;
struct Helpertree_Map_Entry_s *e = 0;
int res = 0;
Dwarf_Attribute attr = 0;
Dwarf_Attribute encodingattr = 0;
Dwarf_Error error = 0;
Dwarf_Unsigned diegoffset = 0;
Dwarf_Unsigned typedieoffset = 0;
Dwarf_Die typedie = 0;
Dwarf_Unsigned tempud = 0;
int show_form_here = FALSE;
int retval = 0;
if(!die) {
return 0;
}
is_info = dwarf_get_die_infotypes_flag(die);
if(is_info) {
helperbase = &helpertree_offsets_base_info;
} else {
helperbase = &helpertree_offsets_base_types;
}
res = dwarf_dieoffset(die,&diegoffset,&error);
if (res == DW_DLV_ERROR) {
/* esb_append(esbp,"<helper dieoffset FAIL >"); */
return 0;
} else if (res == DW_DLV_NO_ENTRY) {
/* We don't know sign. */
/*esb_append(esbp,"<helper dieoffset NO ENTRY>"); */
return 0;
}
/* This might be wrong. See the typedieoffset check below,
which is correct... */
e = helpertree_find(diegoffset,helperbase);
if(e) {
/*bracket_hex("<helper FOUND offset ",diegoffset,">",esbp);
bracket_hex("<helper FOUND val ",e->hm_val,">",esbp); */
return e->hm_val;
}
/* We look up the DW_AT_type die, if any, and
use that offset to check for signedness. */
res = dwarf_attr(die, DW_AT_type, &attr,&error);
if (res == DW_DLV_ERROR) {
/*bracket_hex("<helper dwarf_attr FAIL ",diegoffset,">",esbp); */
helpertree_add_entry(diegoffset, 0,helperbase);
return 0;
} else if (res == DW_DLV_NO_ENTRY) {
/* We don't know sign. */
/*bracket_hex( "<helper dwarf_attr no entry ",diegoffset,">",esbp); */
helpertree_add_entry(diegoffset, 0,helperbase);
return 0;
}
res = dwarf_global_formref(attr, &typedieoffset,&error);
if (res == DW_DLV_ERROR) {
/*bracket_hex( "<helper global_formreff FAIL" ,diegoffset,">",esbp); */
dwarf_dealloc(dbg,attr,DW_DLA_ATTR);
helpertree_add_entry(diegoffset, 0,helperbase);
return 0;
} else if (res == DW_DLV_NO_ENTRY) {
/*esb_append(esbp,"helper NO ENTRY FAIL ");
bracket_hex( "<helper global_formreff NO ENTRY" ,diegoffset,">",esbp); */
dwarf_dealloc(dbg,attr,DW_DLA_ATTR);
helpertree_add_entry(diegoffset, 0,helperbase);
return 0;
}
dwarf_dealloc(dbg,attr,DW_DLA_ATTR);
attr = 0;
e = helpertree_find(typedieoffset,helperbase);
if(e) {
/*bracket_hex("<helper FOUND typedieoffset ",typedieoffset,">",esbp);
bracket_hex("<helper FOUND val ",e->hm_val,">",esbp); */
return e->hm_val;
}
res = dwarf_offdie_b(dbg,typedieoffset,is_info, &typedie,&error);
if (res == DW_DLV_ERROR) {
/*bracket_hex( "<helper dwarf_offdie_b FAIL ",diegoffset,">",esbp); */
helpertree_add_entry(diegoffset, 0,helperbase);
helpertree_add_entry(typedieoffset, 0,helperbase);
return 0;
} else if (res == DW_DLV_NO_ENTRY) {
/*bracket_hex( "<helper dwarf_offdie_b NO ENTRY ",diegoffset,">",esbp); */
helpertree_add_entry(diegoffset, 0,helperbase);
helpertree_add_entry(typedieoffset, 0,helperbase);
return 0;
}
res = dwarf_attr(typedie, DW_AT_encoding, &encodingattr,&error);
if (res == DW_DLV_ERROR) {
/*bracket_hex( "<helper dwarf_attr typedie FAIL",diegoffset,">",esbp); */
dwarf_dealloc(dbg,typedie,DW_DLA_DIE);
helpertree_add_entry(diegoffset, 0,helperbase);
helpertree_add_entry(typedieoffset, 0,helperbase);
return 0;
} else if (res == DW_DLV_NO_ENTRY) {
/*bracket_hex( "<helper dwarf_attr typedie NO ENTRY",diegoffset,">",esbp);*/
dwarf_dealloc(dbg,typedie,DW_DLA_DIE);
helpertree_add_entry(diegoffset, 0,helperbase);
helpertree_add_entry(typedieoffset, 0,helperbase);
return 0;
}
res = get_small_encoding_integer_and_name(dbg,
encodingattr,
&tempud,
/* attrname */ (const char *) NULL,
/* err_string */ ( struct esb_s *) NULL,
(encoding_type_func) 0,
&error,show_form_here);
if (res != DW_DLV_OK) {
/*bracket_hex( "<helper small encoding FAIL",diegoffset,">",esbp);*/
dwarf_dealloc(dbg,typedie,DW_DLA_DIE);
dwarf_dealloc(dbg,encodingattr,DW_DLA_ATTR);
helpertree_add_entry(diegoffset, 0,helperbase);
helpertree_add_entry(typedieoffset, 0,helperbase);
return 0;
}
if (tempud == DW_ATE_signed || tempud == DW_ATE_signed_char) {
/*esb_append(esbp,"helper small encoding SIGNED ");*/
retval = -1;
} else {
if (tempud == DW_ATE_unsigned || tempud == DW_ATE_unsigned_char) {
/*esb_append(esbp,"helper small encoding UNSIGNED ");*/
retval = 1;
}
}
/*bracket_hex( "<helper ENTERED die",diegoffset,">",esbp);
bracket_hex( "<helper ENTERED typedie",typedieoffset,">",esbp);*/
helpertree_add_entry(diegoffset,retval,helperbase);
helpertree_add_entry(typedieoffset, retval,helperbase);
dwarf_dealloc(dbg,typedie,DW_DLA_DIE);
dwarf_dealloc(dbg,encodingattr,DW_DLA_ATTR);
return retval;
}
/* We think this is an integer. Figure out how to print it.
In case the signedness is ambiguous (such as on
DW_FORM_data1 (ie, unknown signedness) print two ways.
If we were to look at DW_AT_type in the base DIE
we could follow it and determine if the type
was unsigned or signed (usually easily) and
use that information.
*/
static int
formxdata_print_value(Dwarf_Debug dbg,
Dwarf_Die die,
Dwarf_Attribute attrib,
Dwarf_Half theform,
struct esb_s *esbp,
Dwarf_Error * pverr, Dwarf_Bool hex_format)
{
Dwarf_Signed tempsd = 0;
Dwarf_Unsigned tempud = 0;
int sres = 0;
int ures = 0;
Dwarf_Error serr = 0;
if (theform == DW_FORM_data16) {
Dwarf_Form_Data16 v16;
ures = dwarf_formdata16(attrib,
&v16,pverr);
if (ures == DW_DLV_OK) {
formx_data16(&v16,
esbp,hex_format);
return DW_DLV_OK;
} else if (ures == DW_DLV_NO_ENTRY) {
/* impossible */
return ures;
} else {
return ures;
}
}
ures = dwarf_formudata(attrib, &tempud, pverr);
sres = dwarf_formsdata(attrib, &tempsd, &serr);
if (ures == DW_DLV_OK) {
if (sres == DW_DLV_OK) {
if (tempud == (Dwarf_Unsigned)tempsd && tempsd >= 0) {
/* Data is the same value and not negative,
so makes no difference which
we print. */
formx_unsigned(tempud,esbp,hex_format);
} else {
/* Here we don't know if signed or not and
Assuming one or the other changes the
interpretation of the bits. */
int helpertree_unsigned = 0;
helpertree_unsigned = check_for_type_unsigned(dbg,die,esbp);
if (!die || !helpertree_unsigned) {
/* Signedness unclear. */
formx_unsigned_and_signed_if_neg(tempud,tempsd,
" (",hex_format,esbp);
} else if (helpertree_unsigned > 0) {
formx_unsigned(tempud,esbp,hex_format);
} else {
/* Value signed. */
formx_signed(tempsd,esbp);
}
}
} else if (sres == DW_DLV_NO_ENTRY) {
formx_unsigned(tempud,esbp,hex_format);
} else /* DW_DLV_ERROR */{
formx_unsigned(tempud,esbp,hex_format);
}
goto cleanup;
} else {
/* ures == DW_DLV_ERROR or DW_DLV_NO_ENTRY*/
if (sres == DW_DLV_OK) {
formx_signed(tempsd,esbp);
} else {
/* Neither worked. */
}
}
/* Clean up any unused Dwarf_Error data.
DW_DLV_NO_ENTRY cannot really happen,
so a complete cleanup for that is
not necessary. */
cleanup:
if (sres == DW_DLV_OK || ures == DW_DLV_OK) {
DROP_ERROR_INSTANCE(dbg,sres,serr);
DROP_ERROR_INSTANCE(dbg,ures,*pverr);
return DW_DLV_OK;
}
if (sres == DW_DLV_ERROR || ures == DW_DLV_ERROR) {
if (sres == DW_DLV_ERROR && ures == DW_DLV_ERROR) {
dwarf_dealloc(dbg,serr,DW_DLA_ERROR);
serr = 0;
return DW_DLV_ERROR;
}
if (sres == DW_DLV_ERROR) {
*pverr = serr;
serr = 0;
}
return DW_DLV_ERROR;
}
/* Both are DW_DLV_NO_ENTRY which is crazy, impossible. */
return DW_DLV_NO_ENTRY;
}
static void
bracket_hex(const char *s1,
Dwarf_Unsigned v,
const char *s2,
struct esb_s * esbp)
{
Dwarf_Bool hex_format = TRUE;
esb_append(esbp,s1);
formx_unsigned(v,esbp,hex_format);
esb_append(esbp,s2);
}
static char *
get_form_number_as_string(int form, char *buf, unsigned bufsize)
{
snprintf(buf,bufsize," %d",form);
return buf;
}
static void
print_exprloc_content(Dwarf_Debug dbg,Dwarf_Die die,
Dwarf_Attribute attrib,
int showhextoo, struct esb_s *esbp)
{
Dwarf_Ptr x = 0;
Dwarf_Unsigned tempud = 0;
char small_buf[80];
Dwarf_Error ecerr = 0;
int wres = 0;
wres = dwarf_formexprloc(attrib,&tempud,&x,&ecerr);
if (wres == DW_DLV_NO_ENTRY) {
/* Show nothing? Impossible. */
} else if (wres == DW_DLV_ERROR) {
print_error(dbg, "Cannot get a DW_FORM_exprloc....", wres, ecerr);
} else {
Dwarf_Half address_size = 0;
Dwarf_Half offset_size = 0;
Dwarf_Half version = 0;
int ares = 0;
unsigned u = 0;
snprintf(small_buf, sizeof(small_buf),
"len 0x%04" DW_PR_DUx ": ",tempud);
esb_append(esbp, small_buf);
if (showhextoo) {
for (u = 0; u < tempud; u++) {
snprintf(small_buf, sizeof(small_buf), "%02x",
*(u + (unsigned char *) x));
esb_append(esbp, small_buf);
}
esb_append(esbp,": ");
}
ares = dwarf_get_version_of_die(die,&version,&offset_size);
if (ares != DW_DLV_OK) {
print_error(dbg,"ERROR: Cannot get version size for exprloc die",
DW_DLV_ERROR,ecerr);
}
ares = dwarf_get_die_address_size(die,&address_size,&ecerr);
if (wres == DW_DLV_NO_ENTRY) {
print_error(dbg,"Cannot get die address size for exprloc",
ares,ecerr);
} else if (wres == DW_DLV_ERROR) {
print_error(dbg,"Cannot Get die address size for exprloc",
ares,ecerr);
} else {
get_string_from_locs(dbg,x,tempud,address_size,
offset_size,version, esbp);
}
}
}
/* Borrow the definition from pro_encode_nm.h */
/* Bytes needed to encode a number.
Not a tight bound, just a reasonable bound.
*/
#ifndef ENCODE_SPACE_NEEDED
#define ENCODE_SPACE_NEEDED (2*sizeof(Dwarf_Unsigned))
#endif /* ENCODE_SPACE_NEEDED */
/* Table indexed by the attribute value; only standard attributes
are included, ie. in the range [1..DW_AT_lo_user]; we waste a
little bit of space, but accessing the table is fast. */
typedef struct attr_encoding {
Dwarf_Unsigned entries; /* Attribute occurrences */
Dwarf_Unsigned formx; /* Space used by current encoding */
Dwarf_Unsigned leb128; /* Space used with LEB128 encoding */
} a_attr_encoding;
/* The other DW_FORM_datan are lower form values than data16,
so the following is safe for the unchanging static table. */
static int attributes_encoding_factor[DW_FORM_data16 + 1];
/* These must be reset for each object if we are processing
an archive! see print_attributes_encoding(). */
static a_attr_encoding *attributes_encoding_table = NULL;
static boolean attributes_encoding_do_init = TRUE;
/* Check the potential amount of space wasted by attributes values that can
be represented as an unsigned LEB128. Only attributes with forms:
DW_FORM_data1, DW_FORM_data2, DW_FORM_data4 and DW_FORM_data are checked
*/
static void
check_attributes_encoding(Dwarf_Half attr,Dwarf_Half theform,
Dwarf_Unsigned value)
{
if (attributes_encoding_do_init) {
/* Create table on first call */
attributes_encoding_table = (a_attr_encoding *)calloc(DW_AT_lo_user,
sizeof(a_attr_encoding));
/* We use only 5 slots in the table, for quick access */
attributes_encoding_factor[DW_FORM_data1] = 1; /* index 0x0b */
attributes_encoding_factor[DW_FORM_data2] = 2; /* index 0x05 */
attributes_encoding_factor[DW_FORM_data4] = 4; /* index 0x06 */
attributes_encoding_factor[DW_FORM_data8] = 8; /* index 0x07 */
attributes_encoding_factor[DW_FORM_data16] = 16;/* index 0x1e */
attributes_encoding_do_init = FALSE;
}
/* Regardless of the encoding form, count the checks. */
DWARF_CHECK_COUNT(attr_encoding_result,1);
/* For 'DW_AT_stmt_list', due to the way is generated, the value
can be unknown at compile time and only the assembler can decide
how to represent the offset; ignore this attribute. */
if (DW_AT_stmt_list == attr ||
DW_AT_macros == attr ||
DW_AT_GNU_macros == attr) {
return;
}
/* Only checks those attributes that have DW_FORM_dataX:
DW_FORM_data1, DW_FORM_data2, DW_FORM_data4 and DW_FORM_data8
DWARF5 adds DW_FORM_data16, but we ignore data16 here
as it makes no sense as a uleb. */
if (theform == DW_FORM_data1 || theform == DW_FORM_data2 ||
theform == DW_FORM_data4 || theform == DW_FORM_data8 ) {
int res = 0;
/* Size of the byte stream buffer that needs to be
memcpy-ed. */
int leb128_size = 0;
/* To encode the attribute value */
char encode_buffer[ENCODE_SPACE_NEEDED];
char small_buf[64]; /* Just a small buffer */
res = dwarf_encode_leb128(value,&leb128_size,
encode_buffer,sizeof(encode_buffer));
if (res == DW_DLV_OK) {
if (attributes_encoding_factor[theform] > leb128_size) {
int wasted_bytes = attributes_encoding_factor[theform]
- leb128_size;
snprintf(small_buf, sizeof(small_buf),
"%d wasted byte(s)",wasted_bytes);
DWARF_CHECK_ERROR2(attr_encoding_result,
get_AT_name(attr,pd_dwarf_names_print_on_error),
small_buf);
/* Add the optimized size to the specific
attribute, only if we are dealing with
a standard attribute. */
if (attr < DW_AT_lo_user) {
attributes_encoding_table[attr].entries += 1;
attributes_encoding_table[attr].formx +=
attributes_encoding_factor[theform];
attributes_encoding_table[attr].leb128 +=
leb128_size;
}
}
}
}
}
/* Print a detailed encoding usage per attribute */
void
print_attributes_encoding(Dwarf_Debug dbg)
{
if (attributes_encoding_table) {
boolean print_header = TRUE;
Dwarf_Unsigned total_entries = 0;
Dwarf_Unsigned total_bytes_formx = 0;
Dwarf_Unsigned total_bytes_leb128 = 0;
Dwarf_Unsigned entries = 0;
Dwarf_Unsigned bytes_formx = 0;
Dwarf_Unsigned bytes_leb128 = 0;
int index;
int count = 0;
float saved_rate = 0.0;
Dwarf_Error attr_error = 0;
for (index = 0; index < DW_AT_lo_user; ++index) {
if (attributes_encoding_table[index].leb128) {
if (print_header) {
printf("\n*** SPACE USED BY ATTRIBUTE ENCODINGS ***\n");
printf("Nro Attribute Name "
" Entries Data_x leb128 Rate\n");
print_header = FALSE;
}
entries = attributes_encoding_table[index].entries;
bytes_formx = attributes_encoding_table[index].formx;
bytes_leb128 = attributes_encoding_table[index].leb128;
total_entries += entries;
total_bytes_formx += bytes_formx;
total_bytes_leb128 += bytes_leb128;
saved_rate = bytes_leb128 * 100 / bytes_formx;
printf("%3d %-25s "
"%10" /*DW_PR_XZEROS*/ DW_PR_DUu " " /* Entries */
"%10" /*DW_PR_XZEROS*/ DW_PR_DUu " " /* FORMx */
"%10" /*DW_PR_XZEROS*/ DW_PR_DUu " " /* LEB128 */
"%3.0f%%"
"\n",
++count,
get_AT_name(index,pd_dwarf_names_print_on_error),
entries,
bytes_formx,
bytes_leb128,
saved_rate);
}
}
if (!print_header) {
/* At least we have an entry, print summary and percentage */
Dwarf_Addr lower = 0;
Dwarf_Unsigned size = 0;
int infoerr = 0;
saved_rate = total_bytes_leb128 * 100 / total_bytes_formx;
printf("** Summary ** "
"%10" /*DW_PR_XZEROS*/ DW_PR_DUu " " /* Entries */
"%10" /*DW_PR_XZEROS*/ DW_PR_DUu " " /* FORMx */
"%10" /*DW_PR_XZEROS*/ DW_PR_DUu " " /* LEB128 */
"%3.0f%%"
"\n",
total_entries,
total_bytes_formx,
total_bytes_leb128,
saved_rate);
/* Get .debug_info size (Very unlikely to have an error here). */
infoerr = dwarf_get_section_info_by_name(dbg,".debug_info",&lower,
&size,&attr_error);
if (infoerr == DW_DLV_ERROR) {
print_error(dbg, "get_section_info_by_name",
infoerr,attr_error);
}
saved_rate = (total_bytes_formx - total_bytes_leb128) * 100 / size;
if (saved_rate > 0) {
printf("\n** .debug_info size can be reduced by %.0f%% **\n",
saved_rate);
}
}
free(attributes_encoding_table);
attributes_encoding_table = 0;
attributes_encoding_do_init = TRUE;
}
}
/* Fill buffer with attribute value.
We pass in tag so we can try to do the right thing with
broken compiler DW_TAG_enumerator
'cnt' is signed for historical reasons (a mistake
in an interface), but the value is never negative.
We append to esbp's buffer.
*/
void
get_attr_value(Dwarf_Debug dbg, Dwarf_Half tag,
Dwarf_Die die,
Dwarf_Off dieprint_cu_goffset,
Dwarf_Attribute attrib,
char **srcfiles, Dwarf_Signed cnt, struct esb_s *esbp,
int show_form,
int local_verbose)
{
Dwarf_Half theform = 0;
char * temps = 0;
Dwarf_Block *tempb = 0;
Dwarf_Signed tempsd = 0;
Dwarf_Unsigned tempud = 0;
Dwarf_Off off = 0;
Dwarf_Die die_for_check = 0;
Dwarf_Half tag_for_check = 0;
Dwarf_Bool tempbool = 0;
Dwarf_Addr addr = 0;
int fres = 0;
int bres = 0;
int wres = 0;
int dres = 0;
Dwarf_Half direct_form = 0;
char small_buf[COMPILE_UNIT_NAME_LEN]; /* Size to hold a filename */
Dwarf_Bool is_info = TRUE;
Dwarf_Error err = 0;
is_info = dwarf_get_die_infotypes_flag(die);
/* Dwarf_whatform gets the real form, DW_FORM_indir is
never returned: instead the real form following
DW_FORM_indir is returned. */
fres = dwarf_whatform(attrib, &theform, &err);
/* Depending on the form and the attribute, process the form. */
if (fres == DW_DLV_ERROR) {
print_error(dbg, "dwarf_whatform cannot Find Attr Form", fres,
err);
} else if (fres == DW_DLV_NO_ENTRY) {
return;
}
/* dwarf_whatform_direct gets the 'direct' form, so if
the form is DW_FORM_indir that is what is returned. */
dwarf_whatform_direct(attrib, &direct_form, &err);
/* Ignore errors in dwarf_whatform_direct() */
switch (theform) {
case DW_FORM_GNU_addr_index:
case DW_FORM_addrx:
case DW_FORM_addr:
bres = dwarf_formaddr(attrib, &addr, &err);
if (bres == DW_DLV_OK) {
if (theform == DW_FORM_GNU_addr_index ||
theform == DW_FORM_addrx) {
Dwarf_Unsigned index = 0;
int res = dwarf_get_debug_addr_index(attrib,&index,&err);
if(res != DW_DLV_OK) {
print_error(dbg, "addr missing index ?!", res, err);
}
bracket_hex("(addr_index: ",index, ")",esbp);
}
bracket_hex("",addr,"",esbp);
} else if (bres == DW_DLV_ERROR) {
if (DW_DLE_MISSING_NEEDED_DEBUG_ADDR_SECTION ==
dwarf_errno(err)) {
Dwarf_Unsigned index = 0;
int res = dwarf_get_debug_addr_index(attrib,&index,&err);
if(res != DW_DLV_OK) {
print_error(dbg, "addr missing index ?!", bres, err);
}
addr = 0;
bracket_hex("(addr_index: ",index,
")<no .debug_addr section>",esbp);
/* This is normal in a .dwo file. The .debug_addr
is in a .o and in the final executable. */
} else {
print_error(dbg, "addr form with no addr?!", bres, err);
}
} else {
print_error(dbg, "addr is a DW_DLV_NO_ENTRY? Impossible.",
bres, err);
}
break;
case DW_FORM_ref_addr:
{
Dwarf_Half attr = 0;
/* DW_FORM_ref_addr is not accessed thru formref: ** it is an
address (global section offset) in ** the .debug_info
section. */
bres = dwarf_global_formref(attrib, &off, &err);
if (bres == DW_DLV_OK) {
bracket_hex("<GOFF=",off, ">",esbp);
} else {
print_error(dbg,
"DW_FORM_ref_addr form with no reference?!",
bres, err);
}
wres = dwarf_whatattr(attrib, &attr, &err);
if (wres == DW_DLV_ERROR) {
} else if (wres == DW_DLV_NO_ENTRY) {
} else {
if (attr == DW_AT_sibling) {
/* The value had better be inside the current CU
else there is a nasty error here, as a sibling
has to be in the same CU, it seems. */
/* The target offset (off) had better be
following the die's global offset else
we have a serious botch. this FORM
defines the value as a .debug_info
global offset. */
Dwarf_Off cuoff = 0;
Dwarf_Off culen = 0;
Dwarf_Off die_overall_offset = 0;
int res = 0;
int ores = dwarf_dieoffset(die, &die_overall_offset, &err);
if (ores != DW_DLV_OK) {
print_error(dbg, "dwarf_dieoffset", ores, err);
}
SET_DIE_STACK_SIBLING(off);
if (die_overall_offset >= off) {
snprintf(small_buf,sizeof(small_buf),
"ERROR: Sibling DW_FORM_ref_offset 0x%"
DW_PR_XZEROS DW_PR_DUx
" points %s die Global offset "
"0x%" DW_PR_XZEROS DW_PR_DUx,
off,(die_overall_offset == off)?"at":"before",
die_overall_offset);
print_error(dbg,small_buf,DW_DLV_OK,0);
}
res = dwarf_die_CU_offset_range(die,&cuoff,
&culen,&err);
DWARF_CHECK_COUNT(tag_tree_result,1);
if (res != DW_DLV_OK) {
} else {
Dwarf_Off cuend = cuoff+culen;
if (off < cuoff || off >= cuend) {
DWARF_CHECK_ERROR(tag_tree_result,
"DW_AT_sibling DW_FORM_ref_addr offset points "
"outside of current CU");
}
}
}
}
}
break;
case DW_FORM_ref1:
case DW_FORM_ref2:
case DW_FORM_ref4:
case DW_FORM_ref8:
case DW_FORM_ref_udata:
{
int refres = 0;
Dwarf_Half attr = 0;
Dwarf_Off goff = 0; /* Global offset */
Dwarf_Error referr = 0;
/* CU-relative offset returned. */
refres = dwarf_formref(attrib, &off, &referr);
if (refres != DW_DLV_OK) {
/* Report incorrect offset */
snprintf(small_buf,sizeof(small_buf),
"%s, offset=<0x%" DW_PR_XZEROS DW_PR_DUx
">","reference form with no valid local ref?!",off);
print_error(dbg, small_buf, refres, referr);
}
refres = dwarf_whatattr(attrib, &attr, &referr);
if (refres != DW_DLV_OK) {
snprintf(small_buf,sizeof(small_buf),
"Form %d, has no attribute value?!" ,theform);
print_error(dbg, small_buf, refres, referr);
}
/* Convert the local offset 'off' into a global section
offset 'goff'. */
refres = dwarf_convert_to_global_offset(attrib,
off, &goff, &referr);
if (refres != DW_DLV_OK) {
/* Report incorrect offset */
snprintf(small_buf,sizeof(small_buf),
"%s, GOFF=<0x%" DW_PR_XZEROS DW_PR_DUx
">","invalid offset",goff);
print_error(dbg, small_buf, refres, referr);
}
if (attr == DW_AT_sibling) {
/* The value had better be inside the current CU
else there is a nasty error here, as a sibling
has to be in the same CU, it seems. */
/* The target offset (off) had better be
following the die's global offset else
we have a serious botch. this FORM
defines the value as a .debug_info
global offset. */
Dwarf_Off die_overall_offset = 0;
int ores = dwarf_dieoffset(die, &die_overall_offset, &referr);
if (ores != DW_DLV_OK) {
print_error(dbg, "dwarf_dieoffset", ores, referr);
}
SET_DIE_STACK_SIBLING(goff);
if (die_overall_offset >= goff) {
snprintf(small_buf,sizeof(small_buf),
"ERROR: Sibling offset 0x%" DW_PR_XZEROS DW_PR_DUx
" points %s its own die GOFF="
"0x%" DW_PR_XZEROS DW_PR_DUx,
goff,
(die_overall_offset == goff)?"at":"before",
die_overall_offset);
print_error(dbg,small_buf,DW_DLV_OK,0);
}
}
/* Do references inside <> to distinguish them ** from
constants. In dense form this results in <<>>. Ugly for
dense form, but better than ambiguous. davea 9/94 */
if (glflags.gf_show_global_offsets) {
bracket_hex("<",off,"",esbp);
bracket_hex(" GOFF=",goff,">",esbp);
} else {
bracket_hex("<",off,">",esbp);
}
if (glflags.gf_check_type_offset) {
if (attr == DW_AT_type && form_refers_local_info(theform)) {
dres = dwarf_offdie_b(dbg, goff,
is_info,
&die_for_check, &referr);
if (dres != DW_DLV_OK) {
snprintf(small_buf,sizeof(small_buf),
"DW_AT_type offset does not point to a DIE "
"for global offset 0x%" DW_PR_XZEROS DW_PR_DUx
" cu off 0x%" DW_PR_XZEROS DW_PR_DUx
" local offset 0x%" DW_PR_XZEROS DW_PR_DUx
" tag 0x%x",
goff,dieprint_cu_goffset,off,tag);
DWARF_CHECK_ERROR(type_offset_result,small_buf);
} else {
int tres2 =
dwarf_tag(die_for_check, &tag_for_check, &err);
if (tres2 == DW_DLV_OK) {
switch (tag_for_check) {
case DW_TAG_array_type:
case DW_TAG_class_type:
case DW_TAG_enumeration_type:
case DW_TAG_pointer_type:
case DW_TAG_reference_type:
case DW_TAG_rvalue_reference_type:
case DW_TAG_restrict_type:
case DW_TAG_string_type:
case DW_TAG_structure_type:
case DW_TAG_subroutine_type:
case DW_TAG_typedef:
case DW_TAG_union_type:
case DW_TAG_ptr_to_member_type:
case DW_TAG_set_type:
case DW_TAG_subrange_type:
case DW_TAG_base_type:
case DW_TAG_const_type:
case DW_TAG_file_type:
case DW_TAG_packed_type:
case DW_TAG_thrown_type:
case DW_TAG_volatile_type:
case DW_TAG_template_type_parameter:
case DW_TAG_template_value_parameter:
case DW_TAG_unspecified_type:
/* Template alias */
case DW_TAG_template_alias:
/* OK */
break;
default:
{
snprintf(small_buf,sizeof(small_buf),
"DW_AT_type offset "
"0x%" DW_PR_XZEROS DW_PR_DUx
" does not point to Type"
" info we got tag 0x%x %s",
(Dwarf_Unsigned)goff,
tag_for_check,
get_TAG_name(tag_for_check,
pd_dwarf_names_print_on_error));
DWARF_CHECK_ERROR(type_offset_result,small_buf);
}
break;
}
dwarf_dealloc(dbg, die_for_check, DW_DLA_DIE);
die_for_check = 0;
} else {
DWARF_CHECK_ERROR(type_offset_result,
"DW_AT_type offset does not exist");
}
}
}
}
}
break;
case DW_FORM_block:
case DW_FORM_block1:
case DW_FORM_block2:
case DW_FORM_block4:
fres = dwarf_formblock(attrib, &tempb, &err);
if (fres == DW_DLV_OK) {
unsigned u = 0;
for (u = 0; u < tempb->bl_len; u++) {
snprintf(small_buf, sizeof(small_buf), "%02x",
*(u + (unsigned char *) tempb->bl_data));
esb_append(esbp, small_buf);
}
dwarf_dealloc(dbg, tempb, DW_DLA_BLOCK);
tempb = 0;
} else {
print_error(dbg, "DW_FORM_blockn cannot get block\n", fres,
err);
}
break;
case DW_FORM_data1:
case DW_FORM_data2:
case DW_FORM_data4:
case DW_FORM_data8:
case DW_FORM_data16:
{
Dwarf_Half attr = 0;
fres = dwarf_whatattr(attrib, &attr, &err);
if (fres == DW_DLV_ERROR) {
print_error(dbg, "FORM_datan cannot get attr", fres, err);
} else if (fres == DW_DLV_NO_ENTRY) {
print_error(dbg, "FORM_datan cannot get attr", fres, err);
} else {
switch (attr) {
case DW_AT_ordering:
case DW_AT_byte_size:
case DW_AT_bit_offset:
case DW_AT_bit_size:
case DW_AT_inline:
case DW_AT_language:
case DW_AT_visibility:
case DW_AT_virtuality:
case DW_AT_accessibility:
case DW_AT_address_class:
case DW_AT_calling_convention:
case DW_AT_discr_list: /* DWARF2 */
case DW_AT_encoding:
case DW_AT_identifier_case:
case DW_AT_MIPS_loop_unroll_factor:
case DW_AT_MIPS_software_pipeline_depth:
case DW_AT_decl_column:
case DW_AT_decl_file:
case DW_AT_decl_line:
case DW_AT_call_column:
case DW_AT_call_file:
case DW_AT_call_line:
case DW_AT_start_scope:
case DW_AT_byte_stride:
case DW_AT_bit_stride:
case DW_AT_count:
case DW_AT_stmt_list:
case DW_AT_MIPS_fde:
{ int show_form_here = 0;
wres = get_small_encoding_integer_and_name(dbg,
attrib,
&tempud,
/* attrname */ (const char *) NULL,
/* err_string */ ( struct esb_s *) NULL,
(encoding_type_func) 0,
&err,show_form_here);
if (wres == DW_DLV_OK) {
Dwarf_Bool hex_format = TRUE;
formx_unsigned(tempud,esbp,hex_format);
/* Check attribute encoding */
if (glflags.gf_check_attr_encoding) {
check_attributes_encoding(attr,theform,tempud);
}
if (attr == DW_AT_decl_file || attr == DW_AT_call_file) {
if (srcfiles && tempud > 0 &&
/* ASSERT: cnt >= 0 */
tempud <= (Dwarf_Unsigned)cnt) {
/* added by user request */
/* srcfiles is indexed starting at 0, but
DW_AT_decl_file defines that 0 means no
file, so tempud 1 means the 0th entry in
srcfiles, thus tempud-1 is the correct
index into srcfiles. */
char *fname = srcfiles[tempud - 1];
esb_append(esbp, " ");
esb_append(esbp, fname);
}
/* Validate integrity of files
referenced in .debug_line */
if (glflags.gf_check_decl_file) {
DWARF_CHECK_COUNT(decl_file_result,1);
/* Zero is always a legal index, it means
no source name provided. */
if (tempud != 0 &&
tempud > ((Dwarf_Unsigned)cnt)) {
if (!srcfiles) {
snprintf(small_buf,sizeof(small_buf),
"There is a file number=%" DW_PR_DUu
" but no source files "
" are known.",tempud);
} else {
snprintf(small_buf, sizeof(small_buf),
"Does not point to valid file info "
" filenum=%" DW_PR_DUu
" filecount=%" DW_PR_DUu ".",
tempud,cnt);
}
DWARF_CHECK_ERROR2(decl_file_result,
get_AT_name(attr,
pd_dwarf_names_print_on_error),
small_buf);
}
}
}
} else {
print_error(dbg, "Cannot get encoding attribute ..",
wres, err);
}
}
break;
case DW_AT_const_value:
/* Do not use hexadecimal format */
wres = formxdata_print_value(dbg,die,attrib,
theform,esbp, &err, FALSE);
if (wres == DW_DLV_OK){
/* String appended already. */
} else if (wres == DW_DLV_NO_ENTRY) {
/* nothing? */
} else {
print_error(dbg,"Cannot get DW_AT_const_value ",wres,err);
}
break;
case DW_AT_GNU_dwo_id:
case DW_AT_dwo_id:
{
Dwarf_Sig8 v;
memset(&v,0,sizeof(v));
wres = dwarf_formsig8_const(attrib,&v,&err);
if (wres == DW_DLV_OK){
struct esb_s t;
esb_constructor(&t);
format_sig8_string(&v,&t);
esb_append(esbp,esb_get_string(&t));
esb_destructor(&t);
} else if (wres == DW_DLV_NO_ENTRY) {
/* nothing? */
esb_append(esbp,"Impossible: no entry for formsig8 dwo_id");
} else {
print_error(dbg,"Cannot get DW_AT_const_value ",wres,err);
}
}
break;
case DW_AT_upper_bound:
case DW_AT_lower_bound:
default: {
Dwarf_Bool chex = FALSE;
Dwarf_Die tdie = die;
if(DW_AT_ranges == attr) {
/* In this case do not look for data
type for unsigned/signed.
and do use HEX. */
chex = TRUE;
tdie = NULL;
}
/* Do not use hexadecimal format except for
DW_AT_ranges. */
wres = formxdata_print_value(dbg,
tdie,attrib,
theform,esbp, &err, chex);
if (wres == DW_DLV_OK) {
/* String appended already. */
} else if (wres == DW_DLV_NO_ENTRY) {
/* nothing? */
} else {
print_error(dbg, "Cannot get form data..", wres,
err);
}
}
break;
}
}
if (glflags.gf_cu_name_flag) {
if (attr == DW_AT_MIPS_fde) {
if (fde_offset_for_cu_low == DW_DLV_BADOFFSET) {
fde_offset_for_cu_low
= fde_offset_for_cu_high = tempud;
} else if (tempud < fde_offset_for_cu_low) {
fde_offset_for_cu_low = tempud;
} else if (tempud > fde_offset_for_cu_high) {
fde_offset_for_cu_high = tempud;
}
}
}
}
break;
case DW_FORM_sdata:
wres = dwarf_formsdata(attrib, &tempsd, &err);
if (wres == DW_DLV_OK) {
Dwarf_Bool hxform=TRUE;
tempud = tempsd;
formx_unsigned_and_signed_if_neg(tempud,tempsd,
" (",hxform,esbp);
} else if (wres == DW_DLV_NO_ENTRY) {
/* nothing? */
} else {
print_error(dbg, "Cannot get formsdata..", wres, err);
}
break;
case DW_FORM_udata:
wres = dwarf_formudata(attrib, &tempud, &err);
if (wres == DW_DLV_OK) {
Dwarf_Bool hex_format = TRUE;
formx_unsigned(tempud,esbp,hex_format);
} else if (wres == DW_DLV_NO_ENTRY) {
/* nothing? */
} else {
print_error(dbg, "Cannot get formudata....", wres, err);
}
break;
case DW_FORM_string:
case DW_FORM_strp:
case DW_FORM_strx:
case DW_FORM_strp_sup: /* An offset to alternate file: tied file */
case DW_FORM_GNU_strp_alt: /* An offset to alternate file: tied file */
case DW_FORM_GNU_str_index: {
int sres = dwarf_formstring(attrib, &temps, &err);
if (sres == DW_DLV_OK) {
if (theform == DW_FORM_strx ||
theform == DW_FORM_GNU_str_index) {
struct esb_s saver;
Dwarf_Unsigned index = 0;
esb_constructor(&saver);
sres = dwarf_get_debug_str_index(attrib,&index,&err);
esb_append(&saver,temps);
if(sres == DW_DLV_OK) {
bracket_hex("(indexed string: ",index,")",esbp);
} else {
esb_append(esbp,"(indexed string:no string provided?)");
}
esb_append(esbp, esb_get_string(&saver));
esb_destructor(&saver);
} else {
esb_append(esbp,temps);
}
} else if (sres == DW_DLV_NO_ENTRY) {
if (theform == DW_FORM_strx ||
theform == DW_FORM_GNU_str_index) {
esb_append(esbp, "(indexed string,no string provided?)");
} else {
esb_append(esbp, "<no string provided?>");
}
} else {
if (theform == DW_FORM_strx ||
theform == DW_FORM_GNU_str_index) {
print_error(dbg, "Cannot get an indexed string....",
sres, err);
} else {
print_error(dbg, "Cannot get a formstr (or a formstrp)....",
sres, err);
}
}
}
break;
case DW_FORM_flag:
wres = dwarf_formflag(attrib, &tempbool, &err);
if (wres == DW_DLV_OK) {
if (tempbool) {
snprintf(small_buf, sizeof(small_buf), "yes(%d)",
tempbool);
esb_append(esbp, small_buf);
} else {
snprintf(small_buf, sizeof(small_buf), "no");
esb_append(esbp, small_buf);
}
} else if (wres == DW_DLV_NO_ENTRY) {
/* nothing? */
} else {
print_error(dbg, "Cannot get formflag/p....", wres, err);
}
break;
case DW_FORM_indirect:
/* We should not ever get here, since the true form was
determined and direct_form has the DW_FORM_indirect if it is
used here in this attr. */
esb_append(esbp, get_FORM_name(theform,
pd_dwarf_names_print_on_error));
break;
case DW_FORM_exprloc: { /* DWARF4 */
int showhextoo = 1;
print_exprloc_content(dbg,die,attrib,showhextoo,esbp);
}
break;
case DW_FORM_sec_offset: { /* DWARF4 */
char* emptyattrname = 0;
int show_form_here = 0;
wres = get_small_encoding_integer_and_name(dbg,
attrib,
&tempud,
emptyattrname,
/* err_string */ NULL,
(encoding_type_func) 0,
&err,show_form_here);
if (wres == DW_DLV_NO_ENTRY) {
/* Show nothing? */
} else if (wres == DW_DLV_ERROR) {
print_error(dbg,
"Cannot get a DW_FORM_sec_offset....",
wres, err);
} else {
bracket_hex("",tempud,"",esbp);
}
}
break;
case DW_FORM_flag_present: /* DWARF4 */
esb_append(esbp,"yes(1)");
break;
case DW_FORM_ref_sig8: { /* DWARF4 */
Dwarf_Sig8 sig8data;
wres = dwarf_formsig8(attrib,&sig8data,&err);
if (wres != DW_DLV_OK) {
/* Show nothing? */
print_error(dbg,
"Cannot get a DW_FORM_ref_sig8 ....",
wres, err);
} else {
struct esb_s sig8str;
esb_constructor(&sig8str);
format_sig8_string(&sig8data,&sig8str);
esb_append(esbp,esb_get_string(&sig8str));
esb_destructor(&sig8str);
if (!show_form) {
esb_append(esbp," <type signature>");
}
}
}
break;
case DW_FORM_GNU_ref_alt: {
bres = dwarf_global_formref(attrib, &off, &err);
if (bres == DW_DLV_OK) {
bracket_hex("",off,"",esbp);
} else {
print_error(dbg,
"DW_FORM_GNU_ref_alt form with no reference?!",
bres, err);
}
}
break;
default:
print_error(dbg, "dwarf_whatform unexpected value", DW_DLV_OK,
err);
}
show_form_itself(show_form,local_verbose,theform, direct_form,esbp);
}
void
format_sig8_string(Dwarf_Sig8*data, struct esb_s *out)
{
unsigned i = 0;
char small_buf[40];
esb_append(out,"0x");
for (; i < sizeof(data->signature); ++i) {
#if 0
/* The signature is logically one glob of
8 bytes, not two, so show as one glob
using 16 ascii hex digits */
if (i == 4) {
esb_append(out," 0x");
}
#endif
snprintf(small_buf,sizeof(small_buf), "%02x",
(unsigned char)(data->signature[i]));
esb_append(out,small_buf);
}
}
/* This leaks Dwarf_Error in case of error. FIXME */
static int
get_form_values(Dwarf_Debug dbg,Dwarf_Attribute attrib,
Dwarf_Half * theform, Dwarf_Half * directform)
{
Dwarf_Error verr = 0;
int res = 0;
res = dwarf_whatform(attrib, theform, &verr);
DROP_ERROR_INSTANCE(dbg,res,verr);
res = dwarf_whatform_direct(attrib, directform, &verr);
DROP_ERROR_INSTANCE(dbg,res,verr);
return res;
}
static void
show_form_itself(int local_show_form,
int local_verbose,
int theform,
int directform, struct esb_s *esbp)
{
char small_buf[100];
if (local_show_form
&& directform && directform == DW_FORM_indirect) {
char *form_indir = " (used DW_FORM_indirect";
char *form_indir2 = ") ";
esb_append(esbp, form_indir);
if (local_verbose) {
esb_append(esbp, get_form_number_as_string(DW_FORM_indirect,
small_buf,sizeof(small_buf)));
}
esb_append(esbp, form_indir2);
}
if (local_show_form) {
esb_append(esbp," <form ");
esb_append(esbp,get_FORM_name(theform,
pd_dwarf_names_print_on_error));
if (local_verbose) {
esb_append(esbp, get_form_number_as_string(theform,
small_buf, sizeof(small_buf)));
}
esb_append(esbp,">");
}
}
#include "tmp-ta-table.c"
#include "tmp-ta-ext-table.c"
static int
legal_tag_attr_combination(Dwarf_Half tag, Dwarf_Half attr)
{
if (tag <= 0) {
return FALSE;
}
if (tag < ATTR_TREE_ROW_COUNT) {
int index = attr / BITS_PER_WORD;
if (index < ATTR_TREE_COLUMN_COUNT) {
unsigned bitflag = ((unsigned)1) << (attr % BITS_PER_WORD);
int known = ((tag_attr_combination_table[tag][index]
& bitflag) > 0 ? TRUE : FALSE);
if (known) {
#ifdef HAVE_USAGE_TAG_ATTR
/* Record usage of pair (tag,attr) */
if ( glflags.gf_print_usage_tag_attr) {
Usage_Tag_Attr *usage_ptr = usage_tag_attr[tag];
while (usage_ptr->attr) {
if (attr == usage_ptr->attr) {
++usage_ptr->count;
break;
}
++usage_ptr;
}
}
#endif /* HAVE_USAGE_TAG_ATTR */
return TRUE;
}
}
}
/* DW_AT_MIPS_fde used to return TRUE as that was
convenient for SGI/MIPS users. */
if (!glflags.gf_suppress_check_extensions_tables) {
int r = 0;
for (; r < ATTR_TREE_EXT_ROW_COUNT; ++r ) {
int c = 1;
if (tag != tag_attr_combination_ext_table[r][0]) {
continue;
}
for (; c < ATTR_TREE_EXT_COLUMN_COUNT ; ++c) {
if (tag_attr_combination_ext_table[r][c] == attr) {
return TRUE;
}
}
}
}
return (FALSE);
}
#include "tmp-tt-table.c"
#include "tmp-tt-ext-table.c"
/* Look only at valid table entries
The check here must match the building-logic in
tag_tree.c
And must match the tags defined in dwarf.h
The tag_tree_combination_table is a table of bit flags. */
static int
legal_tag_tree_combination(Dwarf_Half tag_parent, Dwarf_Half tag_child)
{
if (tag_parent <= 0) {
return FALSE;
}
if (tag_parent < TAG_TREE_ROW_COUNT) {
int index = tag_child / BITS_PER_WORD;
if (index < TAG_TREE_COLUMN_COUNT) {
unsigned bitflag = ((unsigned)1) << (tag_child % BITS_PER_WORD);
int known = ((tag_tree_combination_table[tag_parent]
[index] & bitflag) > 0 ? TRUE : FALSE);
if (known) {
#ifdef HAVE_USAGE_TAG_ATTR
/* Record usage of pair (tag_parent,tag_child) */
if ( glflags.gf_print_usage_tag_attr) {
Usage_Tag_Tree *usage_ptr = usage_tag_tree[tag_parent];
while (usage_ptr->tag) {
if (tag_child == usage_ptr->tag) {
++usage_ptr->count;
break;
}
++usage_ptr;
}
}
#endif /* HAVE_USAGE_TAG_ATTR */
return TRUE;
}
}
}
if (!glflags.gf_suppress_check_extensions_tables) {
int r = 0;
for (; r < TAG_TREE_EXT_ROW_COUNT; ++r ) {
int c = 1;
if (tag_parent != tag_tree_combination_ext_table[r][0]) {
continue;
}
for (; c < TAG_TREE_EXT_COLUMN_COUNT ; ++c) {
if (tag_tree_combination_ext_table[r][c] == tag_child) {
return TRUE;
}
}
}
}
return (FALSE);
}
/* Print a detailed tag and attributes usage */
void
print_tag_attributes_usage(UNUSEDARG Dwarf_Debug dbg)
{
#ifdef HAVE_USAGE_TAG_ATTR
/* Traverse the tag-tree table to print its usage and then use the
DW_TAG value as an index into the tag_attr table to print its
associated usage all together. */
boolean print_header = TRUE;
Rate_Tag_Tree *tag_rate;
Rate_Tag_Attr *atr_rate;
Usage_Tag_Tree *usage_tag_tree_ptr;
Usage_Tag_Attr *usage_tag_attr_ptr;
Dwarf_Unsigned total_tags = 0;
Dwarf_Unsigned total_atrs = 0;
Dwarf_Half total_found_tags = 0;
Dwarf_Half total_found_atrs = 0;
Dwarf_Half total_legal_tags = 0;
Dwarf_Half total_legal_atrs = 0;
float rate_1;
float rate_2;
int tag;
printf("\n*** TAGS AND ATTRIBUTES USAGE ***\n");
for (tag = 1; tag < DW_TAG_last; ++tag) {
/* Print usage of children TAGs */
if ( glflags.gf_print_usage_tag_attr_full || tag_usage[tag]) {
usage_tag_tree_ptr = usage_tag_tree[tag];
if (usage_tag_tree_ptr && print_header) {
total_tags += tag_usage[tag];
printf("%6d %s\n",
tag_usage[tag],
get_TAG_name(tag,pd_dwarf_names_print_on_error));
print_header = FALSE;
}
while (usage_tag_tree_ptr && usage_tag_tree_ptr->tag) {
if ( glflags.gf_print_usage_tag_attr_full || usage_tag_tree_ptr->count) {
total_tags += usage_tag_tree_ptr->count;
printf("%6s %6d %s\n",
" ",
usage_tag_tree_ptr->count,
get_TAG_name(usage_tag_tree_ptr->tag,
pd_dwarf_names_print_on_error));
/* Record the tag as found */
if (usage_tag_tree_ptr->count) {
++rate_tag_tree[tag].found;
}
}
++usage_tag_tree_ptr;
}
}
/* Print usage of attributes */
if ( glflags.gf_print_usage_tag_attr_full || tag_usage[tag]) {
usage_tag_attr_ptr = usage_tag_attr[tag];
if (usage_tag_attr_ptr && print_header) {
total_tags += tag_usage[tag];
printf("%6d %s\n",
tag_usage[tag],
get_TAG_name(tag,pd_dwarf_names_print_on_error));
print_header = FALSE;
}
while (usage_tag_attr_ptr && usage_tag_attr_ptr->attr) {
if ( glflags.gf_print_usage_tag_attr_full || usage_tag_attr_ptr->count) {
total_atrs += usage_tag_attr_ptr->count;
printf("%6s %6d %s\n",
" ",
usage_tag_attr_ptr->count,
get_AT_name(usage_tag_attr_ptr->attr,
pd_dwarf_names_print_on_error));
/* Record the attribute as found */
if (usage_tag_attr_ptr->count) {
++rate_tag_attr[tag].found;
}
}
++usage_tag_attr_ptr;
}
}
print_header = TRUE;
}
printf("** Summary **\n"
"Number of tags : %10" /*DW_PR_XZEROS*/ DW_PR_DUu "\n" /* TAGs */
"Number of attributes: %10" /*DW_PR_XZEROS*/ DW_PR_DUu "\n" /* ATRs */,
total_tags,
total_atrs);
total_legal_tags = 0;
total_found_tags = 0;
total_legal_atrs = 0;
total_found_atrs = 0;
/* Print percentage of TAGs covered */
printf("\n*** TAGS AND ATTRIBUTES USAGE RATE ***\n");
printf("%-32s %-16s %-16s\n"," ","Tags","Attributes");
printf("%-32s legal found rate legal found rate\n","TAG name");
for (tag = 1; tag < DW_TAG_last; ++tag) {
tag_rate = &rate_tag_tree[tag];
atr_rate = &rate_tag_attr[tag];
if ( glflags.gf_print_usage_tag_attr_full ||
tag_rate->found || atr_rate->found) {
rate_1 = tag_rate->legal ?
(float)((tag_rate->found * 100) / tag_rate->legal) : 0;
rate_2 = atr_rate->legal ?
(float)((atr_rate->found * 100) / atr_rate->legal) : 0;
/* Skip not defined DW_TAG values (See dwarf.h) */
if (usage_tag_tree[tag]) {
total_legal_tags += tag_rate->legal;
total_found_tags += tag_rate->found;
total_legal_atrs += atr_rate->legal;
total_found_atrs += atr_rate->found;
printf("%-32s %5d %5d %3.0f%% %5d %5d %3.0f%%\n",
get_TAG_name(tag,pd_dwarf_names_print_on_error),
tag_rate->legal,tag_rate->found,rate_1,
atr_rate->legal,atr_rate->found,rate_2);
}
}
}
/* Print a whole summary */
rate_1 = total_legal_tags ?
(float)((total_found_tags * 100) / total_legal_tags) : 0;
rate_2 = total_legal_atrs ?
(float)((total_found_atrs * 100) / total_legal_atrs) : 0;
printf("%-32s %5d %5d %3.0f%% %5d %5d %3.0f%%\n",
"** Summary **",
total_legal_tags,total_found_tags,rate_1,
total_legal_atrs,total_found_atrs,rate_2);
#endif /* HAVE_USAGE_TAG_ATTR */
}