diff options
author | Alan Modra <amodra@gmail.com> | 2023-04-03 22:21:59 +0930 |
---|---|---|
committer | Alan Modra <amodra@gmail.com> | 2023-04-04 12:47:56 +0930 |
commit | 19cacf672930cee20feaf1f3468e3d5ac3099ffd (patch) | |
tree | a85da4040a0d01240981754533db366f2ce172d5 /binutils/rdcoff.c | |
parent | 59bb724b3591f270be752fb252bc6e01f1aae559 (diff) | |
download | gdb-19cacf672930cee20feaf1f3468e3d5ac3099ffd.zip gdb-19cacf672930cee20feaf1f3468e3d5ac3099ffd.tar.gz gdb-19cacf672930cee20feaf1f3468e3d5ac3099ffd.tar.bz2 |
Use bfd_alloc memory for read_debugging_info storage
Trying to free malloc'd memory used by the stabs and coff debug info
parsers is complicated, and traversing the trees generated requires a
lot of code. It's better to bfd_alloc the memory which allows it all
to be freed without fuss when the bfd is closed. In the process of
doing this I reverted most of commit a6336913332.
Some of the stabs handling code grows arrays of pointers with realloc,
to deal with arbitrary numbers of fields, function args, etc. The
code still does that but copies over to bfd_alloc memory when
finished. The alternative is to parse twice, once to size, then again
to populate the arrays. I think that complication is unwarranted.
Note that there is a greater than zero chance this patch breaks
something, eg. that I missed an attempt to free obj_alloc memory.
Also it seems there are no tests in the binutils testsuite aimed at
exercising objdump --debugging.
* budbg.h (finish_stab, parse_stab): Update prototypes
* debug.c: Include bucomm.h.
(struct debug_handle): Add "abfd" field.
(debug_init): Add "abfd" param. bfd_alloc handle.
(debug_xalloc, debug_xzalloc): New functions. Use throughout
in place of xmalloc and memset.
(debug_start_source): Remove "name_used" param.
* debug.h (debug_init, debug_start_source): Update prototypes.
(debug_xalloc, debug_xzalloc): Declare.
* objcopy.c (copy_object): Don't free dhandle.
* objdump.c (dump_bfd): Likewise.
* rdcoff.c (coff_get_slot): Add dhandle arg. debug_xzalloc
memory in place of xcalloc. Update callers.
(parse_coff_struct_type): Don't leak on error return. Copy
fields over to debug_xalloc memory.
(parse_coff_enum_type): Copy names and vals over the
debug_xalloc memory.
* rddbg.c (read_debugging_info): Adjust debug_init call.
Don't free dhandle.
(read_section_stabs_debugging_info): Don't free shandle.
Adjust parse_stab call. Call finish_stab on error return.
(read_symbol_stabs_debugging_info): Similarly.
* stabs.c (savestring): Delete unnecessary forward declaration.
Add dhandle param. debug_xalloc memory. Update callers.
(start_stab): Delete unnecessary casts.
(finish_stab): Add "emit" param. Free file_types, so_string,
and stabs handle.
(parse_stab): Delete string_used param. Revert code dealing
with string_used. Copy so_string passed to debug_set_filename
and stored as main_filename to debug_xalloc memory. Similarly
for string passed to debug_start_source and push_bincl. Copy
args to debug_xalloc memory. Don't leak args.
(parse_stab_enum_type): Copy names and values to debug_xalloc
memory. Don't free name.
(parse_stab_struct_type): Don't free fields.
(parse_stab_baseclasses): Delete unnecessary cast.
(parse_stab_struct_fields): Return debug_xalloc fields.
(parse_stab_cpp_abbrev): Use debug_xalloc for _vb$ type name.
(parse_stab_one_struct_field): Don't free name.
(parse_stab_members): Copy variants and methods to
debug_xalloc memory. Don't free name or argtypes.
(parse_stab_argtypes): Use debug_xalloc memory for physname
and args.
(push_bincl): Add dhandle param. Use debug_xalloc memory.
(stab_record_variable): Use debug_xalloc memory.
(stab_emit_pending_vars): Don't free var list.
(stab_find_slot): Add dhandle param. Use debug_xzalloc
memory. Update all callers.
(stab_find_tagged_type): Don't free name. Use debug_xzalloc.
(stab_demangle_qualified): Don't free name.
(stab_demangle_template): Don't free s1.
(stab_demangle_args): Tidy pvarargs refs. Copy *pargs on
success to debug_xalloc memory, free on failure.
(stab_demangle_fund_type): Don't free name.
(stab_demangle_v3_arglist): Copy args to debug_xalloc memory.
Don't free dt.
Diffstat (limited to 'binutils/rdcoff.c')
-rw-r--r-- | binutils/rdcoff.c | 37 |
1 files changed, 25 insertions, 12 deletions
diff --git a/binutils/rdcoff.c b/binutils/rdcoff.c index 95b3c54..17e89e8 100644 --- a/binutils/rdcoff.c +++ b/binutils/rdcoff.c @@ -84,7 +84,7 @@ static debug_type parse_coff_enum_type /* Return the slot for a type. */ static debug_type * -coff_get_slot (struct coff_types **types, long indx) +coff_get_slot (void *dhandle, struct coff_types **types, long indx) { unsigned int base_index; @@ -96,7 +96,7 @@ coff_get_slot (struct coff_types **types, long indx) if (*types == NULL || (*types)->base_index != base_index) { - struct coff_types *n = xcalloc (1, sizeof (*n)); + struct coff_types *n = debug_xzalloc (dhandle, sizeof (*n)); n->next = *types; n->base_index = base_index; *types = n; @@ -182,7 +182,7 @@ parse_coff_type (bfd *abfd, struct coff_symbols *symbols, /* This is a reference to an existing type. FIXME: gdb checks that the class is not C_STRTAG, nor C_UNTAG, nor C_ENTAG. */ - slot = coff_get_slot (types, pauxent->x_sym.x_tagndx.u32); + slot = coff_get_slot (dhandle, types, pauxent->x_sym.x_tagndx.u32); if (*slot != DEBUG_TYPE_NULL) return *slot; else @@ -309,7 +309,7 @@ parse_coff_base_type (bfd *abfd, struct coff_symbols *symbols, if (name != NULL) ret = debug_name_type (dhandle, name, ret); - debug_type *slot = coff_get_slot (types, coff_symno); + debug_type *slot = coff_get_slot (dhandle, types, coff_symno); *slot = ret; return ret; @@ -324,7 +324,7 @@ parse_coff_struct_type (bfd *abfd, struct coff_symbols *symbols, { long symend; int alloc; - debug_field *fields; + debug_field *fields, *xfields; int count; bool done; @@ -403,7 +403,10 @@ parse_coff_struct_type (bfd *abfd, struct coff_symbols *symbols, f = debug_make_field (dhandle, bfd_asymbol_name (sym), ftype, bitpos, bitsize, DEBUG_VISIBILITY_PUBLIC); if (f == DEBUG_FIELD_NULL) - return DEBUG_TYPE_NULL; + { + free (fields); + return DEBUG_TYPE_NULL; + } if (count + 1 >= alloc) { @@ -418,10 +421,13 @@ parse_coff_struct_type (bfd *abfd, struct coff_symbols *symbols, } fields[count] = DEBUG_FIELD_NULL; + xfields = debug_xalloc (dhandle, (count + 1) * sizeof (*fields)); + memcpy (xfields, fields, (count + 1) * sizeof (*fields)); + free (fields); return debug_make_struct_type (dhandle, ntype == T_STRUCT, pauxent->x_sym.x_misc.x_lnsz.x_size, - fields); + xfields); } /* Parse an enum type. */ @@ -433,8 +439,8 @@ parse_coff_enum_type (bfd *abfd, struct coff_symbols *symbols, { long symend; int alloc; - const char **names; - bfd_signed_vma *vals; + const char **names, **xnames; + bfd_signed_vma *vals, *xvals; int count; bool done; @@ -491,8 +497,15 @@ parse_coff_enum_type (bfd *abfd, struct coff_symbols *symbols, } names[count] = NULL; - - return debug_make_enum_type (dhandle, names, vals); + vals[count] = 0; + xnames = debug_xalloc (dhandle, (count + 1) * sizeof (*names)); + memcpy (xnames, names, (count + 1) * sizeof (*names)); + free (names); + xvals = debug_xalloc (dhandle, (count + 1) * sizeof (*vals)); + memcpy (xvals, vals, (count + 1) * sizeof (*vals)); + free (vals); + + return debug_make_enum_type (dhandle, xnames, xvals); } /* Handle a single COFF symbol. */ @@ -571,7 +584,7 @@ parse_coff_symbol (bfd *abfd ATTRIBUTE_UNUSED, struct coff_types **types, /* Store the named type into the slot, so that references get the name. */ - slot = coff_get_slot (types, coff_symno); + slot = coff_get_slot (dhandle, types, coff_symno); *slot = type; } break; |