|  | /* Coff file dumper. | 
|  | Copyright (C) 1994-2025 Free Software Foundation, Inc. | 
|  |  | 
|  | This file is part of GNU Binutils. | 
|  |  | 
|  | This program is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 3 of the License, or (at | 
|  | your option) any later version. | 
|  |  | 
|  | This program is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with this program; if not, write to the Free Software | 
|  | Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, | 
|  | MA 02110-1301, USA.  */ | 
|  |  | 
|  |  | 
|  | /* Written by Steve Chamberlain <sac@cygnus.com> | 
|  |  | 
|  | This module reads a type tree generated by coffgrok and prints | 
|  | it out so we can test the grokker.  */ | 
|  |  | 
|  | #include "sysdep.h" | 
|  | #include "bfd.h" | 
|  | #include <stdint.h> | 
|  | #include "libiberty.h" | 
|  | #include "bucomm.h" | 
|  |  | 
|  | #include "coffgrok.h" | 
|  | #include "getopt.h" | 
|  |  | 
|  | static int atnl; | 
|  |  | 
|  | static void tab (int); | 
|  | static void nl (void); | 
|  | static void dump_coff_lines (struct coff_line *); | 
|  | static void dump_coff_type (struct coff_type *); | 
|  | static void dump_coff_where (struct coff_where *); | 
|  | static void dump_coff_visible (struct coff_visible *); | 
|  | static void dump_coff_scope (struct coff_scope *); | 
|  | static void dump_coff_sfile (struct coff_sfile *); | 
|  | static void dump_coff_section (struct coff_section *); | 
|  | static void show_usage (FILE *, int); | 
|  | extern int main (int, char **); | 
|  |  | 
|  | static void | 
|  | tab (int x) | 
|  | { | 
|  | static int indent; | 
|  | int i; | 
|  |  | 
|  | if (atnl) | 
|  | { | 
|  | if (x < 0) | 
|  | { | 
|  | printf (")"); | 
|  | indent += x; | 
|  |  | 
|  | return; | 
|  | } | 
|  | else | 
|  | { | 
|  | printf ("\n"); | 
|  | atnl = 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (x == -1) | 
|  | { | 
|  | for (i = 0; i < indent; i++) | 
|  | printf ("   "); | 
|  |  | 
|  | indent += x; | 
|  | printf (")"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | indent += x; | 
|  |  | 
|  | for (i = 0; i < indent; i++) | 
|  | printf ("   "); | 
|  |  | 
|  | if (x) | 
|  | { | 
|  | printf ("("); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void | 
|  | nl (void) | 
|  | { | 
|  | atnl = 1; | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_lines (struct coff_line *p) | 
|  | { | 
|  | int i; | 
|  | int online = 0; | 
|  |  | 
|  | tab (1); | 
|  | printf (_("#lines %d "),p->nlines); | 
|  |  | 
|  | for (i = 0; i < p->nlines; i++) | 
|  | { | 
|  | printf ("(%d 0x%x)", p->lines[i], p->addresses[i]); | 
|  |  | 
|  | online++; | 
|  |  | 
|  | if (online > 6) | 
|  | { | 
|  | nl (); | 
|  | tab (0); | 
|  | online = 0; | 
|  | } | 
|  | } | 
|  | nl (); | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_type (struct coff_type *p) | 
|  | { | 
|  | tab (1); | 
|  | printf (_("size %d "), p->size); | 
|  |  | 
|  | switch (p->type) | 
|  | { | 
|  | case coff_secdef_type: | 
|  | printf (_("section definition at %x size %x\n"), | 
|  | p->u.asecdef.address, | 
|  | p->u.asecdef.size); | 
|  | nl (); | 
|  | break; | 
|  | case coff_pointer_type: | 
|  | printf (_("pointer to")); | 
|  | nl (); | 
|  | dump_coff_type (p->u.pointer.points_to); | 
|  | break; | 
|  | case coff_array_type: | 
|  | printf (_("array [%d] of"), p->u.array.dim); | 
|  | nl (); | 
|  | dump_coff_type (p->u.array.array_of); | 
|  | break; | 
|  | case coff_function_type: | 
|  | printf (_("function returning")); | 
|  | nl (); | 
|  | dump_coff_type (p->u.function.function_returns); | 
|  | dump_coff_lines (p->u.function.lines); | 
|  | printf (_("arguments")); | 
|  | nl (); | 
|  | dump_coff_scope (p->u.function.parameters); | 
|  | tab (0); | 
|  | printf (_("code")); | 
|  | nl (); | 
|  | dump_coff_scope (p->u.function.code); | 
|  | tab(0); | 
|  | break; | 
|  | case coff_structdef_type: | 
|  | printf (_("structure definition")); | 
|  | nl (); | 
|  | dump_coff_scope (p->u.astructdef.elements); | 
|  | break; | 
|  | case coff_structref_type: | 
|  | if (!p->u.aenumref.ref) | 
|  | printf (_("structure ref to UNKNOWN struct")); | 
|  | else | 
|  | printf (_("structure ref to %s"), p->u.aenumref.ref->name); | 
|  | break; | 
|  | case coff_enumref_type: | 
|  | printf (_("enum ref to %s"), p->u.astructref.ref->name); | 
|  | break; | 
|  | case coff_enumdef_type: | 
|  | printf (_("enum definition")); | 
|  | nl (); | 
|  | dump_coff_scope (p->u.aenumdef.elements); | 
|  | break; | 
|  | case coff_basic_type: | 
|  | switch (p->u.basic) | 
|  | { | 
|  | case T_NULL: | 
|  | printf ("NULL"); | 
|  | break; | 
|  | case T_VOID: | 
|  | printf ("VOID"); | 
|  | break; | 
|  | case T_CHAR: | 
|  | printf ("CHAR"); | 
|  | break; | 
|  | case T_SHORT: | 
|  | printf ("SHORT"); | 
|  | break; | 
|  | case T_INT: | 
|  | printf ("INT "); | 
|  | break; | 
|  | case T_LONG: | 
|  | printf ("LONG"); | 
|  | break; | 
|  | case T_FLOAT: | 
|  | printf ("FLOAT"); | 
|  | break; | 
|  | case T_DOUBLE: | 
|  | printf ("DOUBLE"); | 
|  | break; | 
|  | case T_STRUCT: | 
|  | printf ("STRUCT"); | 
|  | break; | 
|  | case T_UNION: | 
|  | printf ("UNION"); | 
|  | break; | 
|  | case T_ENUM: | 
|  | printf ("ENUM"); | 
|  | break; | 
|  | case T_MOE: | 
|  | printf ("MOE "); | 
|  | break; | 
|  | case T_UCHAR: | 
|  | printf ("UCHAR"); | 
|  | break; | 
|  | case T_USHORT: | 
|  | printf ("USHORT"); | 
|  | break; | 
|  | case T_UINT: | 
|  | printf ("UINT"); | 
|  | break; | 
|  | case T_ULONG: | 
|  | printf ("ULONG"); | 
|  | break; | 
|  | case T_LNGDBL: | 
|  | printf ("LNGDBL"); | 
|  | break; | 
|  | default: | 
|  | abort (); | 
|  | } | 
|  | } | 
|  | nl (); | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_where (struct coff_where *p) | 
|  | { | 
|  | tab (1); | 
|  | switch (p->where) | 
|  | { | 
|  | case coff_where_stack: | 
|  | printf (_("Stack offset %x"), p->offset); | 
|  | break; | 
|  | case coff_where_memory: | 
|  | printf (_("Memory section %s+%x"), p->section->name, p->offset); | 
|  | break; | 
|  | case coff_where_register: | 
|  | printf (_("Register %d"), p->offset); | 
|  | break; | 
|  | case coff_where_member_of_struct: | 
|  | printf (_("Struct Member offset %x"), p->offset); | 
|  | break; | 
|  | case coff_where_member_of_enum: | 
|  | printf (_("Enum Member offset %x"), p->offset); | 
|  | break; | 
|  | case coff_where_unknown: | 
|  | printf (_("Undefined symbol")); | 
|  | break; | 
|  | case coff_where_strtag: | 
|  | printf ("STRTAG"); | 
|  | break; | 
|  | case coff_where_entag: | 
|  | printf ("ENTAG"); | 
|  | break; | 
|  | case coff_where_typedef: | 
|  | printf ("TYPEDEF"); | 
|  | break; | 
|  | default: | 
|  | abort (); | 
|  | } | 
|  | nl (); | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_visible (struct coff_visible *p) | 
|  | { | 
|  | tab (1); | 
|  | switch (p->type) | 
|  | { | 
|  | case coff_vis_ext_def: | 
|  | printf ("coff_vis_ext_def"); | 
|  | break; | 
|  | case coff_vis_ext_ref: | 
|  | printf ("coff_vis_ext_ref"); | 
|  | break; | 
|  | case coff_vis_int_def: | 
|  | printf ("coff_vis_int_def"); | 
|  | break; | 
|  | case coff_vis_common: | 
|  | printf ("coff_vis_common"); | 
|  | break; | 
|  | case coff_vis_auto: | 
|  | printf ("coff_vis_auto"); | 
|  | break; | 
|  | case coff_vis_autoparam: | 
|  | printf ("coff_vis_autoparam"); | 
|  | break; | 
|  | case coff_vis_regparam: | 
|  | printf ("coff_vis_regparam"); | 
|  | break; | 
|  | case coff_vis_register: | 
|  | printf ("coff_vis_register"); | 
|  | break; | 
|  | case coff_vis_tag: | 
|  | printf ("coff_vis_tag"); | 
|  | break; | 
|  | case coff_vis_member_of_struct: | 
|  | printf ("coff_vis_member_of_struct"); | 
|  | break; | 
|  | case coff_vis_member_of_enum: | 
|  | printf ("coff_vis_member_of_enum"); | 
|  | break; | 
|  | default: | 
|  | abort (); | 
|  | } | 
|  | nl (); | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_symbol (struct coff_symbol *p) | 
|  | { | 
|  | tab (1); | 
|  | printf (_("List of symbols")); | 
|  | nl (); | 
|  |  | 
|  | while (p) | 
|  | { | 
|  | tab (1); | 
|  | tab (1); | 
|  | printf (_("Symbol  %s, tag %d, number %d"), p->name, p->tag, p->number); | 
|  | nl (); | 
|  | tab (-1); | 
|  | tab (1); | 
|  | printf (_("Type")); | 
|  | nl (); | 
|  | dump_coff_type (p->type); | 
|  | tab (-1); | 
|  | tab (1); | 
|  | printf (_("Where")); | 
|  | dump_coff_where (p->where); | 
|  | tab (-1); | 
|  | tab (1); | 
|  | printf (_("Visible")); | 
|  | dump_coff_visible (p->visible); | 
|  | tab (-1); | 
|  | p = p->next; | 
|  | tab (-1); | 
|  | } | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_scope (struct coff_scope *p) | 
|  | { | 
|  | if (p) | 
|  | { | 
|  | tab (1); | 
|  | printf ("%s %p ", _("List of blocks "), p); | 
|  |  | 
|  | if (p->sec) | 
|  | printf( "  %s %x..%x",  p->sec->name,p->offset, p->offset + p->size -1); | 
|  |  | 
|  | nl (); | 
|  | tab (0); | 
|  | printf ("*****************"); | 
|  | nl (); | 
|  |  | 
|  | while (p) | 
|  | { | 
|  | tab (0); | 
|  | printf (_("vars %d"), p->nvars); | 
|  | nl (); | 
|  | dump_coff_symbol (p->vars_head); | 
|  | printf (_("blocks")); | 
|  | nl (); | 
|  | dump_coff_scope (p->list_head); | 
|  | nl (); | 
|  | p = p->next; | 
|  | } | 
|  |  | 
|  | tab (0); | 
|  | printf ("*****************"); | 
|  | nl (); | 
|  | tab (-1); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_sfile (struct coff_sfile *p) | 
|  | { | 
|  | tab (1); | 
|  | printf (_("List of source files")); | 
|  | nl (); | 
|  |  | 
|  | while (p) | 
|  | { | 
|  | tab (0); | 
|  | printf (_("Source file %s"), p->name); | 
|  | nl (); | 
|  | dump_coff_scope (p->scope); | 
|  | p = p->next; | 
|  | } | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | dump_coff_section (struct coff_section *ptr) | 
|  | { | 
|  | unsigned int i; | 
|  |  | 
|  | tab (1); | 
|  | printf (_("section %s %d %d address %x size %x number %d nrelocs %u"), | 
|  | ptr->name, ptr->code, ptr->data, ptr->address,ptr->size, | 
|  | ptr->number, ptr->nrelocs); | 
|  | nl (); | 
|  |  | 
|  | for (i = 0; i < ptr->nrelocs; i++) | 
|  | { | 
|  | struct coff_reloc * r = ptr->relocs + i; | 
|  | tab (0); | 
|  | printf ("(%x %s %x)", | 
|  | r->offset, | 
|  | /* PR 17512: file: 0a38fb7c.  */ | 
|  | r->symbol == NULL ? _("<no sym>") : r->symbol->name, | 
|  | r->addend); | 
|  | nl (); | 
|  | } | 
|  |  | 
|  | tab (-1); | 
|  | } | 
|  |  | 
|  | static void | 
|  | coff_dump (struct coff_ofile *ptr) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | printf ("Coff dump"); | 
|  | nl (); | 
|  | printf (_("#sources %d"), ptr->nsources); | 
|  | nl (); | 
|  | dump_coff_sfile (ptr->source_head); | 
|  |  | 
|  | for (i = 0; i < ptr->nsections; i++) | 
|  | dump_coff_section (ptr->sections + i); | 
|  | } | 
|  |  | 
|  | static void | 
|  | show_usage (FILE *file, int status) | 
|  | { | 
|  | fprintf (file, _("Usage: %s [option(s)] in-file\n"), program_name); | 
|  | fprintf (file, _(" Print a human readable interpretation of a COFF object file\n")); | 
|  | fprintf (file, _(" The options are:\n\ | 
|  | @<file>                Read options from <file>\n\ | 
|  | -h --help              Display this information\n\ | 
|  | -v --version           Display the program's version\n\ | 
|  | \n")); | 
|  |  | 
|  | if (REPORT_BUGS_TO[0] && status == 0) | 
|  | fprintf (file, _("Report bugs to %s\n"), REPORT_BUGS_TO); | 
|  |  | 
|  | exit (status); | 
|  | } | 
|  |  | 
|  | int | 
|  | main (int ac, char **av) | 
|  | { | 
|  | bfd *abfd; | 
|  | struct coff_ofile *tree; | 
|  | char **matching; | 
|  | char *input_file = NULL; | 
|  | int opt; | 
|  | static struct option long_options[] = | 
|  | { | 
|  | { "help", no_argument, 0, 'h' }, | 
|  | { "version", no_argument, 0, 'V' }, | 
|  | { NULL, no_argument, 0, 0 } | 
|  | }; | 
|  |  | 
|  | #ifdef HAVE_LC_MESSAGES | 
|  | setlocale (LC_MESSAGES, ""); | 
|  | #endif | 
|  | setlocale (LC_CTYPE, ""); | 
|  | bindtextdomain (PACKAGE, LOCALEDIR); | 
|  | textdomain (PACKAGE); | 
|  |  | 
|  | program_name = av[0]; | 
|  | xmalloc_set_program_name (program_name); | 
|  | bfd_set_error_program_name (program_name); | 
|  |  | 
|  | expandargv (&ac, &av); | 
|  |  | 
|  | while ((opt = getopt_long (ac, av, "HhVv", long_options, | 
|  | (int *) NULL)) | 
|  | != EOF) | 
|  | { | 
|  | switch (opt) | 
|  | { | 
|  | case 'H': | 
|  | case 'h': | 
|  | show_usage (stdout, 0); | 
|  | break; | 
|  | case 'v': | 
|  | case 'V': | 
|  | print_version ("coffdump"); | 
|  | exit (0); | 
|  | case 0: | 
|  | break; | 
|  | default: | 
|  | show_usage (stderr, 1); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (optind < ac) | 
|  | { | 
|  | input_file = av[optind]; | 
|  | } | 
|  |  | 
|  | if (!input_file) | 
|  | fatal (_("no input file specified")); | 
|  |  | 
|  | abfd = bfd_openr (input_file, 0); | 
|  |  | 
|  | if (!abfd) | 
|  | bfd_fatal (input_file); | 
|  |  | 
|  | if (! bfd_check_format_matches (abfd, bfd_object, &matching)) | 
|  | { | 
|  | bfd_nonfatal (input_file); | 
|  |  | 
|  | if (bfd_get_error () == bfd_error_file_ambiguously_recognized) | 
|  | list_matching_formats (matching); | 
|  | exit (1); | 
|  | } | 
|  |  | 
|  | tree = coff_grok (abfd); | 
|  | if (tree) | 
|  | { | 
|  | coff_dump (tree); | 
|  | printf ("\n"); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } |