OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [gnu-stable/] [binutils-2.20.1/] [binutils/] [objdump.c] - Diff between revs 816 and 818

Go to most recent revision | Only display areas with differences | Details | Blame | View Log

Rev 816 Rev 818
/* objdump.c -- dump information about an object file.
/* objdump.c -- dump information about an object file.
   Copyright 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999,
   Copyright 1990, 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999,
   2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009
   2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007, 2008, 2009
   Free Software Foundation, Inc.
   Free Software Foundation, Inc.
 
 
   This file is part of GNU Binutils.
   This file is part of GNU Binutils.
 
 
   This program is free software; you can redistribute it and/or modify
   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
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 3, or (at your option)
   the Free Software Foundation; either version 3, or (at your option)
   any later version.
   any later version.
 
 
   This program is distributed in the hope that it will be useful,
   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.
   GNU General Public License for more details.
 
 
   You should have received a copy of the GNU General Public License
   You should have received a copy of the GNU General Public License
   along with this program; if not, write to the Free Software
   along with this program; if not, write to the Free Software
   Foundation, 51 Franklin Street - Fifth Floor, Boston,
   Foundation, 51 Franklin Street - Fifth Floor, Boston,
   MA 02110-1301, USA.  */
   MA 02110-1301, USA.  */
 
 
 
 
/* Objdump overview.
/* Objdump overview.
 
 
   Objdump displays information about one or more object files, either on
   Objdump displays information about one or more object files, either on
   their own, or inside libraries.  It is commonly used as a disassembler,
   their own, or inside libraries.  It is commonly used as a disassembler,
   but it can also display information about file headers, symbol tables,
   but it can also display information about file headers, symbol tables,
   relocations, debugging directives and more.
   relocations, debugging directives and more.
 
 
   The flow of execution is as follows:
   The flow of execution is as follows:
 
 
   1. Command line arguments are checked for control switches and the
   1. Command line arguments are checked for control switches and the
      information to be displayed is selected.
      information to be displayed is selected.
 
 
   2. Any remaining arguments are assumed to be object files, and they are
   2. Any remaining arguments are assumed to be object files, and they are
      processed in order by display_bfd().  If the file is an archive each
      processed in order by display_bfd().  If the file is an archive each
      of its elements is processed in turn.
      of its elements is processed in turn.
 
 
   3. The file's target architecture and binary file format are determined
   3. The file's target architecture and binary file format are determined
      by bfd_check_format().  If they are recognised, then dump_bfd() is
      by bfd_check_format().  If they are recognised, then dump_bfd() is
      called.
      called.
 
 
   4. dump_bfd() in turn calls separate functions to display the requested
   4. dump_bfd() in turn calls separate functions to display the requested
      item(s) of information(s).  For example disassemble_data() is called if
      item(s) of information(s).  For example disassemble_data() is called if
      a disassembly has been requested.
      a disassembly has been requested.
 
 
   When disassembling the code loops through blocks of instructions bounded
   When disassembling the code loops through blocks of instructions bounded
   by symbols, calling disassemble_bytes() on each block.  The actual
   by symbols, calling disassemble_bytes() on each block.  The actual
   disassembling is done by the libopcodes library, via a function pointer
   disassembling is done by the libopcodes library, via a function pointer
   supplied by the disassembler() function.  */
   supplied by the disassembler() function.  */
 
 
#include "sysdep.h"
#include "sysdep.h"
#include "bfd.h"
#include "bfd.h"
#include "elf-bfd.h"
#include "elf-bfd.h"
#include "progress.h"
#include "progress.h"
#include "bucomm.h"
#include "bucomm.h"
#include "dwarf.h"
#include "dwarf.h"
#include "getopt.h"
#include "getopt.h"
#include "safe-ctype.h"
#include "safe-ctype.h"
#include "dis-asm.h"
#include "dis-asm.h"
#include "libiberty.h"
#include "libiberty.h"
#include "demangle.h"
#include "demangle.h"
#include "filenames.h"
#include "filenames.h"
#include "debug.h"
#include "debug.h"
#include "budbg.h"
#include "budbg.h"
 
 
#ifdef HAVE_MMAP
#ifdef HAVE_MMAP
#include <sys/mman.h>
#include <sys/mman.h>
#endif
#endif
 
 
#include <sys/stat.h>
#include <sys/stat.h>
 
 
/* Internal headers for the ELF .stab-dump code - sorry.  */
/* Internal headers for the ELF .stab-dump code - sorry.  */
#define BYTES_IN_WORD   32
#define BYTES_IN_WORD   32
#include "aout/aout64.h"
#include "aout/aout64.h"
 
 
/* Exit status.  */
/* Exit status.  */
static int exit_status = 0;
static int exit_status = 0;
 
 
static char *default_target = NULL;     /* Default at runtime.  */
static char *default_target = NULL;     /* Default at runtime.  */
 
 
/* The following variables are set based on arguments passed on the
/* The following variables are set based on arguments passed on the
   command line.  */
   command line.  */
static int show_version = 0;             /* Show the version number.  */
static int show_version = 0;             /* Show the version number.  */
static int dump_section_contents;       /* -s */
static int dump_section_contents;       /* -s */
static int dump_section_headers;        /* -h */
static int dump_section_headers;        /* -h */
static bfd_boolean dump_file_header;    /* -f */
static bfd_boolean dump_file_header;    /* -f */
static int dump_symtab;                 /* -t */
static int dump_symtab;                 /* -t */
static int dump_dynamic_symtab;         /* -T */
static int dump_dynamic_symtab;         /* -T */
static int dump_reloc_info;             /* -r */
static int dump_reloc_info;             /* -r */
static int dump_dynamic_reloc_info;     /* -R */
static int dump_dynamic_reloc_info;     /* -R */
static int dump_ar_hdrs;                /* -a */
static int dump_ar_hdrs;                /* -a */
static int dump_private_headers;        /* -p */
static int dump_private_headers;        /* -p */
static int prefix_addresses;            /* --prefix-addresses */
static int prefix_addresses;            /* --prefix-addresses */
static int with_line_numbers;           /* -l */
static int with_line_numbers;           /* -l */
static bfd_boolean with_source_code;    /* -S */
static bfd_boolean with_source_code;    /* -S */
static int show_raw_insn;               /* --show-raw-insn */
static int show_raw_insn;               /* --show-raw-insn */
static int dump_dwarf_section_info;     /* --dwarf */
static int dump_dwarf_section_info;     /* --dwarf */
static int dump_stab_section_info;      /* --stabs */
static int dump_stab_section_info;      /* --stabs */
static int do_demangle;                 /* -C, --demangle */
static int do_demangle;                 /* -C, --demangle */
static bfd_boolean disassemble;         /* -d */
static bfd_boolean disassemble;         /* -d */
static bfd_boolean disassemble_all;     /* -D */
static bfd_boolean disassemble_all;     /* -D */
static int disassemble_zeroes;          /* --disassemble-zeroes */
static int disassemble_zeroes;          /* --disassemble-zeroes */
static bfd_boolean formats_info;        /* -i */
static bfd_boolean formats_info;        /* -i */
static int wide_output;                 /* -w */
static int wide_output;                 /* -w */
static int insn_width;                  /* --insn-width */
static int insn_width;                  /* --insn-width */
static bfd_vma start_address = (bfd_vma) -1; /* --start-address */
static bfd_vma start_address = (bfd_vma) -1; /* --start-address */
static bfd_vma stop_address = (bfd_vma) -1;  /* --stop-address */
static bfd_vma stop_address = (bfd_vma) -1;  /* --stop-address */
static int dump_debugging;              /* --debugging */
static int dump_debugging;              /* --debugging */
static int dump_debugging_tags;         /* --debugging-tags */
static int dump_debugging_tags;         /* --debugging-tags */
static int dump_special_syms = 0;        /* --special-syms */
static int dump_special_syms = 0;        /* --special-syms */
static bfd_vma adjust_section_vma = 0;   /* --adjust-vma */
static bfd_vma adjust_section_vma = 0;   /* --adjust-vma */
static int file_start_context = 0;      /* --file-start-context */
static int file_start_context = 0;      /* --file-start-context */
static bfd_boolean display_file_offsets;/* -F */
static bfd_boolean display_file_offsets;/* -F */
static const char *prefix;              /* --prefix */
static const char *prefix;              /* --prefix */
static int prefix_strip;                /* --prefix-strip */
static int prefix_strip;                /* --prefix-strip */
static size_t prefix_length;
static size_t prefix_length;
 
 
/* Pointer to an array of section names provided by
/* Pointer to an array of section names provided by
   one or more "-j secname" command line options.  */
   one or more "-j secname" command line options.  */
static char **only;
static char **only;
/* The total number of slots in the only[] array.  */
/* The total number of slots in the only[] array.  */
static size_t only_size = 0;
static size_t only_size = 0;
/* The number of occupied slots in the only[] array.  */
/* The number of occupied slots in the only[] array.  */
static size_t only_used = 0;
static size_t only_used = 0;
 
 
/* Variables for handling include file path table.  */
/* Variables for handling include file path table.  */
static const char **include_paths;
static const char **include_paths;
static int include_path_count;
static int include_path_count;
 
 
/* Extra info to pass to the section disassembler and address printing
/* Extra info to pass to the section disassembler and address printing
   function.  */
   function.  */
struct objdump_disasm_info
struct objdump_disasm_info
{
{
  bfd *              abfd;
  bfd *              abfd;
  asection *         sec;
  asection *         sec;
  bfd_boolean        require_sec;
  bfd_boolean        require_sec;
  arelent **         dynrelbuf;
  arelent **         dynrelbuf;
  long               dynrelcount;
  long               dynrelcount;
  disassembler_ftype disassemble_fn;
  disassembler_ftype disassemble_fn;
  arelent *          reloc;
  arelent *          reloc;
};
};
 
 
/* Architecture to disassemble for, or default if NULL.  */
/* Architecture to disassemble for, or default if NULL.  */
static char *machine = NULL;
static char *machine = NULL;
 
 
/* Target specific options to the disassembler.  */
/* Target specific options to the disassembler.  */
static char *disassembler_options = NULL;
static char *disassembler_options = NULL;
 
 
/* Endianness to disassemble for, or default if BFD_ENDIAN_UNKNOWN.  */
/* Endianness to disassemble for, or default if BFD_ENDIAN_UNKNOWN.  */
static enum bfd_endian endian = BFD_ENDIAN_UNKNOWN;
static enum bfd_endian endian = BFD_ENDIAN_UNKNOWN;
 
 
/* The symbol table.  */
/* The symbol table.  */
static asymbol **syms;
static asymbol **syms;
 
 
/* Number of symbols in `syms'.  */
/* Number of symbols in `syms'.  */
static long symcount = 0;
static long symcount = 0;
 
 
/* The sorted symbol table.  */
/* The sorted symbol table.  */
static asymbol **sorted_syms;
static asymbol **sorted_syms;
 
 
/* Number of symbols in `sorted_syms'.  */
/* Number of symbols in `sorted_syms'.  */
static long sorted_symcount = 0;
static long sorted_symcount = 0;
 
 
/* The dynamic symbol table.  */
/* The dynamic symbol table.  */
static asymbol **dynsyms;
static asymbol **dynsyms;
 
 
/* The synthetic symbol table.  */
/* The synthetic symbol table.  */
static asymbol *synthsyms;
static asymbol *synthsyms;
static long synthcount = 0;
static long synthcount = 0;
 
 
/* Number of symbols in `dynsyms'.  */
/* Number of symbols in `dynsyms'.  */
static long dynsymcount = 0;
static long dynsymcount = 0;
 
 
static bfd_byte *stabs;
static bfd_byte *stabs;
static bfd_size_type stab_size;
static bfd_size_type stab_size;
 
 
static char *strtab;
static char *strtab;
static bfd_size_type stabstr_size;
static bfd_size_type stabstr_size;
 
 
static bfd_boolean is_relocatable = FALSE;
static bfd_boolean is_relocatable = FALSE;


static void
static void
usage (FILE *stream, int status)
usage (FILE *stream, int status)
{
{
  fprintf (stream, _("Usage: %s <option(s)> <file(s)>\n"), program_name);
  fprintf (stream, _("Usage: %s <option(s)> <file(s)>\n"), program_name);
  fprintf (stream, _(" Display information from object <file(s)>.\n"));
  fprintf (stream, _(" Display information from object <file(s)>.\n"));
  fprintf (stream, _(" At least one of the following switches must be given:\n"));
  fprintf (stream, _(" At least one of the following switches must be given:\n"));
  fprintf (stream, _("\
  fprintf (stream, _("\
  -a, --archive-headers    Display archive header information\n\
  -a, --archive-headers    Display archive header information\n\
  -f, --file-headers       Display the contents of the overall file header\n\
  -f, --file-headers       Display the contents of the overall file header\n\
  -p, --private-headers    Display object format specific file header contents\n\
  -p, --private-headers    Display object format specific file header contents\n\
  -h, --[section-]headers  Display the contents of the section headers\n\
  -h, --[section-]headers  Display the contents of the section headers\n\
  -x, --all-headers        Display the contents of all headers\n\
  -x, --all-headers        Display the contents of all headers\n\
  -d, --disassemble        Display assembler contents of executable sections\n\
  -d, --disassemble        Display assembler contents of executable sections\n\
  -D, --disassemble-all    Display assembler contents of all sections\n\
  -D, --disassemble-all    Display assembler contents of all sections\n\
  -S, --source             Intermix source code with disassembly\n\
  -S, --source             Intermix source code with disassembly\n\
  -s, --full-contents      Display the full contents of all sections requested\n\
  -s, --full-contents      Display the full contents of all sections requested\n\
  -g, --debugging          Display debug information in object file\n\
  -g, --debugging          Display debug information in object file\n\
  -e, --debugging-tags     Display debug information using ctags style\n\
  -e, --debugging-tags     Display debug information using ctags style\n\
  -G, --stabs              Display (in raw form) any STABS info in the file\n\
  -G, --stabs              Display (in raw form) any STABS info in the file\n\
  -W[lLiaprmfFsoR] or\n\
  -W[lLiaprmfFsoR] or\n\
  --dwarf[=rawline,=decodedline,=info,=abbrev,=pubnames,=aranges,=macro,=frames,=str,=loc,=Ranges]\n\
  --dwarf[=rawline,=decodedline,=info,=abbrev,=pubnames,=aranges,=macro,=frames,=str,=loc,=Ranges]\n\
                           Display DWARF info in the file\n\
                           Display DWARF info in the file\n\
  -t, --syms               Display the contents of the symbol table(s)\n\
  -t, --syms               Display the contents of the symbol table(s)\n\
  -T, --dynamic-syms       Display the contents of the dynamic symbol table\n\
  -T, --dynamic-syms       Display the contents of the dynamic symbol table\n\
  -r, --reloc              Display the relocation entries in the file\n\
  -r, --reloc              Display the relocation entries in the file\n\
  -R, --dynamic-reloc      Display the dynamic relocation entries in the file\n\
  -R, --dynamic-reloc      Display the dynamic relocation entries in the file\n\
  @<file>                  Read options from <file>\n\
  @<file>                  Read options from <file>\n\
  -v, --version            Display this program's version number\n\
  -v, --version            Display this program's version number\n\
  -i, --info               List object formats and architectures supported\n\
  -i, --info               List object formats and architectures supported\n\
  -H, --help               Display this information\n\
  -H, --help               Display this information\n\
"));
"));
  if (status != 2)
  if (status != 2)
    {
    {
      fprintf (stream, _("\n The following switches are optional:\n"));
      fprintf (stream, _("\n The following switches are optional:\n"));
      fprintf (stream, _("\
      fprintf (stream, _("\
  -b, --target=BFDNAME           Specify the target object format as BFDNAME\n\
  -b, --target=BFDNAME           Specify the target object format as BFDNAME\n\
  -m, --architecture=MACHINE     Specify the target architecture as MACHINE\n\
  -m, --architecture=MACHINE     Specify the target architecture as MACHINE\n\
  -j, --section=NAME             Only display information for section NAME\n\
  -j, --section=NAME             Only display information for section NAME\n\
  -M, --disassembler-options=OPT Pass text OPT on to the disassembler\n\
  -M, --disassembler-options=OPT Pass text OPT on to the disassembler\n\
  -EB --endian=big               Assume big endian format when disassembling\n\
  -EB --endian=big               Assume big endian format when disassembling\n\
  -EL --endian=little            Assume little endian format when disassembling\n\
  -EL --endian=little            Assume little endian format when disassembling\n\
      --file-start-context       Include context from start of file (with -S)\n\
      --file-start-context       Include context from start of file (with -S)\n\
  -I, --include=DIR              Add DIR to search list for source files\n\
  -I, --include=DIR              Add DIR to search list for source files\n\
  -l, --line-numbers             Include line numbers and filenames in output\n\
  -l, --line-numbers             Include line numbers and filenames in output\n\
  -F, --file-offsets             Include file offsets when displaying information\n\
  -F, --file-offsets             Include file offsets when displaying information\n\
  -C, --demangle[=STYLE]         Decode mangled/processed symbol names\n\
  -C, --demangle[=STYLE]         Decode mangled/processed symbol names\n\
                                  The STYLE, if specified, can be `auto', `gnu',\n\
                                  The STYLE, if specified, can be `auto', `gnu',\n\
                                  `lucid', `arm', `hp', `edg', `gnu-v3', `java'\n\
                                  `lucid', `arm', `hp', `edg', `gnu-v3', `java'\n\
                                  or `gnat'\n\
                                  or `gnat'\n\
  -w, --wide                     Format output for more than 80 columns\n\
  -w, --wide                     Format output for more than 80 columns\n\
  -z, --disassemble-zeroes       Do not skip blocks of zeroes when disassembling\n\
  -z, --disassemble-zeroes       Do not skip blocks of zeroes when disassembling\n\
      --start-address=ADDR       Only process data whose address is >= ADDR\n\
      --start-address=ADDR       Only process data whose address is >= ADDR\n\
      --stop-address=ADDR        Only process data whose address is <= ADDR\n\
      --stop-address=ADDR        Only process data whose address is <= ADDR\n\
      --prefix-addresses         Print complete address alongside disassembly\n\
      --prefix-addresses         Print complete address alongside disassembly\n\
      --[no-]show-raw-insn       Display hex alongside symbolic disassembly\n\
      --[no-]show-raw-insn       Display hex alongside symbolic disassembly\n\
      --insn-width=WIDTH         Display WIDTH bytes on a signle line for -d\n\
      --insn-width=WIDTH         Display WIDTH bytes on a signle line for -d\n\
      --adjust-vma=OFFSET        Add OFFSET to all displayed section addresses\n\
      --adjust-vma=OFFSET        Add OFFSET to all displayed section addresses\n\
      --special-syms             Include special symbols in symbol dumps\n\
      --special-syms             Include special symbols in symbol dumps\n\
      --prefix=PREFIX            Add PREFIX to absolute paths for -S\n\
      --prefix=PREFIX            Add PREFIX to absolute paths for -S\n\
      --prefix-strip=LEVEL       Strip initial directory names for -S\n\
      --prefix-strip=LEVEL       Strip initial directory names for -S\n\
\n"));
\n"));
      list_supported_targets (program_name, stream);
      list_supported_targets (program_name, stream);
      list_supported_architectures (program_name, stream);
      list_supported_architectures (program_name, stream);
 
 
      disassembler_usage (stream);
      disassembler_usage (stream);
    }
    }
  if (REPORT_BUGS_TO[0] && status == 0)
  if (REPORT_BUGS_TO[0] && status == 0)
    fprintf (stream, _("Report bugs to %s.\n"), REPORT_BUGS_TO);
    fprintf (stream, _("Report bugs to %s.\n"), REPORT_BUGS_TO);
  exit (status);
  exit (status);
}
}
 
 
/* 150 isn't special; it's just an arbitrary non-ASCII char value.  */
/* 150 isn't special; it's just an arbitrary non-ASCII char value.  */
enum option_values
enum option_values
  {
  {
    OPTION_ENDIAN=150,
    OPTION_ENDIAN=150,
    OPTION_START_ADDRESS,
    OPTION_START_ADDRESS,
    OPTION_STOP_ADDRESS,
    OPTION_STOP_ADDRESS,
    OPTION_DWARF,
    OPTION_DWARF,
    OPTION_PREFIX,
    OPTION_PREFIX,
    OPTION_PREFIX_STRIP,
    OPTION_PREFIX_STRIP,
    OPTION_INSN_WIDTH,
    OPTION_INSN_WIDTH,
    OPTION_ADJUST_VMA
    OPTION_ADJUST_VMA
  };
  };
 
 
static struct option long_options[]=
static struct option long_options[]=
{
{
  {"adjust-vma", required_argument, NULL, OPTION_ADJUST_VMA},
  {"adjust-vma", required_argument, NULL, OPTION_ADJUST_VMA},
  {"all-headers", no_argument, NULL, 'x'},
  {"all-headers", no_argument, NULL, 'x'},
  {"private-headers", no_argument, NULL, 'p'},
  {"private-headers", no_argument, NULL, 'p'},
  {"architecture", required_argument, NULL, 'm'},
  {"architecture", required_argument, NULL, 'm'},
  {"archive-headers", no_argument, NULL, 'a'},
  {"archive-headers", no_argument, NULL, 'a'},
  {"debugging", no_argument, NULL, 'g'},
  {"debugging", no_argument, NULL, 'g'},
  {"debugging-tags", no_argument, NULL, 'e'},
  {"debugging-tags", no_argument, NULL, 'e'},
  {"demangle", optional_argument, NULL, 'C'},
  {"demangle", optional_argument, NULL, 'C'},
  {"disassemble", no_argument, NULL, 'd'},
  {"disassemble", no_argument, NULL, 'd'},
  {"disassemble-all", no_argument, NULL, 'D'},
  {"disassemble-all", no_argument, NULL, 'D'},
  {"disassembler-options", required_argument, NULL, 'M'},
  {"disassembler-options", required_argument, NULL, 'M'},
  {"disassemble-zeroes", no_argument, NULL, 'z'},
  {"disassemble-zeroes", no_argument, NULL, 'z'},
  {"dynamic-reloc", no_argument, NULL, 'R'},
  {"dynamic-reloc", no_argument, NULL, 'R'},
  {"dynamic-syms", no_argument, NULL, 'T'},
  {"dynamic-syms", no_argument, NULL, 'T'},
  {"endian", required_argument, NULL, OPTION_ENDIAN},
  {"endian", required_argument, NULL, OPTION_ENDIAN},
  {"file-headers", no_argument, NULL, 'f'},
  {"file-headers", no_argument, NULL, 'f'},
  {"file-offsets", no_argument, NULL, 'F'},
  {"file-offsets", no_argument, NULL, 'F'},
  {"file-start-context", no_argument, &file_start_context, 1},
  {"file-start-context", no_argument, &file_start_context, 1},
  {"full-contents", no_argument, NULL, 's'},
  {"full-contents", no_argument, NULL, 's'},
  {"headers", no_argument, NULL, 'h'},
  {"headers", no_argument, NULL, 'h'},
  {"help", no_argument, NULL, 'H'},
  {"help", no_argument, NULL, 'H'},
  {"info", no_argument, NULL, 'i'},
  {"info", no_argument, NULL, 'i'},
  {"line-numbers", no_argument, NULL, 'l'},
  {"line-numbers", no_argument, NULL, 'l'},
  {"no-show-raw-insn", no_argument, &show_raw_insn, -1},
  {"no-show-raw-insn", no_argument, &show_raw_insn, -1},
  {"prefix-addresses", no_argument, &prefix_addresses, 1},
  {"prefix-addresses", no_argument, &prefix_addresses, 1},
  {"reloc", no_argument, NULL, 'r'},
  {"reloc", no_argument, NULL, 'r'},
  {"section", required_argument, NULL, 'j'},
  {"section", required_argument, NULL, 'j'},
  {"section-headers", no_argument, NULL, 'h'},
  {"section-headers", no_argument, NULL, 'h'},
  {"show-raw-insn", no_argument, &show_raw_insn, 1},
  {"show-raw-insn", no_argument, &show_raw_insn, 1},
  {"source", no_argument, NULL, 'S'},
  {"source", no_argument, NULL, 'S'},
  {"special-syms", no_argument, &dump_special_syms, 1},
  {"special-syms", no_argument, &dump_special_syms, 1},
  {"include", required_argument, NULL, 'I'},
  {"include", required_argument, NULL, 'I'},
  {"dwarf", optional_argument, NULL, OPTION_DWARF},
  {"dwarf", optional_argument, NULL, OPTION_DWARF},
  {"stabs", no_argument, NULL, 'G'},
  {"stabs", no_argument, NULL, 'G'},
  {"start-address", required_argument, NULL, OPTION_START_ADDRESS},
  {"start-address", required_argument, NULL, OPTION_START_ADDRESS},
  {"stop-address", required_argument, NULL, OPTION_STOP_ADDRESS},
  {"stop-address", required_argument, NULL, OPTION_STOP_ADDRESS},
  {"syms", no_argument, NULL, 't'},
  {"syms", no_argument, NULL, 't'},
  {"target", required_argument, NULL, 'b'},
  {"target", required_argument, NULL, 'b'},
  {"version", no_argument, NULL, 'V'},
  {"version", no_argument, NULL, 'V'},
  {"wide", no_argument, NULL, 'w'},
  {"wide", no_argument, NULL, 'w'},
  {"prefix", required_argument, NULL, OPTION_PREFIX},
  {"prefix", required_argument, NULL, OPTION_PREFIX},
  {"prefix-strip", required_argument, NULL, OPTION_PREFIX_STRIP},
  {"prefix-strip", required_argument, NULL, OPTION_PREFIX_STRIP},
  {"insn-width", required_argument, NULL, OPTION_INSN_WIDTH},
  {"insn-width", required_argument, NULL, OPTION_INSN_WIDTH},
  {0, no_argument, 0, 0}
  {0, no_argument, 0, 0}
};
};


static void
static void
nonfatal (const char *msg)
nonfatal (const char *msg)
{
{
  bfd_nonfatal (msg);
  bfd_nonfatal (msg);
  exit_status = 1;
  exit_status = 1;
}
}


/* Returns TRUE if the specified section should be dumped.  */
/* Returns TRUE if the specified section should be dumped.  */
 
 
static bfd_boolean
static bfd_boolean
process_section_p (asection * section)
process_section_p (asection * section)
{
{
  size_t i;
  size_t i;
 
 
  if (only == NULL)
  if (only == NULL)
    return TRUE;
    return TRUE;
 
 
  for (i = 0; i < only_used; i++)
  for (i = 0; i < only_used; i++)
    if (strcmp (only [i], section->name) == 0)
    if (strcmp (only [i], section->name) == 0)
      return TRUE;
      return TRUE;
 
 
  return FALSE;
  return FALSE;
}
}


static void
static void
dump_section_header (bfd *abfd, asection *section,
dump_section_header (bfd *abfd, asection *section,
                     void *ignored ATTRIBUTE_UNUSED)
                     void *ignored ATTRIBUTE_UNUSED)
{
{
  char *comma = "";
  char *comma = "";
  unsigned int opb = bfd_octets_per_byte (abfd);
  unsigned int opb = bfd_octets_per_byte (abfd);
 
 
  /* Ignore linker created section.  See elfNN_ia64_object_p in
  /* Ignore linker created section.  See elfNN_ia64_object_p in
     bfd/elfxx-ia64.c.  */
     bfd/elfxx-ia64.c.  */
  if (section->flags & SEC_LINKER_CREATED)
  if (section->flags & SEC_LINKER_CREATED)
    return;
    return;
 
 
  /* PR 10413: Skip sections that we are ignoring.  */
  /* PR 10413: Skip sections that we are ignoring.  */
  if (! process_section_p (section))
  if (! process_section_p (section))
    return;
    return;
 
 
  printf ("%3d %-13s %08lx  ", section->index,
  printf ("%3d %-13s %08lx  ", section->index,
          bfd_get_section_name (abfd, section),
          bfd_get_section_name (abfd, section),
          (unsigned long) bfd_section_size (abfd, section) / opb);
          (unsigned long) bfd_section_size (abfd, section) / opb);
  bfd_printf_vma (abfd, bfd_get_section_vma (abfd, section));
  bfd_printf_vma (abfd, bfd_get_section_vma (abfd, section));
  printf ("  ");
  printf ("  ");
  bfd_printf_vma (abfd, section->lma);
  bfd_printf_vma (abfd, section->lma);
  printf ("  %08lx  2**%u", (unsigned long) section->filepos,
  printf ("  %08lx  2**%u", (unsigned long) section->filepos,
          bfd_get_section_alignment (abfd, section));
          bfd_get_section_alignment (abfd, section));
  if (! wide_output)
  if (! wide_output)
    printf ("\n                ");
    printf ("\n                ");
  printf ("  ");
  printf ("  ");
 
 
#define PF(x, y) \
#define PF(x, y) \
  if (section->flags & x) { printf ("%s%s", comma, y); comma = ", "; }
  if (section->flags & x) { printf ("%s%s", comma, y); comma = ", "; }
 
 
  PF (SEC_HAS_CONTENTS, "CONTENTS");
  PF (SEC_HAS_CONTENTS, "CONTENTS");
  PF (SEC_ALLOC, "ALLOC");
  PF (SEC_ALLOC, "ALLOC");
  PF (SEC_CONSTRUCTOR, "CONSTRUCTOR");
  PF (SEC_CONSTRUCTOR, "CONSTRUCTOR");
  PF (SEC_LOAD, "LOAD");
  PF (SEC_LOAD, "LOAD");
  PF (SEC_RELOC, "RELOC");
  PF (SEC_RELOC, "RELOC");
  PF (SEC_READONLY, "READONLY");
  PF (SEC_READONLY, "READONLY");
  PF (SEC_CODE, "CODE");
  PF (SEC_CODE, "CODE");
  PF (SEC_DATA, "DATA");
  PF (SEC_DATA, "DATA");
  PF (SEC_ROM, "ROM");
  PF (SEC_ROM, "ROM");
  PF (SEC_DEBUGGING, "DEBUGGING");
  PF (SEC_DEBUGGING, "DEBUGGING");
  PF (SEC_NEVER_LOAD, "NEVER_LOAD");
  PF (SEC_NEVER_LOAD, "NEVER_LOAD");
  PF (SEC_EXCLUDE, "EXCLUDE");
  PF (SEC_EXCLUDE, "EXCLUDE");
  PF (SEC_SORT_ENTRIES, "SORT_ENTRIES");
  PF (SEC_SORT_ENTRIES, "SORT_ENTRIES");
  if (bfd_get_arch (abfd) == bfd_arch_tic54x)
  if (bfd_get_arch (abfd) == bfd_arch_tic54x)
    {
    {
      PF (SEC_TIC54X_BLOCK, "BLOCK");
      PF (SEC_TIC54X_BLOCK, "BLOCK");
      PF (SEC_TIC54X_CLINK, "CLINK");
      PF (SEC_TIC54X_CLINK, "CLINK");
    }
    }
  PF (SEC_SMALL_DATA, "SMALL_DATA");
  PF (SEC_SMALL_DATA, "SMALL_DATA");
  if (bfd_get_flavour (abfd) == bfd_target_coff_flavour)
  if (bfd_get_flavour (abfd) == bfd_target_coff_flavour)
    PF (SEC_COFF_SHARED, "SHARED");
    PF (SEC_COFF_SHARED, "SHARED");
  PF (SEC_THREAD_LOCAL, "THREAD_LOCAL");
  PF (SEC_THREAD_LOCAL, "THREAD_LOCAL");
  PF (SEC_GROUP, "GROUP");
  PF (SEC_GROUP, "GROUP");
 
 
  if ((section->flags & SEC_LINK_ONCE) != 0)
  if ((section->flags & SEC_LINK_ONCE) != 0)
    {
    {
      const char *ls;
      const char *ls;
      struct coff_comdat_info *comdat;
      struct coff_comdat_info *comdat;
 
 
      switch (section->flags & SEC_LINK_DUPLICATES)
      switch (section->flags & SEC_LINK_DUPLICATES)
        {
        {
        default:
        default:
          abort ();
          abort ();
        case SEC_LINK_DUPLICATES_DISCARD:
        case SEC_LINK_DUPLICATES_DISCARD:
          ls = "LINK_ONCE_DISCARD";
          ls = "LINK_ONCE_DISCARD";
          break;
          break;
        case SEC_LINK_DUPLICATES_ONE_ONLY:
        case SEC_LINK_DUPLICATES_ONE_ONLY:
          ls = "LINK_ONCE_ONE_ONLY";
          ls = "LINK_ONCE_ONE_ONLY";
          break;
          break;
        case SEC_LINK_DUPLICATES_SAME_SIZE:
        case SEC_LINK_DUPLICATES_SAME_SIZE:
          ls = "LINK_ONCE_SAME_SIZE";
          ls = "LINK_ONCE_SAME_SIZE";
          break;
          break;
        case SEC_LINK_DUPLICATES_SAME_CONTENTS:
        case SEC_LINK_DUPLICATES_SAME_CONTENTS:
          ls = "LINK_ONCE_SAME_CONTENTS";
          ls = "LINK_ONCE_SAME_CONTENTS";
          break;
          break;
        }
        }
      printf ("%s%s", comma, ls);
      printf ("%s%s", comma, ls);
 
 
      comdat = bfd_coff_get_comdat_section (abfd, section);
      comdat = bfd_coff_get_comdat_section (abfd, section);
      if (comdat != NULL)
      if (comdat != NULL)
        printf (" (COMDAT %s %ld)", comdat->name, comdat->symbol);
        printf (" (COMDAT %s %ld)", comdat->name, comdat->symbol);
 
 
      comma = ", ";
      comma = ", ";
    }
    }
 
 
  printf ("\n");
  printf ("\n");
#undef PF
#undef PF
}
}
 
 
static void
static void
dump_headers (bfd *abfd)
dump_headers (bfd *abfd)
{
{
  printf (_("Sections:\n"));
  printf (_("Sections:\n"));
 
 
#ifndef BFD64
#ifndef BFD64
  printf (_("Idx Name          Size      VMA       LMA       File off  Algn"));
  printf (_("Idx Name          Size      VMA       LMA       File off  Algn"));
#else
#else
  /* With BFD64, non-ELF returns -1 and wants always 64 bit addresses.  */
  /* With BFD64, non-ELF returns -1 and wants always 64 bit addresses.  */
  if (bfd_get_arch_size (abfd) == 32)
  if (bfd_get_arch_size (abfd) == 32)
    printf (_("Idx Name          Size      VMA       LMA       File off  Algn"));
    printf (_("Idx Name          Size      VMA       LMA       File off  Algn"));
  else
  else
    printf (_("Idx Name          Size      VMA               LMA               File off  Algn"));
    printf (_("Idx Name          Size      VMA               LMA               File off  Algn"));
#endif
#endif
 
 
  if (wide_output)
  if (wide_output)
    printf (_("  Flags"));
    printf (_("  Flags"));
  if (abfd->flags & HAS_LOAD_PAGE)
  if (abfd->flags & HAS_LOAD_PAGE)
    printf (_("  Pg"));
    printf (_("  Pg"));
  printf ("\n");
  printf ("\n");
 
 
  bfd_map_over_sections (abfd, dump_section_header, NULL);
  bfd_map_over_sections (abfd, dump_section_header, NULL);
}
}


static asymbol **
static asymbol **
slurp_symtab (bfd *abfd)
slurp_symtab (bfd *abfd)
{
{
  asymbol **sy = NULL;
  asymbol **sy = NULL;
  long storage;
  long storage;
 
 
  if (!(bfd_get_file_flags (abfd) & HAS_SYMS))
  if (!(bfd_get_file_flags (abfd) & HAS_SYMS))
    {
    {
      symcount = 0;
      symcount = 0;
      return NULL;
      return NULL;
    }
    }
 
 
  storage = bfd_get_symtab_upper_bound (abfd);
  storage = bfd_get_symtab_upper_bound (abfd);
  if (storage < 0)
  if (storage < 0)
    bfd_fatal (bfd_get_filename (abfd));
    bfd_fatal (bfd_get_filename (abfd));
  if (storage)
  if (storage)
    sy = (asymbol **) xmalloc (storage);
    sy = (asymbol **) xmalloc (storage);
 
 
  symcount = bfd_canonicalize_symtab (abfd, sy);
  symcount = bfd_canonicalize_symtab (abfd, sy);
  if (symcount < 0)
  if (symcount < 0)
    bfd_fatal (bfd_get_filename (abfd));
    bfd_fatal (bfd_get_filename (abfd));
  return sy;
  return sy;
}
}
 
 
/* Read in the dynamic symbols.  */
/* Read in the dynamic symbols.  */
 
 
static asymbol **
static asymbol **
slurp_dynamic_symtab (bfd *abfd)
slurp_dynamic_symtab (bfd *abfd)
{
{
  asymbol **sy = NULL;
  asymbol **sy = NULL;
  long storage;
  long storage;
 
 
  storage = bfd_get_dynamic_symtab_upper_bound (abfd);
  storage = bfd_get_dynamic_symtab_upper_bound (abfd);
  if (storage < 0)
  if (storage < 0)
    {
    {
      if (!(bfd_get_file_flags (abfd) & DYNAMIC))
      if (!(bfd_get_file_flags (abfd) & DYNAMIC))
        {
        {
          non_fatal (_("%s: not a dynamic object"), bfd_get_filename (abfd));
          non_fatal (_("%s: not a dynamic object"), bfd_get_filename (abfd));
          dynsymcount = 0;
          dynsymcount = 0;
          return NULL;
          return NULL;
        }
        }
 
 
      bfd_fatal (bfd_get_filename (abfd));
      bfd_fatal (bfd_get_filename (abfd));
    }
    }
  if (storage)
  if (storage)
    sy = (asymbol **) xmalloc (storage);
    sy = (asymbol **) xmalloc (storage);
 
 
  dynsymcount = bfd_canonicalize_dynamic_symtab (abfd, sy);
  dynsymcount = bfd_canonicalize_dynamic_symtab (abfd, sy);
  if (dynsymcount < 0)
  if (dynsymcount < 0)
    bfd_fatal (bfd_get_filename (abfd));
    bfd_fatal (bfd_get_filename (abfd));
  return sy;
  return sy;
}
}
 
 
/* Filter out (in place) symbols that are useless for disassembly.
/* Filter out (in place) symbols that are useless for disassembly.
   COUNT is the number of elements in SYMBOLS.
   COUNT is the number of elements in SYMBOLS.
   Return the number of useful symbols.  */
   Return the number of useful symbols.  */
 
 
static long
static long
remove_useless_symbols (asymbol **symbols, long count)
remove_useless_symbols (asymbol **symbols, long count)
{
{
  asymbol **in_ptr = symbols, **out_ptr = symbols;
  asymbol **in_ptr = symbols, **out_ptr = symbols;
 
 
  while (--count >= 0)
  while (--count >= 0)
    {
    {
      asymbol *sym = *in_ptr++;
      asymbol *sym = *in_ptr++;
 
 
      if (sym->name == NULL || sym->name[0] == '\0')
      if (sym->name == NULL || sym->name[0] == '\0')
        continue;
        continue;
      if (sym->flags & (BSF_DEBUGGING | BSF_SECTION_SYM))
      if (sym->flags & (BSF_DEBUGGING | BSF_SECTION_SYM))
        continue;
        continue;
      if (bfd_is_und_section (sym->section)
      if (bfd_is_und_section (sym->section)
          || bfd_is_com_section (sym->section))
          || bfd_is_com_section (sym->section))
        continue;
        continue;
 
 
      *out_ptr++ = sym;
      *out_ptr++ = sym;
    }
    }
  return out_ptr - symbols;
  return out_ptr - symbols;
}
}
 
 
/* Sort symbols into value order.  */
/* Sort symbols into value order.  */
 
 
static int
static int
compare_symbols (const void *ap, const void *bp)
compare_symbols (const void *ap, const void *bp)
{
{
  const asymbol *a = * (const asymbol **) ap;
  const asymbol *a = * (const asymbol **) ap;
  const asymbol *b = * (const asymbol **) bp;
  const asymbol *b = * (const asymbol **) bp;
  const char *an;
  const char *an;
  const char *bn;
  const char *bn;
  size_t anl;
  size_t anl;
  size_t bnl;
  size_t bnl;
  bfd_boolean af;
  bfd_boolean af;
  bfd_boolean bf;
  bfd_boolean bf;
  flagword aflags;
  flagword aflags;
  flagword bflags;
  flagword bflags;
 
 
  if (bfd_asymbol_value (a) > bfd_asymbol_value (b))
  if (bfd_asymbol_value (a) > bfd_asymbol_value (b))
    return 1;
    return 1;
  else if (bfd_asymbol_value (a) < bfd_asymbol_value (b))
  else if (bfd_asymbol_value (a) < bfd_asymbol_value (b))
    return -1;
    return -1;
 
 
  if (a->section > b->section)
  if (a->section > b->section)
    return 1;
    return 1;
  else if (a->section < b->section)
  else if (a->section < b->section)
    return -1;
    return -1;
 
 
  an = bfd_asymbol_name (a);
  an = bfd_asymbol_name (a);
  bn = bfd_asymbol_name (b);
  bn = bfd_asymbol_name (b);
  anl = strlen (an);
  anl = strlen (an);
  bnl = strlen (bn);
  bnl = strlen (bn);
 
 
  /* The symbols gnu_compiled and gcc2_compiled convey no real
  /* The symbols gnu_compiled and gcc2_compiled convey no real
     information, so put them after other symbols with the same value.  */
     information, so put them after other symbols with the same value.  */
  af = (strstr (an, "gnu_compiled") != NULL
  af = (strstr (an, "gnu_compiled") != NULL
        || strstr (an, "gcc2_compiled") != NULL);
        || strstr (an, "gcc2_compiled") != NULL);
  bf = (strstr (bn, "gnu_compiled") != NULL
  bf = (strstr (bn, "gnu_compiled") != NULL
        || strstr (bn, "gcc2_compiled") != NULL);
        || strstr (bn, "gcc2_compiled") != NULL);
 
 
  if (af && ! bf)
  if (af && ! bf)
    return 1;
    return 1;
  if (! af && bf)
  if (! af && bf)
    return -1;
    return -1;
 
 
  /* We use a heuristic for the file name, to try to sort it after
  /* We use a heuristic for the file name, to try to sort it after
     more useful symbols.  It may not work on non Unix systems, but it
     more useful symbols.  It may not work on non Unix systems, but it
     doesn't really matter; the only difference is precisely which
     doesn't really matter; the only difference is precisely which
     symbol names get printed.  */
     symbol names get printed.  */
 
 
#define file_symbol(s, sn, snl)                 \
#define file_symbol(s, sn, snl)                 \
  (((s)->flags & BSF_FILE) != 0                  \
  (((s)->flags & BSF_FILE) != 0                  \
   || ((sn)[(snl) - 2] == '.'                   \
   || ((sn)[(snl) - 2] == '.'                   \
       && ((sn)[(snl) - 1] == 'o'               \
       && ((sn)[(snl) - 1] == 'o'               \
           || (sn)[(snl) - 1] == 'a')))
           || (sn)[(snl) - 1] == 'a')))
 
 
  af = file_symbol (a, an, anl);
  af = file_symbol (a, an, anl);
  bf = file_symbol (b, bn, bnl);
  bf = file_symbol (b, bn, bnl);
 
 
  if (af && ! bf)
  if (af && ! bf)
    return 1;
    return 1;
  if (! af && bf)
  if (! af && bf)
    return -1;
    return -1;
 
 
  /* Try to sort global symbols before local symbols before function
  /* Try to sort global symbols before local symbols before function
     symbols before debugging symbols.  */
     symbols before debugging symbols.  */
 
 
  aflags = a->flags;
  aflags = a->flags;
  bflags = b->flags;
  bflags = b->flags;
 
 
  if ((aflags & BSF_DEBUGGING) != (bflags & BSF_DEBUGGING))
  if ((aflags & BSF_DEBUGGING) != (bflags & BSF_DEBUGGING))
    {
    {
      if ((aflags & BSF_DEBUGGING) != 0)
      if ((aflags & BSF_DEBUGGING) != 0)
        return 1;
        return 1;
      else
      else
        return -1;
        return -1;
    }
    }
  if ((aflags & BSF_FUNCTION) != (bflags & BSF_FUNCTION))
  if ((aflags & BSF_FUNCTION) != (bflags & BSF_FUNCTION))
    {
    {
      if ((aflags & BSF_FUNCTION) != 0)
      if ((aflags & BSF_FUNCTION) != 0)
        return -1;
        return -1;
      else
      else
        return 1;
        return 1;
    }
    }
  if ((aflags & BSF_LOCAL) != (bflags & BSF_LOCAL))
  if ((aflags & BSF_LOCAL) != (bflags & BSF_LOCAL))
    {
    {
      if ((aflags & BSF_LOCAL) != 0)
      if ((aflags & BSF_LOCAL) != 0)
        return 1;
        return 1;
      else
      else
        return -1;
        return -1;
    }
    }
  if ((aflags & BSF_GLOBAL) != (bflags & BSF_GLOBAL))
  if ((aflags & BSF_GLOBAL) != (bflags & BSF_GLOBAL))
    {
    {
      if ((aflags & BSF_GLOBAL) != 0)
      if ((aflags & BSF_GLOBAL) != 0)
        return -1;
        return -1;
      else
      else
        return 1;
        return 1;
    }
    }
 
 
  /* Symbols that start with '.' might be section names, so sort them
  /* Symbols that start with '.' might be section names, so sort them
     after symbols that don't start with '.'.  */
     after symbols that don't start with '.'.  */
  if (an[0] == '.' && bn[0] != '.')
  if (an[0] == '.' && bn[0] != '.')
    return 1;
    return 1;
  if (an[0] != '.' && bn[0] == '.')
  if (an[0] != '.' && bn[0] == '.')
    return -1;
    return -1;
 
 
  /* Finally, if we can't distinguish them in any other way, try to
  /* Finally, if we can't distinguish them in any other way, try to
     get consistent results by sorting the symbols by name.  */
     get consistent results by sorting the symbols by name.  */
  return strcmp (an, bn);
  return strcmp (an, bn);
}
}
 
 
/* Sort relocs into address order.  */
/* Sort relocs into address order.  */
 
 
static int
static int
compare_relocs (const void *ap, const void *bp)
compare_relocs (const void *ap, const void *bp)
{
{
  const arelent *a = * (const arelent **) ap;
  const arelent *a = * (const arelent **) ap;
  const arelent *b = * (const arelent **) bp;
  const arelent *b = * (const arelent **) bp;
 
 
  if (a->address > b->address)
  if (a->address > b->address)
    return 1;
    return 1;
  else if (a->address < b->address)
  else if (a->address < b->address)
    return -1;
    return -1;
 
 
  /* So that associated relocations tied to the same address show up
  /* So that associated relocations tied to the same address show up
     in the correct order, we don't do any further sorting.  */
     in the correct order, we don't do any further sorting.  */
  if (a > b)
  if (a > b)
    return 1;
    return 1;
  else if (a < b)
  else if (a < b)
    return -1;
    return -1;
  else
  else
    return 0;
    return 0;
}
}
 
 
/* Print an address (VMA) to the output stream in INFO.
/* Print an address (VMA) to the output stream in INFO.
   If SKIP_ZEROES is TRUE, omit leading zeroes.  */
   If SKIP_ZEROES is TRUE, omit leading zeroes.  */
 
 
static void
static void
objdump_print_value (bfd_vma vma, struct disassemble_info *info,
objdump_print_value (bfd_vma vma, struct disassemble_info *info,
                     bfd_boolean skip_zeroes)
                     bfd_boolean skip_zeroes)
{
{
  char buf[30];
  char buf[30];
  char *p;
  char *p;
  struct objdump_disasm_info *aux;
  struct objdump_disasm_info *aux;
 
 
  aux = (struct objdump_disasm_info *) info->application_data;
  aux = (struct objdump_disasm_info *) info->application_data;
  bfd_sprintf_vma (aux->abfd, buf, vma);
  bfd_sprintf_vma (aux->abfd, buf, vma);
  if (! skip_zeroes)
  if (! skip_zeroes)
    p = buf;
    p = buf;
  else
  else
    {
    {
      for (p = buf; *p == '0'; ++p)
      for (p = buf; *p == '0'; ++p)
        ;
        ;
      if (*p == '\0')
      if (*p == '\0')
        --p;
        --p;
    }
    }
  (*info->fprintf_func) (info->stream, "%s", p);
  (*info->fprintf_func) (info->stream, "%s", p);
}
}
 
 
/* Print the name of a symbol.  */
/* Print the name of a symbol.  */
 
 
static void
static void
objdump_print_symname (bfd *abfd, struct disassemble_info *info,
objdump_print_symname (bfd *abfd, struct disassemble_info *info,
                       asymbol *sym)
                       asymbol *sym)
{
{
  char *alloc;
  char *alloc;
  const char *name;
  const char *name;
 
 
  alloc = NULL;
  alloc = NULL;
  name = bfd_asymbol_name (sym);
  name = bfd_asymbol_name (sym);
  if (do_demangle && name[0] != '\0')
  if (do_demangle && name[0] != '\0')
    {
    {
      /* Demangle the name.  */
      /* Demangle the name.  */
      alloc = bfd_demangle (abfd, name, DMGL_ANSI | DMGL_PARAMS);
      alloc = bfd_demangle (abfd, name, DMGL_ANSI | DMGL_PARAMS);
      if (alloc != NULL)
      if (alloc != NULL)
        name = alloc;
        name = alloc;
    }
    }
 
 
  if (info != NULL)
  if (info != NULL)
    (*info->fprintf_func) (info->stream, "%s", name);
    (*info->fprintf_func) (info->stream, "%s", name);
  else
  else
    printf ("%s", name);
    printf ("%s", name);
 
 
  if (alloc != NULL)
  if (alloc != NULL)
    free (alloc);
    free (alloc);
}
}
 
 
/* Locate a symbol given a bfd and a section (from INFO->application_data),
/* Locate a symbol given a bfd and a section (from INFO->application_data),
   and a VMA.  If INFO->application_data->require_sec is TRUE, then always
   and a VMA.  If INFO->application_data->require_sec is TRUE, then always
   require the symbol to be in the section.  Returns NULL if there is no
   require the symbol to be in the section.  Returns NULL if there is no
   suitable symbol.  If PLACE is not NULL, then *PLACE is set to the index
   suitable symbol.  If PLACE is not NULL, then *PLACE is set to the index
   of the symbol in sorted_syms.  */
   of the symbol in sorted_syms.  */
 
 
static asymbol *
static asymbol *
find_symbol_for_address (bfd_vma vma,
find_symbol_for_address (bfd_vma vma,
                         struct disassemble_info *info,
                         struct disassemble_info *info,
                         long *place)
                         long *place)
{
{
  /* @@ Would it speed things up to cache the last two symbols returned,
  /* @@ Would it speed things up to cache the last two symbols returned,
     and maybe their address ranges?  For many processors, only one memory
     and maybe their address ranges?  For many processors, only one memory
     operand can be present at a time, so the 2-entry cache wouldn't be
     operand can be present at a time, so the 2-entry cache wouldn't be
     constantly churned by code doing heavy memory accesses.  */
     constantly churned by code doing heavy memory accesses.  */
 
 
  /* Indices in `sorted_syms'.  */
  /* Indices in `sorted_syms'.  */
  long min = 0;
  long min = 0;
  long max = sorted_symcount;
  long max = sorted_symcount;
  long thisplace;
  long thisplace;
  struct objdump_disasm_info *aux;
  struct objdump_disasm_info *aux;
  bfd *abfd;
  bfd *abfd;
  asection *sec;
  asection *sec;
  unsigned int opb;
  unsigned int opb;
  bfd_boolean want_section;
  bfd_boolean want_section;
 
 
  if (sorted_symcount < 1)
  if (sorted_symcount < 1)
    return NULL;
    return NULL;
 
 
  aux = (struct objdump_disasm_info *) info->application_data;
  aux = (struct objdump_disasm_info *) info->application_data;
  abfd = aux->abfd;
  abfd = aux->abfd;
  sec = aux->sec;
  sec = aux->sec;
  opb = info->octets_per_byte;
  opb = info->octets_per_byte;
 
 
  /* Perform a binary search looking for the closest symbol to the
  /* Perform a binary search looking for the closest symbol to the
     required value.  We are searching the range (min, max].  */
     required value.  We are searching the range (min, max].  */
  while (min + 1 < max)
  while (min + 1 < max)
    {
    {
      asymbol *sym;
      asymbol *sym;
 
 
      thisplace = (max + min) / 2;
      thisplace = (max + min) / 2;
      sym = sorted_syms[thisplace];
      sym = sorted_syms[thisplace];
 
 
      if (bfd_asymbol_value (sym) > vma)
      if (bfd_asymbol_value (sym) > vma)
        max = thisplace;
        max = thisplace;
      else if (bfd_asymbol_value (sym) < vma)
      else if (bfd_asymbol_value (sym) < vma)
        min = thisplace;
        min = thisplace;
      else
      else
        {
        {
          min = thisplace;
          min = thisplace;
          break;
          break;
        }
        }
    }
    }
 
 
  /* The symbol we want is now in min, the low end of the range we
  /* The symbol we want is now in min, the low end of the range we
     were searching.  If there are several symbols with the same
     were searching.  If there are several symbols with the same
     value, we want the first one.  */
     value, we want the first one.  */
  thisplace = min;
  thisplace = min;
  while (thisplace > 0
  while (thisplace > 0
         && (bfd_asymbol_value (sorted_syms[thisplace])
         && (bfd_asymbol_value (sorted_syms[thisplace])
             == bfd_asymbol_value (sorted_syms[thisplace - 1])))
             == bfd_asymbol_value (sorted_syms[thisplace - 1])))
    --thisplace;
    --thisplace;
 
 
  /* Prefer a symbol in the current section if we have multple symbols
  /* Prefer a symbol in the current section if we have multple symbols
     with the same value, as can occur with overlays or zero size
     with the same value, as can occur with overlays or zero size
     sections.  */
     sections.  */
  min = thisplace;
  min = thisplace;
  while (min < max
  while (min < max
         && (bfd_asymbol_value (sorted_syms[min])
         && (bfd_asymbol_value (sorted_syms[min])
             == bfd_asymbol_value (sorted_syms[thisplace])))
             == bfd_asymbol_value (sorted_syms[thisplace])))
    {
    {
      if (sorted_syms[min]->section == sec
      if (sorted_syms[min]->section == sec
          && info->symbol_is_valid (sorted_syms[min], info))
          && info->symbol_is_valid (sorted_syms[min], info))
        {
        {
          thisplace = min;
          thisplace = min;
 
 
          if (place != NULL)
          if (place != NULL)
            *place = thisplace;
            *place = thisplace;
 
 
          return sorted_syms[thisplace];
          return sorted_syms[thisplace];
        }
        }
      ++min;
      ++min;
    }
    }
 
 
  /* If the file is relocatable, and the symbol could be from this
  /* If the file is relocatable, and the symbol could be from this
     section, prefer a symbol from this section over symbols from
     section, prefer a symbol from this section over symbols from
     others, even if the other symbol's value might be closer.
     others, even if the other symbol's value might be closer.
 
 
     Note that this may be wrong for some symbol references if the
     Note that this may be wrong for some symbol references if the
     sections have overlapping memory ranges, but in that case there's
     sections have overlapping memory ranges, but in that case there's
     no way to tell what's desired without looking at the relocation
     no way to tell what's desired without looking at the relocation
     table.
     table.
 
 
     Also give the target a chance to reject symbols.  */
     Also give the target a chance to reject symbols.  */
  want_section = (aux->require_sec
  want_section = (aux->require_sec
                  || ((abfd->flags & HAS_RELOC) != 0
                  || ((abfd->flags & HAS_RELOC) != 0
                      && vma >= bfd_get_section_vma (abfd, sec)
                      && vma >= bfd_get_section_vma (abfd, sec)
                      && vma < (bfd_get_section_vma (abfd, sec)
                      && vma < (bfd_get_section_vma (abfd, sec)
                                + bfd_section_size (abfd, sec) / opb)));
                                + bfd_section_size (abfd, sec) / opb)));
  if ((sorted_syms[thisplace]->section != sec && want_section)
  if ((sorted_syms[thisplace]->section != sec && want_section)
      || !info->symbol_is_valid (sorted_syms[thisplace], info))
      || !info->symbol_is_valid (sorted_syms[thisplace], info))
    {
    {
      long i;
      long i;
      long newplace = sorted_symcount;
      long newplace = sorted_symcount;
 
 
      for (i = min - 1; i >= 0; i--)
      for (i = min - 1; i >= 0; i--)
        {
        {
          if ((sorted_syms[i]->section == sec || !want_section)
          if ((sorted_syms[i]->section == sec || !want_section)
              && info->symbol_is_valid (sorted_syms[i], info))
              && info->symbol_is_valid (sorted_syms[i], info))
            {
            {
              if (newplace == sorted_symcount)
              if (newplace == sorted_symcount)
                newplace = i;
                newplace = i;
 
 
              if (bfd_asymbol_value (sorted_syms[i])
              if (bfd_asymbol_value (sorted_syms[i])
                  != bfd_asymbol_value (sorted_syms[newplace]))
                  != bfd_asymbol_value (sorted_syms[newplace]))
                break;
                break;
 
 
              /* Remember this symbol and keep searching until we reach
              /* Remember this symbol and keep searching until we reach
                 an earlier address.  */
                 an earlier address.  */
              newplace = i;
              newplace = i;
            }
            }
        }
        }
 
 
      if (newplace != sorted_symcount)
      if (newplace != sorted_symcount)
        thisplace = newplace;
        thisplace = newplace;
      else
      else
        {
        {
          /* We didn't find a good symbol with a smaller value.
          /* We didn't find a good symbol with a smaller value.
             Look for one with a larger value.  */
             Look for one with a larger value.  */
          for (i = thisplace + 1; i < sorted_symcount; i++)
          for (i = thisplace + 1; i < sorted_symcount; i++)
            {
            {
              if ((sorted_syms[i]->section == sec || !want_section)
              if ((sorted_syms[i]->section == sec || !want_section)
                  && info->symbol_is_valid (sorted_syms[i], info))
                  && info->symbol_is_valid (sorted_syms[i], info))
                {
                {
                  thisplace = i;
                  thisplace = i;
                  break;
                  break;
                }
                }
            }
            }
        }
        }
 
 
      if ((sorted_syms[thisplace]->section != sec && want_section)
      if ((sorted_syms[thisplace]->section != sec && want_section)
          || !info->symbol_is_valid (sorted_syms[thisplace], info))
          || !info->symbol_is_valid (sorted_syms[thisplace], info))
        /* There is no suitable symbol.  */
        /* There is no suitable symbol.  */
        return NULL;
        return NULL;
    }
    }
 
 
  if (place != NULL)
  if (place != NULL)
    *place = thisplace;
    *place = thisplace;
 
 
  return sorted_syms[thisplace];
  return sorted_syms[thisplace];
}
}
 
 
/* Print an address and the offset to the nearest symbol.  */
/* Print an address and the offset to the nearest symbol.  */
 
 
static void
static void
objdump_print_addr_with_sym (bfd *abfd, asection *sec, asymbol *sym,
objdump_print_addr_with_sym (bfd *abfd, asection *sec, asymbol *sym,
                             bfd_vma vma, struct disassemble_info *info,
                             bfd_vma vma, struct disassemble_info *info,
                             bfd_boolean skip_zeroes)
                             bfd_boolean skip_zeroes)
{
{
  objdump_print_value (vma, info, skip_zeroes);
  objdump_print_value (vma, info, skip_zeroes);
 
 
  if (sym == NULL)
  if (sym == NULL)
    {
    {
      bfd_vma secaddr;
      bfd_vma secaddr;
 
 
      (*info->fprintf_func) (info->stream, " <%s",
      (*info->fprintf_func) (info->stream, " <%s",
                             bfd_get_section_name (abfd, sec));
                             bfd_get_section_name (abfd, sec));
      secaddr = bfd_get_section_vma (abfd, sec);
      secaddr = bfd_get_section_vma (abfd, sec);
      if (vma < secaddr)
      if (vma < secaddr)
        {
        {
          (*info->fprintf_func) (info->stream, "-0x");
          (*info->fprintf_func) (info->stream, "-0x");
          objdump_print_value (secaddr - vma, info, TRUE);
          objdump_print_value (secaddr - vma, info, TRUE);
        }
        }
      else if (vma > secaddr)
      else if (vma > secaddr)
        {
        {
          (*info->fprintf_func) (info->stream, "+0x");
          (*info->fprintf_func) (info->stream, "+0x");
          objdump_print_value (vma - secaddr, info, TRUE);
          objdump_print_value (vma - secaddr, info, TRUE);
        }
        }
      (*info->fprintf_func) (info->stream, ">");
      (*info->fprintf_func) (info->stream, ">");
    }
    }
  else
  else
    {
    {
      (*info->fprintf_func) (info->stream, " <");
      (*info->fprintf_func) (info->stream, " <");
      objdump_print_symname (abfd, info, sym);
      objdump_print_symname (abfd, info, sym);
      if (bfd_asymbol_value (sym) > vma)
      if (bfd_asymbol_value (sym) > vma)
        {
        {
          (*info->fprintf_func) (info->stream, "-0x");
          (*info->fprintf_func) (info->stream, "-0x");
          objdump_print_value (bfd_asymbol_value (sym) - vma, info, TRUE);
          objdump_print_value (bfd_asymbol_value (sym) - vma, info, TRUE);
        }
        }
      else if (vma > bfd_asymbol_value (sym))
      else if (vma > bfd_asymbol_value (sym))
        {
        {
          (*info->fprintf_func) (info->stream, "+0x");
          (*info->fprintf_func) (info->stream, "+0x");
          objdump_print_value (vma - bfd_asymbol_value (sym), info, TRUE);
          objdump_print_value (vma - bfd_asymbol_value (sym), info, TRUE);
        }
        }
      (*info->fprintf_func) (info->stream, ">");
      (*info->fprintf_func) (info->stream, ">");
    }
    }
 
 
  if (display_file_offsets)
  if (display_file_offsets)
    info->fprintf_func (info->stream, _(" (File Offset: 0x%lx)"),
    info->fprintf_func (info->stream, _(" (File Offset: 0x%lx)"),
                        (long int)(sec->filepos + (vma - sec->vma)));
                        (long int)(sec->filepos + (vma - sec->vma)));
}
}
 
 
/* Print an address (VMA), symbolically if possible.
/* Print an address (VMA), symbolically if possible.
   If SKIP_ZEROES is TRUE, don't output leading zeroes.  */
   If SKIP_ZEROES is TRUE, don't output leading zeroes.  */
 
 
static void
static void
objdump_print_addr (bfd_vma vma,
objdump_print_addr (bfd_vma vma,
                    struct disassemble_info *info,
                    struct disassemble_info *info,
                    bfd_boolean skip_zeroes)
                    bfd_boolean skip_zeroes)
{
{
  struct objdump_disasm_info *aux;
  struct objdump_disasm_info *aux;
  asymbol *sym = NULL;
  asymbol *sym = NULL;
  bfd_boolean skip_find = FALSE;
  bfd_boolean skip_find = FALSE;
 
 
  aux = (struct objdump_disasm_info *) info->application_data;
  aux = (struct objdump_disasm_info *) info->application_data;
 
 
  if (sorted_symcount < 1)
  if (sorted_symcount < 1)
    {
    {
      (*info->fprintf_func) (info->stream, "0x");
      (*info->fprintf_func) (info->stream, "0x");
      objdump_print_value (vma, info, skip_zeroes);
      objdump_print_value (vma, info, skip_zeroes);
 
 
      if (display_file_offsets)
      if (display_file_offsets)
        info->fprintf_func (info->stream, _(" (File Offset: 0x%lx)"),
        info->fprintf_func (info->stream, _(" (File Offset: 0x%lx)"),
                            (long int)(aux->sec->filepos + (vma - aux->sec->vma)));
                            (long int)(aux->sec->filepos + (vma - aux->sec->vma)));
      return;
      return;
    }
    }
 
 
  if (aux->reloc != NULL
  if (aux->reloc != NULL
      && aux->reloc->sym_ptr_ptr != NULL
      && aux->reloc->sym_ptr_ptr != NULL
      && * aux->reloc->sym_ptr_ptr != NULL)
      && * aux->reloc->sym_ptr_ptr != NULL)
    {
    {
      sym = * aux->reloc->sym_ptr_ptr;
      sym = * aux->reloc->sym_ptr_ptr;
 
 
      /* Adjust the vma to the reloc.  */
      /* Adjust the vma to the reloc.  */
      vma += bfd_asymbol_value (sym);
      vma += bfd_asymbol_value (sym);
 
 
      if (bfd_is_und_section (bfd_get_section (sym)))
      if (bfd_is_und_section (bfd_get_section (sym)))
        skip_find = TRUE;
        skip_find = TRUE;
    }
    }
 
 
  if (!skip_find)
  if (!skip_find)
    sym = find_symbol_for_address (vma, info, NULL);
    sym = find_symbol_for_address (vma, info, NULL);
 
 
  objdump_print_addr_with_sym (aux->abfd, aux->sec, sym, vma, info,
  objdump_print_addr_with_sym (aux->abfd, aux->sec, sym, vma, info,
                               skip_zeroes);
                               skip_zeroes);
}
}
 
 
/* Print VMA to INFO.  This function is passed to the disassembler
/* Print VMA to INFO.  This function is passed to the disassembler
   routine.  */
   routine.  */
 
 
static void
static void
objdump_print_address (bfd_vma vma, struct disassemble_info *info)
objdump_print_address (bfd_vma vma, struct disassemble_info *info)
{
{
  objdump_print_addr (vma, info, ! prefix_addresses);
  objdump_print_addr (vma, info, ! prefix_addresses);
}
}
 
 
/* Determine if the given address has a symbol associated with it.  */
/* Determine if the given address has a symbol associated with it.  */
 
 
static int
static int
objdump_symbol_at_address (bfd_vma vma, struct disassemble_info * info)
objdump_symbol_at_address (bfd_vma vma, struct disassemble_info * info)
{
{
  asymbol * sym;
  asymbol * sym;
 
 
  sym = find_symbol_for_address (vma, info, NULL);
  sym = find_symbol_for_address (vma, info, NULL);
 
 
  return (sym != NULL && (bfd_asymbol_value (sym) == vma));
  return (sym != NULL && (bfd_asymbol_value (sym) == vma));
}
}
 
 
/* Hold the last function name and the last line number we displayed
/* Hold the last function name and the last line number we displayed
   in a disassembly.  */
   in a disassembly.  */
 
 
static char *prev_functionname;
static char *prev_functionname;
static unsigned int prev_line;
static unsigned int prev_line;
 
 
/* We keep a list of all files that we have seen when doing a
/* We keep a list of all files that we have seen when doing a
   disassembly with source, so that we know how much of the file to
   disassembly with source, so that we know how much of the file to
   display.  This can be important for inlined functions.  */
   display.  This can be important for inlined functions.  */
 
 
struct print_file_list
struct print_file_list
{
{
  struct print_file_list *next;
  struct print_file_list *next;
  const char *filename;
  const char *filename;
  const char *modname;
  const char *modname;
  const char *map;
  const char *map;
  size_t mapsize;
  size_t mapsize;
  const char **linemap;
  const char **linemap;
  unsigned maxline;
  unsigned maxline;
  unsigned last_line;
  unsigned last_line;
  int first;
  int first;
};
};
 
 
static struct print_file_list *print_files;
static struct print_file_list *print_files;
 
 
/* The number of preceding context lines to show when we start
/* The number of preceding context lines to show when we start
   displaying a file for the first time.  */
   displaying a file for the first time.  */
 
 
#define SHOW_PRECEDING_CONTEXT_LINES (5)
#define SHOW_PRECEDING_CONTEXT_LINES (5)
 
 
/* Read a complete file into memory.  */
/* Read a complete file into memory.  */
 
 
static const char *
static const char *
slurp_file (const char *fn, size_t *size)
slurp_file (const char *fn, size_t *size)
{
{
#ifdef HAVE_MMAP
#ifdef HAVE_MMAP
  int ps = getpagesize ();
  int ps = getpagesize ();
  size_t msize;
  size_t msize;
#endif
#endif
  const char *map;
  const char *map;
  struct stat st;
  struct stat st;
  int fd = open (fn, O_RDONLY | O_BINARY);
  int fd = open (fn, O_RDONLY | O_BINARY);
 
 
  if (fd < 0)
  if (fd < 0)
    return NULL;
    return NULL;
  if (fstat (fd, &st) < 0)
  if (fstat (fd, &st) < 0)
    return NULL;
    return NULL;
  *size = st.st_size;
  *size = st.st_size;
#ifdef HAVE_MMAP
#ifdef HAVE_MMAP
  msize = (*size + ps - 1) & ~(ps - 1);
  msize = (*size + ps - 1) & ~(ps - 1);
  map = mmap (NULL, msize, PROT_READ, MAP_SHARED, fd, 0);
  map = mmap (NULL, msize, PROT_READ, MAP_SHARED, fd, 0);
  if (map != (char *)-1L)
  if (map != (char *)-1L)
    {
    {
      close(fd);
      close(fd);
      return map;
      return map;
    }
    }
#endif
#endif
  map = (const char *) malloc (*size);
  map = (const char *) malloc (*size);
  if (!map || (size_t) read (fd, (char *)map, *size) != *size)
  if (!map || (size_t) read (fd, (char *)map, *size) != *size)
    {
    {
      free ((void *)map);
      free ((void *)map);
      map = NULL;
      map = NULL;
    }
    }
  close (fd);
  close (fd);
  return map;
  return map;
}
}
 
 
#define line_map_decrease 5
#define line_map_decrease 5
 
 
/* Precompute array of lines for a mapped file. */
/* Precompute array of lines for a mapped file. */
 
 
static const char **
static const char **
index_file (const char *map, size_t size, unsigned int *maxline)
index_file (const char *map, size_t size, unsigned int *maxline)
{
{
  const char *p, *lstart, *end;
  const char *p, *lstart, *end;
  int chars_per_line = 45; /* First iteration will use 40.  */
  int chars_per_line = 45; /* First iteration will use 40.  */
  unsigned int lineno;
  unsigned int lineno;
  const char **linemap = NULL;
  const char **linemap = NULL;
  unsigned long line_map_size = 0;
  unsigned long line_map_size = 0;
 
 
  lineno = 0;
  lineno = 0;
  lstart = map;
  lstart = map;
  end = map + size;
  end = map + size;
 
 
  for (p = map; p < end; p++)
  for (p = map; p < end; p++)
    {
    {
      if (*p == '\n')
      if (*p == '\n')
        {
        {
          if (p + 1 < end && p[1] == '\r')
          if (p + 1 < end && p[1] == '\r')
            p++;
            p++;
        }
        }
      else if (*p == '\r')
      else if (*p == '\r')
        {
        {
          if (p + 1 < end && p[1] == '\n')
          if (p + 1 < end && p[1] == '\n')
            p++;
            p++;
        }
        }
      else
      else
        continue;
        continue;
 
 
      /* End of line found.  */
      /* End of line found.  */
 
 
      if (linemap == NULL || line_map_size < lineno + 1)
      if (linemap == NULL || line_map_size < lineno + 1)
        {
        {
          unsigned long newsize;
          unsigned long newsize;
 
 
          chars_per_line -= line_map_decrease;
          chars_per_line -= line_map_decrease;
          if (chars_per_line <= 1)
          if (chars_per_line <= 1)
            chars_per_line = 1;
            chars_per_line = 1;
          line_map_size = size / chars_per_line + 1;
          line_map_size = size / chars_per_line + 1;
          if (line_map_size < lineno + 1)
          if (line_map_size < lineno + 1)
            line_map_size = lineno + 1;
            line_map_size = lineno + 1;
          newsize = line_map_size * sizeof (char *);
          newsize = line_map_size * sizeof (char *);
          linemap = (const char **) xrealloc (linemap, newsize);
          linemap = (const char **) xrealloc (linemap, newsize);
        }
        }
 
 
      linemap[lineno++] = lstart;
      linemap[lineno++] = lstart;
      lstart = p + 1;
      lstart = p + 1;
    }
    }
 
 
  *maxline = lineno;
  *maxline = lineno;
  return linemap;
  return linemap;
}
}
 
 
/* Tries to open MODNAME, and if successful adds a node to print_files
/* Tries to open MODNAME, and if successful adds a node to print_files
   linked list and returns that node.  Returns NULL on failure.  */
   linked list and returns that node.  Returns NULL on failure.  */
 
 
static struct print_file_list *
static struct print_file_list *
try_print_file_open (const char *origname, const char *modname)
try_print_file_open (const char *origname, const char *modname)
{
{
  struct print_file_list *p;
  struct print_file_list *p;
 
 
  p = (struct print_file_list *) xmalloc (sizeof (struct print_file_list));
  p = (struct print_file_list *) xmalloc (sizeof (struct print_file_list));
 
 
  p->map = slurp_file (modname, &p->mapsize);
  p->map = slurp_file (modname, &p->mapsize);
  if (p->map == NULL)
  if (p->map == NULL)
    {
    {
      free (p);
      free (p);
      return NULL;
      return NULL;
    }
    }
 
 
  p->linemap = index_file (p->map, p->mapsize, &p->maxline);
  p->linemap = index_file (p->map, p->mapsize, &p->maxline);
  p->last_line = 0;
  p->last_line = 0;
  p->filename = origname;
  p->filename = origname;
  p->modname = modname;
  p->modname = modname;
  p->next = print_files;
  p->next = print_files;
  p->first = 1;
  p->first = 1;
  print_files = p;
  print_files = p;
  return p;
  return p;
}
}
 
 
/* If the the source file, as described in the symtab, is not found
/* If the the source file, as described in the symtab, is not found
   try to locate it in one of the paths specified with -I
   try to locate it in one of the paths specified with -I
   If found, add location to print_files linked list.  */
   If found, add location to print_files linked list.  */
 
 
static struct print_file_list *
static struct print_file_list *
update_source_path (const char *filename)
update_source_path (const char *filename)
{
{
  struct print_file_list *p;
  struct print_file_list *p;
  const char *fname;
  const char *fname;
  int i;
  int i;
 
 
  if (filename == NULL)
  if (filename == NULL)
    return NULL;
    return NULL;
 
 
  p = try_print_file_open (filename, filename);
  p = try_print_file_open (filename, filename);
  if (p != NULL)
  if (p != NULL)
    return p;
    return p;
 
 
  if (include_path_count == 0)
  if (include_path_count == 0)
    return NULL;
    return NULL;
 
 
  /* Get the name of the file.  */
  /* Get the name of the file.  */
  fname = strrchr (filename, '/');
  fname = strrchr (filename, '/');
#ifdef HAVE_DOS_BASED_FILE_SYSTEM
#ifdef HAVE_DOS_BASED_FILE_SYSTEM
  {
  {
    /* We could have a mixed forward/back slash case.  */
    /* We could have a mixed forward/back slash case.  */
    char *backslash = strrchr (filename, '\\');
    char *backslash = strrchr (filename, '\\');
    if (fname == NULL || (backslash != NULL && backslash > fname))
    if (fname == NULL || (backslash != NULL && backslash > fname))
      fname = backslash;
      fname = backslash;
    if (fname == NULL && filename[0] != '\0' && filename[1] == ':')
    if (fname == NULL && filename[0] != '\0' && filename[1] == ':')
      fname = filename + 1;
      fname = filename + 1;
  }
  }
#endif
#endif
  if (fname == NULL)
  if (fname == NULL)
    fname = filename;
    fname = filename;
  else
  else
    ++fname;
    ++fname;
 
 
  /* If file exists under a new path, we need to add it to the list
  /* If file exists under a new path, we need to add it to the list
     so that show_line knows about it.  */
     so that show_line knows about it.  */
  for (i = 0; i < include_path_count; i++)
  for (i = 0; i < include_path_count; i++)
    {
    {
      char *modname = concat (include_paths[i], "/", fname, (const char *) 0);
      char *modname = concat (include_paths[i], "/", fname, (const char *) 0);
 
 
      p = try_print_file_open (filename, modname);
      p = try_print_file_open (filename, modname);
      if (p)
      if (p)
        return p;
        return p;
 
 
      free (modname);
      free (modname);
    }
    }
 
 
  return NULL;
  return NULL;
}
}
 
 
/* Print a source file line.  */
/* Print a source file line.  */
 
 
static void
static void
print_line (struct print_file_list *p, unsigned int line)
print_line (struct print_file_list *p, unsigned int line)
{
{
  const char *l;
  const char *l;
  size_t len;
  size_t len;
 
 
  --line;
  --line;
  if (line >= p->maxline)
  if (line >= p->maxline)
    return;
    return;
  l = p->linemap [line];
  l = p->linemap [line];
  /* Test fwrite return value to quiet glibc warning.  */
  /* Test fwrite return value to quiet glibc warning.  */
  len = strcspn (l, "\n\r");
  len = strcspn (l, "\n\r");
  if (len == 0 || fwrite (l, len, 1, stdout) == 1)
  if (len == 0 || fwrite (l, len, 1, stdout) == 1)
    putchar ('\n');
    putchar ('\n');
}
}
 
 
/* Print a range of source code lines. */
/* Print a range of source code lines. */
 
 
static void
static void
dump_lines (struct print_file_list *p, unsigned int start, unsigned int end)
dump_lines (struct print_file_list *p, unsigned int start, unsigned int end)
{
{
  if (p->map == NULL)
  if (p->map == NULL)
    return;
    return;
  while (start <= end)
  while (start <= end)
    {
    {
      print_line (p, start);
      print_line (p, start);
      start++;
      start++;
    }
    }
}
}
 
 
/* Show the line number, or the source line, in a disassembly
/* Show the line number, or the source line, in a disassembly
   listing.  */
   listing.  */
 
 
static void
static void
show_line (bfd *abfd, asection *section, bfd_vma addr_offset)
show_line (bfd *abfd, asection *section, bfd_vma addr_offset)
{
{
  const char *filename;
  const char *filename;
  const char *functionname;
  const char *functionname;
  unsigned int line;
  unsigned int line;
  bfd_boolean reloc;
  bfd_boolean reloc;
 
 
  if (! with_line_numbers && ! with_source_code)
  if (! with_line_numbers && ! with_source_code)
    return;
    return;
 
 
  if (! bfd_find_nearest_line (abfd, section, syms, addr_offset, &filename,
  if (! bfd_find_nearest_line (abfd, section, syms, addr_offset, &filename,
                               &functionname, &line))
                               &functionname, &line))
    return;
    return;
 
 
  if (filename != NULL && *filename == '\0')
  if (filename != NULL && *filename == '\0')
    filename = NULL;
    filename = NULL;
  if (functionname != NULL && *functionname == '\0')
  if (functionname != NULL && *functionname == '\0')
    functionname = NULL;
    functionname = NULL;
 
 
  if (filename
  if (filename
      && IS_ABSOLUTE_PATH (filename)
      && IS_ABSOLUTE_PATH (filename)
      && prefix)
      && prefix)
    {
    {
      char *path_up;
      char *path_up;
      const char *fname = filename;
      const char *fname = filename;
      char *path = (char *) alloca (prefix_length + PATH_MAX + 1);
      char *path = (char *) alloca (prefix_length + PATH_MAX + 1);
 
 
      if (prefix_length)
      if (prefix_length)
        memcpy (path, prefix, prefix_length);
        memcpy (path, prefix, prefix_length);
      path_up = path + prefix_length;
      path_up = path + prefix_length;
 
 
      /* Build relocated filename, stripping off leading directories
      /* Build relocated filename, stripping off leading directories
         from the initial filename if requested. */
         from the initial filename if requested. */
      if (prefix_strip > 0)
      if (prefix_strip > 0)
        {
        {
          int level = 0;
          int level = 0;
          const char *s;
          const char *s;
 
 
          /* Skip selected directory levels. */
          /* Skip selected directory levels. */
          for (s = fname + 1; *s != '\0' && level < prefix_strip; s++)
          for (s = fname + 1; *s != '\0' && level < prefix_strip; s++)
            if (IS_DIR_SEPARATOR(*s))
            if (IS_DIR_SEPARATOR(*s))
              {
              {
                fname = s;
                fname = s;
                level++;
                level++;
              }
              }
        }
        }
 
 
      /* Update complete filename. */
      /* Update complete filename. */
      strncpy (path_up, fname, PATH_MAX);
      strncpy (path_up, fname, PATH_MAX);
      path_up[PATH_MAX] = '\0';
      path_up[PATH_MAX] = '\0';
 
 
      filename = path;
      filename = path;
      reloc = TRUE;
      reloc = TRUE;
    }
    }
  else
  else
    reloc = FALSE;
    reloc = FALSE;
 
 
  if (with_line_numbers)
  if (with_line_numbers)
    {
    {
      if (functionname != NULL
      if (functionname != NULL
          && (prev_functionname == NULL
          && (prev_functionname == NULL
              || strcmp (functionname, prev_functionname) != 0))
              || strcmp (functionname, prev_functionname) != 0))
        printf ("%s():\n", functionname);
        printf ("%s():\n", functionname);
      if (line > 0 && line != prev_line)
      if (line > 0 && line != prev_line)
        printf ("%s:%u\n", filename == NULL ? "???" : filename, line);
        printf ("%s:%u\n", filename == NULL ? "???" : filename, line);
    }
    }
 
 
  if (with_source_code
  if (with_source_code
      && filename != NULL
      && filename != NULL
      && line > 0)
      && line > 0)
    {
    {
      struct print_file_list **pp, *p;
      struct print_file_list **pp, *p;
      unsigned l;
      unsigned l;
 
 
      for (pp = &print_files; *pp != NULL; pp = &(*pp)->next)
      for (pp = &print_files; *pp != NULL; pp = &(*pp)->next)
        if (strcmp ((*pp)->filename, filename) == 0)
        if (strcmp ((*pp)->filename, filename) == 0)
          break;
          break;
      p = *pp;
      p = *pp;
 
 
      if (p == NULL)
      if (p == NULL)
        {
        {
          if (reloc)
          if (reloc)
            filename = xstrdup (filename);
            filename = xstrdup (filename);
          p = update_source_path (filename);
          p = update_source_path (filename);
        }
        }
 
 
      if (p != NULL && line != p->last_line)
      if (p != NULL && line != p->last_line)
        {
        {
          if (file_start_context && p->first)
          if (file_start_context && p->first)
            l = 1;
            l = 1;
          else
          else
            {
            {
              l = line - SHOW_PRECEDING_CONTEXT_LINES;
              l = line - SHOW_PRECEDING_CONTEXT_LINES;
              if (l >= line)
              if (l >= line)
                l = 1;
                l = 1;
              if (p->last_line >= l && p->last_line <= line)
              if (p->last_line >= l && p->last_line <= line)
                l = p->last_line + 1;
                l = p->last_line + 1;
            }
            }
          dump_lines (p, l, line);
          dump_lines (p, l, line);
          p->last_line = line;
          p->last_line = line;
          p->first = 0;
          p->first = 0;
        }
        }
    }
    }
 
 
  if (functionname != NULL
  if (functionname != NULL
      && (prev_functionname == NULL
      && (prev_functionname == NULL
          || strcmp (functionname, prev_functionname) != 0))
          || strcmp (functionname, prev_functionname) != 0))
    {
    {
      if (prev_functionname != NULL)
      if (prev_functionname != NULL)
        free (prev_functionname);
        free (prev_functionname);
      prev_functionname = (char *) xmalloc (strlen (functionname) + 1);
      prev_functionname = (char *) xmalloc (strlen (functionname) + 1);
      strcpy (prev_functionname, functionname);
      strcpy (prev_functionname, functionname);
    }
    }
 
 
  if (line > 0 && line != prev_line)
  if (line > 0 && line != prev_line)
    prev_line = line;
    prev_line = line;
}
}
 
 
/* Pseudo FILE object for strings.  */
/* Pseudo FILE object for strings.  */
typedef struct
typedef struct
{
{
  char *buffer;
  char *buffer;
  size_t pos;
  size_t pos;
  size_t alloc;
  size_t alloc;
} SFILE;
} SFILE;
 
 
/* sprintf to a "stream".  */
/* sprintf to a "stream".  */
 
 
static int ATTRIBUTE_PRINTF_2
static int ATTRIBUTE_PRINTF_2
objdump_sprintf (SFILE *f, const char *format, ...)
objdump_sprintf (SFILE *f, const char *format, ...)
{
{
  size_t n;
  size_t n;
  va_list args;
  va_list args;
 
 
  while (1)
  while (1)
    {
    {
      size_t space = f->alloc - f->pos;
      size_t space = f->alloc - f->pos;
 
 
      va_start (args, format);
      va_start (args, format);
      n = vsnprintf (f->buffer + f->pos, space, format, args);
      n = vsnprintf (f->buffer + f->pos, space, format, args);
      va_end (args);
      va_end (args);
 
 
      if (space > n)
      if (space > n)
        break;
        break;
 
 
      f->alloc = (f->alloc + n) * 2;
      f->alloc = (f->alloc + n) * 2;
      f->buffer = (char *) xrealloc (f->buffer, f->alloc);
      f->buffer = (char *) xrealloc (f->buffer, f->alloc);
    }
    }
  f->pos += n;
  f->pos += n;
 
 
  return n;
  return n;
}
}
 
 
/* The number of zeroes we want to see before we start skipping them.
/* The number of zeroes we want to see before we start skipping them.
   The number is arbitrarily chosen.  */
   The number is arbitrarily chosen.  */
 
 
#define DEFAULT_SKIP_ZEROES 8
#define DEFAULT_SKIP_ZEROES 8
 
 
/* The number of zeroes to skip at the end of a section.  If the
/* The number of zeroes to skip at the end of a section.  If the
   number of zeroes at the end is between SKIP_ZEROES_AT_END and
   number of zeroes at the end is between SKIP_ZEROES_AT_END and
   SKIP_ZEROES, they will be disassembled.  If there are fewer than
   SKIP_ZEROES, they will be disassembled.  If there are fewer than
   SKIP_ZEROES_AT_END, they will be skipped.  This is a heuristic
   SKIP_ZEROES_AT_END, they will be skipped.  This is a heuristic
   attempt to avoid disassembling zeroes inserted by section
   attempt to avoid disassembling zeroes inserted by section
   alignment.  */
   alignment.  */
 
 
#define DEFAULT_SKIP_ZEROES_AT_END 3
#define DEFAULT_SKIP_ZEROES_AT_END 3
 
 
/* Disassemble some data in memory between given values.  */
/* Disassemble some data in memory between given values.  */
 
 
static void
static void
disassemble_bytes (struct disassemble_info * info,
disassemble_bytes (struct disassemble_info * info,
                   disassembler_ftype        disassemble_fn,
                   disassembler_ftype        disassemble_fn,
                   bfd_boolean               insns,
                   bfd_boolean               insns,
                   bfd_byte *                data,
                   bfd_byte *                data,
                   bfd_vma                   start_offset,
                   bfd_vma                   start_offset,
                   bfd_vma                   stop_offset,
                   bfd_vma                   stop_offset,
                   bfd_vma                   rel_offset,
                   bfd_vma                   rel_offset,
                   arelent ***               relppp,
                   arelent ***               relppp,
                   arelent **                relppend)
                   arelent **                relppend)
{
{
  struct objdump_disasm_info *aux;
  struct objdump_disasm_info *aux;
  asection *section;
  asection *section;
  int octets_per_line;
  int octets_per_line;
  int skip_addr_chars;
  int skip_addr_chars;
  bfd_vma addr_offset;
  bfd_vma addr_offset;
  unsigned int opb = info->octets_per_byte;
  unsigned int opb = info->octets_per_byte;
  unsigned int skip_zeroes = info->skip_zeroes;
  unsigned int skip_zeroes = info->skip_zeroes;
  unsigned int skip_zeroes_at_end = info->skip_zeroes_at_end;
  unsigned int skip_zeroes_at_end = info->skip_zeroes_at_end;
  int octets = opb;
  int octets = opb;
  SFILE sfile;
  SFILE sfile;
 
 
  aux = (struct objdump_disasm_info *) info->application_data;
  aux = (struct objdump_disasm_info *) info->application_data;
  section = aux->sec;
  section = aux->sec;
 
 
  sfile.alloc = 120;
  sfile.alloc = 120;
  sfile.buffer = (char *) xmalloc (sfile.alloc);
  sfile.buffer = (char *) xmalloc (sfile.alloc);
  sfile.pos = 0;
  sfile.pos = 0;
 
 
  if (insn_width)
  if (insn_width)
    octets_per_line = insn_width;
    octets_per_line = insn_width;
  else if (insns)
  else if (insns)
    octets_per_line = 4;
    octets_per_line = 4;
  else
  else
    octets_per_line = 16;
    octets_per_line = 16;
 
 
  /* Figure out how many characters to skip at the start of an
  /* Figure out how many characters to skip at the start of an
     address, to make the disassembly look nicer.  We discard leading
     address, to make the disassembly look nicer.  We discard leading
     zeroes in chunks of 4, ensuring that there is always a leading
     zeroes in chunks of 4, ensuring that there is always a leading
     zero remaining.  */
     zero remaining.  */
  skip_addr_chars = 0;
  skip_addr_chars = 0;
  if (! prefix_addresses)
  if (! prefix_addresses)
    {
    {
      char buf[30];
      char buf[30];
 
 
      bfd_sprintf_vma (aux->abfd, buf, section->vma + section->size / opb);
      bfd_sprintf_vma (aux->abfd, buf, section->vma + section->size / opb);
 
 
      while (buf[skip_addr_chars] == '0')
      while (buf[skip_addr_chars] == '0')
        ++skip_addr_chars;
        ++skip_addr_chars;
 
 
      /* Don't discard zeros on overflow.  */
      /* Don't discard zeros on overflow.  */
      if (buf[skip_addr_chars] == '\0' && section->vma != 0)
      if (buf[skip_addr_chars] == '\0' && section->vma != 0)
        skip_addr_chars = 0;
        skip_addr_chars = 0;
 
 
      if (skip_addr_chars != 0)
      if (skip_addr_chars != 0)
        skip_addr_chars = (skip_addr_chars - 1) & -4;
        skip_addr_chars = (skip_addr_chars - 1) & -4;
    }
    }
 
 
  info->insn_info_valid = 0;
  info->insn_info_valid = 0;
 
 
  addr_offset = start_offset;
  addr_offset = start_offset;
  while (addr_offset < stop_offset)
  while (addr_offset < stop_offset)
    {
    {
      bfd_vma z;
      bfd_vma z;
      bfd_boolean need_nl = FALSE;
      bfd_boolean need_nl = FALSE;
      int previous_octets;
      int previous_octets;
 
 
      /* Remember the length of the previous instruction.  */
      /* Remember the length of the previous instruction.  */
      previous_octets = octets;
      previous_octets = octets;
      octets = 0;
      octets = 0;
 
 
      /* Make sure we don't use relocs from previous instructions.  */
      /* Make sure we don't use relocs from previous instructions.  */
      aux->reloc = NULL;
      aux->reloc = NULL;
 
 
      /* If we see more than SKIP_ZEROES octets of zeroes, we just
      /* If we see more than SKIP_ZEROES octets of zeroes, we just
         print `...'.  */
         print `...'.  */
      for (z = addr_offset * opb; z < stop_offset * opb; z++)
      for (z = addr_offset * opb; z < stop_offset * opb; z++)
        if (data[z] != 0)
        if (data[z] != 0)
          break;
          break;
      if (! disassemble_zeroes
      if (! disassemble_zeroes
          && (info->insn_info_valid == 0
          && (info->insn_info_valid == 0
              || info->branch_delay_insns == 0)
              || info->branch_delay_insns == 0)
          && (z - addr_offset * opb >= skip_zeroes
          && (z - addr_offset * opb >= skip_zeroes
              || (z == stop_offset * opb &&
              || (z == stop_offset * opb &&
                  z - addr_offset * opb < skip_zeroes_at_end)))
                  z - addr_offset * opb < skip_zeroes_at_end)))
        {
        {
          /* If there are more nonzero octets to follow, we only skip
          /* If there are more nonzero octets to follow, we only skip
             zeroes in multiples of 4, to try to avoid running over
             zeroes in multiples of 4, to try to avoid running over
             the start of an instruction which happens to start with
             the start of an instruction which happens to start with
             zero.  */
             zero.  */
          if (z != stop_offset * opb)
          if (z != stop_offset * opb)
            z = addr_offset * opb + ((z - addr_offset * opb) &~ 3);
            z = addr_offset * opb + ((z - addr_offset * opb) &~ 3);
 
 
          octets = z - addr_offset * opb;
          octets = z - addr_offset * opb;
 
 
          /* If we are going to display more data, and we are displaying
          /* If we are going to display more data, and we are displaying
             file offsets, then tell the user how many zeroes we skip
             file offsets, then tell the user how many zeroes we skip
             and the file offset from where we resume dumping.  */
             and the file offset from where we resume dumping.  */
          if (display_file_offsets && ((addr_offset + (octets / opb)) < stop_offset))
          if (display_file_offsets && ((addr_offset + (octets / opb)) < stop_offset))
            printf ("\t... (skipping %d zeroes, resuming at file offset: 0x%lx)\n",
            printf ("\t... (skipping %d zeroes, resuming at file offset: 0x%lx)\n",
                    octets / opb,
                    octets / opb,
                    (unsigned long) (section->filepos
                    (unsigned long) (section->filepos
                                     + (addr_offset + (octets / opb))));
                                     + (addr_offset + (octets / opb))));
          else
          else
            printf ("\t...\n");
            printf ("\t...\n");
        }
        }
      else
      else
        {
        {
          char buf[50];
          char buf[50];
          int bpc = 0;
          int bpc = 0;
          int pb = 0;
          int pb = 0;
 
 
          if (with_line_numbers || with_source_code)
          if (with_line_numbers || with_source_code)
            show_line (aux->abfd, section, addr_offset);
            show_line (aux->abfd, section, addr_offset);
 
 
          if (! prefix_addresses)
          if (! prefix_addresses)
            {
            {
              char *s;
              char *s;
 
 
              bfd_sprintf_vma (aux->abfd, buf, section->vma + addr_offset);
              bfd_sprintf_vma (aux->abfd, buf, section->vma + addr_offset);
              for (s = buf + skip_addr_chars; *s == '0'; s++)
              for (s = buf + skip_addr_chars; *s == '0'; s++)
                *s = ' ';
                *s = ' ';
              if (*s == '\0')
              if (*s == '\0')
                *--s = '0';
                *--s = '0';
              printf ("%s:\t", buf + skip_addr_chars);
              printf ("%s:\t", buf + skip_addr_chars);
            }
            }
          else
          else
            {
            {
              aux->require_sec = TRUE;
              aux->require_sec = TRUE;
              objdump_print_address (section->vma + addr_offset, info);
              objdump_print_address (section->vma + addr_offset, info);
              aux->require_sec = FALSE;
              aux->require_sec = FALSE;
              putchar (' ');
              putchar (' ');
            }
            }
 
 
          if (insns)
          if (insns)
            {
            {
              sfile.pos = 0;
              sfile.pos = 0;
              info->fprintf_func = (fprintf_ftype) objdump_sprintf;
              info->fprintf_func = (fprintf_ftype) objdump_sprintf;
              info->stream = &sfile;
              info->stream = &sfile;
              info->bytes_per_line = 0;
              info->bytes_per_line = 0;
              info->bytes_per_chunk = 0;
              info->bytes_per_chunk = 0;
              info->flags = disassemble_all ? DISASSEMBLE_DATA : 0;
              info->flags = disassemble_all ? DISASSEMBLE_DATA : 0;
              if (machine)
              if (machine)
                info->flags |= USER_SPECIFIED_MACHINE_TYPE;
                info->flags |= USER_SPECIFIED_MACHINE_TYPE;
 
 
              if (info->disassembler_needs_relocs
              if (info->disassembler_needs_relocs
                  && (bfd_get_file_flags (aux->abfd) & EXEC_P) == 0
                  && (bfd_get_file_flags (aux->abfd) & EXEC_P) == 0
                  && (bfd_get_file_flags (aux->abfd) & DYNAMIC) == 0
                  && (bfd_get_file_flags (aux->abfd) & DYNAMIC) == 0
                  && *relppp < relppend)
                  && *relppp < relppend)
                {
                {
                  bfd_signed_vma distance_to_rel;
                  bfd_signed_vma distance_to_rel;
 
 
                  distance_to_rel = (**relppp)->address
                  distance_to_rel = (**relppp)->address
                    - (rel_offset + addr_offset);
                    - (rel_offset + addr_offset);
 
 
                  /* Check to see if the current reloc is associated with
                  /* Check to see if the current reloc is associated with
                     the instruction that we are about to disassemble.  */
                     the instruction that we are about to disassemble.  */
                  if (distance_to_rel == 0
                  if (distance_to_rel == 0
                      /* FIXME: This is wrong.  We are trying to catch
                      /* FIXME: This is wrong.  We are trying to catch
                         relocs that are addressed part way through the
                         relocs that are addressed part way through the
                         current instruction, as might happen with a packed
                         current instruction, as might happen with a packed
                         VLIW instruction.  Unfortunately we do not know the
                         VLIW instruction.  Unfortunately we do not know the
                         length of the current instruction since we have not
                         length of the current instruction since we have not
                         disassembled it yet.  Instead we take a guess based
                         disassembled it yet.  Instead we take a guess based
                         upon the length of the previous instruction.  The
                         upon the length of the previous instruction.  The
                         proper solution is to have a new target-specific
                         proper solution is to have a new target-specific
                         disassembler function which just returns the length
                         disassembler function which just returns the length
                         of an instruction at a given address without trying
                         of an instruction at a given address without trying
                         to display its disassembly. */
                         to display its disassembly. */
                      || (distance_to_rel > 0
                      || (distance_to_rel > 0
                          && distance_to_rel < (bfd_signed_vma) (previous_octets/ opb)))
                          && distance_to_rel < (bfd_signed_vma) (previous_octets/ opb)))
                    {
                    {
                      info->flags |= INSN_HAS_RELOC;
                      info->flags |= INSN_HAS_RELOC;
                      aux->reloc = **relppp;
                      aux->reloc = **relppp;
                    }
                    }
                }
                }
 
 
              octets = (*disassemble_fn) (section->vma + addr_offset, info);
              octets = (*disassemble_fn) (section->vma + addr_offset, info);
              info->fprintf_func = (fprintf_ftype) fprintf;
              info->fprintf_func = (fprintf_ftype) fprintf;
              info->stream = stdout;
              info->stream = stdout;
              if (insn_width == 0 && info->bytes_per_line != 0)
              if (insn_width == 0 && info->bytes_per_line != 0)
                octets_per_line = info->bytes_per_line;
                octets_per_line = info->bytes_per_line;
              if (octets < 0)
              if (octets < 0)
                {
                {
                  if (sfile.pos)
                  if (sfile.pos)
                    printf ("%s\n", sfile.buffer);
                    printf ("%s\n", sfile.buffer);
                  break;
                  break;
                }
                }
            }
            }
          else
          else
            {
            {
              bfd_vma j;
              bfd_vma j;
 
 
              octets = octets_per_line;
              octets = octets_per_line;
              if (addr_offset + octets / opb > stop_offset)
              if (addr_offset + octets / opb > stop_offset)
                octets = (stop_offset - addr_offset) * opb;
                octets = (stop_offset - addr_offset) * opb;
 
 
              for (j = addr_offset * opb; j < addr_offset * opb + octets; ++j)
              for (j = addr_offset * opb; j < addr_offset * opb + octets; ++j)
                {
                {
                  if (ISPRINT (data[j]))
                  if (ISPRINT (data[j]))
                    buf[j - addr_offset * opb] = data[j];
                    buf[j - addr_offset * opb] = data[j];
                  else
                  else
                    buf[j - addr_offset * opb] = '.';
                    buf[j - addr_offset * opb] = '.';
                }
                }
              buf[j - addr_offset * opb] = '\0';
              buf[j - addr_offset * opb] = '\0';
            }
            }
 
 
          if (prefix_addresses
          if (prefix_addresses
              ? show_raw_insn > 0
              ? show_raw_insn > 0
              : show_raw_insn >= 0)
              : show_raw_insn >= 0)
            {
            {
              bfd_vma j;
              bfd_vma j;
 
 
              /* If ! prefix_addresses and ! wide_output, we print
              /* If ! prefix_addresses and ! wide_output, we print
                 octets_per_line octets per line.  */
                 octets_per_line octets per line.  */
              pb = octets;
              pb = octets;
              if (pb > octets_per_line && ! prefix_addresses && ! wide_output)
              if (pb > octets_per_line && ! prefix_addresses && ! wide_output)
                pb = octets_per_line;
                pb = octets_per_line;
 
 
              if (info->bytes_per_chunk)
              if (info->bytes_per_chunk)
                bpc = info->bytes_per_chunk;
                bpc = info->bytes_per_chunk;
              else
              else
                bpc = 1;
                bpc = 1;
 
 
              for (j = addr_offset * opb; j < addr_offset * opb + pb; j += bpc)
              for (j = addr_offset * opb; j < addr_offset * opb + pb; j += bpc)
                {
                {
                  int k;
                  int k;
 
 
                  if (bpc > 1 && info->display_endian == BFD_ENDIAN_LITTLE)
                  if (bpc > 1 && info->display_endian == BFD_ENDIAN_LITTLE)
                    {
                    {
                      for (k = bpc - 1; k >= 0; k--)
                      for (k = bpc - 1; k >= 0; k--)
                        printf ("%02x", (unsigned) data[j + k]);
                        printf ("%02x", (unsigned) data[j + k]);
                      putchar (' ');
                      putchar (' ');
                    }
                    }
                  else
                  else
                    {
                    {
                      for (k = 0; k < bpc; k++)
                      for (k = 0; k < bpc; k++)
                        printf ("%02x", (unsigned) data[j + k]);
                        printf ("%02x", (unsigned) data[j + k]);
                      putchar (' ');
                      putchar (' ');
                    }
                    }
                }
                }
 
 
              for (; pb < octets_per_line; pb += bpc)
              for (; pb < octets_per_line; pb += bpc)
                {
                {
                  int k;
                  int k;
 
 
                  for (k = 0; k < bpc; k++)
                  for (k = 0; k < bpc; k++)
                    printf ("  ");
                    printf ("  ");
                  putchar (' ');
                  putchar (' ');
                }
                }
 
 
              /* Separate raw data from instruction by extra space.  */
              /* Separate raw data from instruction by extra space.  */
              if (insns)
              if (insns)
                putchar ('\t');
                putchar ('\t');
              else
              else
                printf ("    ");
                printf ("    ");
            }
            }
 
 
          if (! insns)
          if (! insns)
            printf ("%s", buf);
            printf ("%s", buf);
          else if (sfile.pos)
          else if (sfile.pos)
            printf ("%s", sfile.buffer);
            printf ("%s", sfile.buffer);
 
 
          if (prefix_addresses
          if (prefix_addresses
              ? show_raw_insn > 0
              ? show_raw_insn > 0
              : show_raw_insn >= 0)
              : show_raw_insn >= 0)
            {
            {
              while (pb < octets)
              while (pb < octets)
                {
                {
                  bfd_vma j;
                  bfd_vma j;
                  char *s;
                  char *s;
 
 
                  putchar ('\n');
                  putchar ('\n');
                  j = addr_offset * opb + pb;
                  j = addr_offset * opb + pb;
 
 
                  bfd_sprintf_vma (aux->abfd, buf, section->vma + j / opb);
                  bfd_sprintf_vma (aux->abfd, buf, section->vma + j / opb);
                  for (s = buf + skip_addr_chars; *s == '0'; s++)
                  for (s = buf + skip_addr_chars; *s == '0'; s++)
                    *s = ' ';
                    *s = ' ';
                  if (*s == '\0')
                  if (*s == '\0')
                    *--s = '0';
                    *--s = '0';
                  printf ("%s:\t", buf + skip_addr_chars);
                  printf ("%s:\t", buf + skip_addr_chars);
 
 
                  pb += octets_per_line;
                  pb += octets_per_line;
                  if (pb > octets)
                  if (pb > octets)
                    pb = octets;
                    pb = octets;
                  for (; j < addr_offset * opb + pb; j += bpc)
                  for (; j < addr_offset * opb + pb; j += bpc)
                    {
                    {
                      int k;
                      int k;
 
 
                      if (bpc > 1 && info->display_endian == BFD_ENDIAN_LITTLE)
                      if (bpc > 1 && info->display_endian == BFD_ENDIAN_LITTLE)
                        {
                        {
                          for (k = bpc - 1; k >= 0; k--)
                          for (k = bpc - 1; k >= 0; k--)
                            printf ("%02x", (unsigned) data[j + k]);
                            printf ("%02x", (unsigned) data[j + k]);
                          putchar (' ');
                          putchar (' ');
                        }
                        }
                      else
                      else
                        {
                        {
                          for (k = 0; k < bpc; k++)
                          for (k = 0; k < bpc; k++)
                            printf ("%02x", (unsigned) data[j + k]);
                            printf ("%02x", (unsigned) data[j + k]);
                          putchar (' ');
                          putchar (' ');
                        }
                        }
                    }
                    }
                }
                }
            }
            }
 
 
          if (!wide_output)
          if (!wide_output)
            putchar ('\n');
            putchar ('\n');
          else
          else
            need_nl = TRUE;
            need_nl = TRUE;
        }
        }
 
 
      while ((*relppp) < relppend
      while ((*relppp) < relppend
             && (**relppp)->address < rel_offset + addr_offset + octets / opb)
             && (**relppp)->address < rel_offset + addr_offset + octets / opb)
        {
        {
          if (dump_reloc_info || dump_dynamic_reloc_info)
          if (dump_reloc_info || dump_dynamic_reloc_info)
            {
            {
              arelent *q;
              arelent *q;
 
 
              q = **relppp;
              q = **relppp;
 
 
              if (wide_output)
              if (wide_output)
                putchar ('\t');
                putchar ('\t');
              else
              else
                printf ("\t\t\t");
                printf ("\t\t\t");
 
 
              objdump_print_value (section->vma - rel_offset + q->address,
              objdump_print_value (section->vma - rel_offset + q->address,
                                   info, TRUE);
                                   info, TRUE);
 
 
              if (q->howto == NULL)
              if (q->howto == NULL)
                printf (": *unknown*\t");
                printf (": *unknown*\t");
              else if (q->howto->name)
              else if (q->howto->name)
                printf (": %s\t", q->howto->name);
                printf (": %s\t", q->howto->name);
              else
              else
                printf (": %d\t", q->howto->type);
                printf (": %d\t", q->howto->type);
 
 
              if (q->sym_ptr_ptr == NULL || *q->sym_ptr_ptr == NULL)
              if (q->sym_ptr_ptr == NULL || *q->sym_ptr_ptr == NULL)
                printf ("*unknown*");
                printf ("*unknown*");
              else
              else
                {
                {
                  const char *sym_name;
                  const char *sym_name;
 
 
                  sym_name = bfd_asymbol_name (*q->sym_ptr_ptr);
                  sym_name = bfd_asymbol_name (*q->sym_ptr_ptr);
                  if (sym_name != NULL && *sym_name != '\0')
                  if (sym_name != NULL && *sym_name != '\0')
                    objdump_print_symname (aux->abfd, info, *q->sym_ptr_ptr);
                    objdump_print_symname (aux->abfd, info, *q->sym_ptr_ptr);
                  else
                  else
                    {
                    {
                      asection *sym_sec;
                      asection *sym_sec;
 
 
                      sym_sec = bfd_get_section (*q->sym_ptr_ptr);
                      sym_sec = bfd_get_section (*q->sym_ptr_ptr);
                      sym_name = bfd_get_section_name (aux->abfd, sym_sec);
                      sym_name = bfd_get_section_name (aux->abfd, sym_sec);
                      if (sym_name == NULL || *sym_name == '\0')
                      if (sym_name == NULL || *sym_name == '\0')
                        sym_name = "*unknown*";
                        sym_name = "*unknown*";
                      printf ("%s", sym_name);
                      printf ("%s", sym_name);
                    }
                    }
                }
                }
 
 
              if (q->addend)
              if (q->addend)
                {
                {
                  printf ("+0x");
                  printf ("+0x");
                  objdump_print_value (q->addend, info, TRUE);
                  objdump_print_value (q->addend, info, TRUE);
                }
                }
 
 
              printf ("\n");
              printf ("\n");
              need_nl = FALSE;
              need_nl = FALSE;
            }
            }
          ++(*relppp);
          ++(*relppp);
        }
        }
 
 
      if (need_nl)
      if (need_nl)
        printf ("\n");
        printf ("\n");
 
 
      addr_offset += octets / opb;
      addr_offset += octets / opb;
    }
    }
 
 
  free (sfile.buffer);
  free (sfile.buffer);
}
}
 
 
static void
static void
disassemble_section (bfd *abfd, asection *section, void *info)
disassemble_section (bfd *abfd, asection *section, void *info)
{
{
  const struct elf_backend_data * bed;
  const struct elf_backend_data * bed;
  bfd_vma                      sign_adjust = 0;
  bfd_vma                      sign_adjust = 0;
  struct disassemble_info *    pinfo = (struct disassemble_info *) info;
  struct disassemble_info *    pinfo = (struct disassemble_info *) info;
  struct objdump_disasm_info * paux;
  struct objdump_disasm_info * paux;
  unsigned int                 opb = pinfo->octets_per_byte;
  unsigned int                 opb = pinfo->octets_per_byte;
  bfd_byte *                   data = NULL;
  bfd_byte *                   data = NULL;
  bfd_size_type                datasize = 0;
  bfd_size_type                datasize = 0;
  arelent **                   rel_pp = NULL;
  arelent **                   rel_pp = NULL;
  arelent **                   rel_ppstart = NULL;
  arelent **                   rel_ppstart = NULL;
  arelent **                   rel_ppend;
  arelent **                   rel_ppend;
  unsigned long                stop_offset;
  unsigned long                stop_offset;
  asymbol *                    sym = NULL;
  asymbol *                    sym = NULL;
  long                         place = 0;
  long                         place = 0;
  long                         rel_count;
  long                         rel_count;
  bfd_vma                      rel_offset;
  bfd_vma                      rel_offset;
  unsigned long                addr_offset;
  unsigned long                addr_offset;
 
 
  /* Sections that do not contain machine
  /* Sections that do not contain machine
     code are not normally disassembled.  */
     code are not normally disassembled.  */
  if (! disassemble_all
  if (! disassemble_all
      && only == NULL
      && only == NULL
      && ((section->flags & (SEC_CODE | SEC_HAS_CONTENTS))
      && ((section->flags & (SEC_CODE | SEC_HAS_CONTENTS))
          != (SEC_CODE | SEC_HAS_CONTENTS)))
          != (SEC_CODE | SEC_HAS_CONTENTS)))
    return;
    return;
 
 
  if (! process_section_p (section))
  if (! process_section_p (section))
    return;
    return;
 
 
  datasize = bfd_get_section_size (section);
  datasize = bfd_get_section_size (section);
  if (datasize == 0)
  if (datasize == 0)
    return;
    return;
 
 
  /* Decide which set of relocs to use.  Load them if necessary.  */
  /* Decide which set of relocs to use.  Load them if necessary.  */
  paux = (struct objdump_disasm_info *) pinfo->application_data;
  paux = (struct objdump_disasm_info *) pinfo->application_data;
  if (paux->dynrelbuf)
  if (paux->dynrelbuf)
    {
    {
      rel_pp = paux->dynrelbuf;
      rel_pp = paux->dynrelbuf;
      rel_count = paux->dynrelcount;
      rel_count = paux->dynrelcount;
      /* Dynamic reloc addresses are absolute, non-dynamic are section
      /* Dynamic reloc addresses are absolute, non-dynamic are section
         relative.  REL_OFFSET specifies the reloc address corresponding
         relative.  REL_OFFSET specifies the reloc address corresponding
         to the start of this section.  */
         to the start of this section.  */
      rel_offset = section->vma;
      rel_offset = section->vma;
    }
    }
  else
  else
    {
    {
      rel_count = 0;
      rel_count = 0;
      rel_pp = NULL;
      rel_pp = NULL;
      rel_offset = 0;
      rel_offset = 0;
 
 
      if ((section->flags & SEC_RELOC) != 0
      if ((section->flags & SEC_RELOC) != 0
          && (dump_reloc_info || pinfo->disassembler_needs_relocs))
          && (dump_reloc_info || pinfo->disassembler_needs_relocs))
        {
        {
          long relsize;
          long relsize;
 
 
          relsize = bfd_get_reloc_upper_bound (abfd, section);
          relsize = bfd_get_reloc_upper_bound (abfd, section);
          if (relsize < 0)
          if (relsize < 0)
            bfd_fatal (bfd_get_filename (abfd));
            bfd_fatal (bfd_get_filename (abfd));
 
 
          if (relsize > 0)
          if (relsize > 0)
            {
            {
              rel_ppstart = rel_pp = (arelent **) xmalloc (relsize);
              rel_ppstart = rel_pp = (arelent **) xmalloc (relsize);
              rel_count = bfd_canonicalize_reloc (abfd, section, rel_pp, syms);
              rel_count = bfd_canonicalize_reloc (abfd, section, rel_pp, syms);
              if (rel_count < 0)
              if (rel_count < 0)
                bfd_fatal (bfd_get_filename (abfd));
                bfd_fatal (bfd_get_filename (abfd));
 
 
              /* Sort the relocs by address.  */
              /* Sort the relocs by address.  */
              qsort (rel_pp, rel_count, sizeof (arelent *), compare_relocs);
              qsort (rel_pp, rel_count, sizeof (arelent *), compare_relocs);
            }
            }
        }
        }
    }
    }
  rel_ppend = rel_pp + rel_count;
  rel_ppend = rel_pp + rel_count;
 
 
  data = (bfd_byte *) xmalloc (datasize);
  data = (bfd_byte *) xmalloc (datasize);
 
 
  bfd_get_section_contents (abfd, section, data, 0, datasize);
  bfd_get_section_contents (abfd, section, data, 0, datasize);
 
 
  paux->sec = section;
  paux->sec = section;
  pinfo->buffer = data;
  pinfo->buffer = data;
  pinfo->buffer_vma = section->vma;
  pinfo->buffer_vma = section->vma;
  pinfo->buffer_length = datasize;
  pinfo->buffer_length = datasize;
  pinfo->section = section;
  pinfo->section = section;
 
 
  if (start_address == (bfd_vma) -1
  if (start_address == (bfd_vma) -1
      || start_address < pinfo->buffer_vma)
      || start_address < pinfo->buffer_vma)
    addr_offset = 0;
    addr_offset = 0;
  else
  else
    addr_offset = start_address - pinfo->buffer_vma;
    addr_offset = start_address - pinfo->buffer_vma;
 
 
  if (stop_address == (bfd_vma) -1)
  if (stop_address == (bfd_vma) -1)
    stop_offset = datasize / opb;
    stop_offset = datasize / opb;
  else
  else
    {
    {
      if (stop_address < pinfo->buffer_vma)
      if (stop_address < pinfo->buffer_vma)
        stop_offset = 0;
        stop_offset = 0;
      else
      else
        stop_offset = stop_address - pinfo->buffer_vma;
        stop_offset = stop_address - pinfo->buffer_vma;
      if (stop_offset > pinfo->buffer_length / opb)
      if (stop_offset > pinfo->buffer_length / opb)
        stop_offset = pinfo->buffer_length / opb;
        stop_offset = pinfo->buffer_length / opb;
    }
    }
 
 
  /* Skip over the relocs belonging to addresses below the
  /* Skip over the relocs belonging to addresses below the
     start address.  */
     start address.  */
  while (rel_pp < rel_ppend
  while (rel_pp < rel_ppend
         && (*rel_pp)->address < rel_offset + addr_offset)
         && (*rel_pp)->address < rel_offset + addr_offset)
    ++rel_pp;
    ++rel_pp;
 
 
  if (addr_offset < stop_offset)
  if (addr_offset < stop_offset)
    printf (_("\nDisassembly of section %s:\n"), section->name);
    printf (_("\nDisassembly of section %s:\n"), section->name);
 
 
  /* Find the nearest symbol forwards from our current position.  */
  /* Find the nearest symbol forwards from our current position.  */
  paux->require_sec = TRUE;
  paux->require_sec = TRUE;
  sym = (asymbol *) find_symbol_for_address (section->vma + addr_offset,
  sym = (asymbol *) find_symbol_for_address (section->vma + addr_offset,
                                             (struct disassemble_info *) info,
                                             (struct disassemble_info *) info,
                                             &place);
                                             &place);
  paux->require_sec = FALSE;
  paux->require_sec = FALSE;
 
 
  /* PR 9774: If the target used signed addresses then we must make
  /* PR 9774: If the target used signed addresses then we must make
     sure that we sign extend the value that we calculate for 'addr'
     sure that we sign extend the value that we calculate for 'addr'
     in the loop below.  */
     in the loop below.  */
  if (bfd_get_flavour (abfd) == bfd_target_elf_flavour
  if (bfd_get_flavour (abfd) == bfd_target_elf_flavour
      && (bed = get_elf_backend_data (abfd)) != NULL
      && (bed = get_elf_backend_data (abfd)) != NULL
      && bed->sign_extend_vma)
      && bed->sign_extend_vma)
    sign_adjust = (bfd_vma) 1 << (bed->s->arch_size - 1);
    sign_adjust = (bfd_vma) 1 << (bed->s->arch_size - 1);
 
 
  /* Disassemble a block of instructions up to the address associated with
  /* Disassemble a block of instructions up to the address associated with
     the symbol we have just found.  Then print the symbol and find the
     the symbol we have just found.  Then print the symbol and find the
     next symbol on.  Repeat until we have disassembled the entire section
     next symbol on.  Repeat until we have disassembled the entire section
     or we have reached the end of the address range we are interested in.  */
     or we have reached the end of the address range we are interested in.  */
  while (addr_offset < stop_offset)
  while (addr_offset < stop_offset)
    {
    {
      bfd_vma addr;
      bfd_vma addr;
      asymbol *nextsym;
      asymbol *nextsym;
      unsigned long nextstop_offset;
      unsigned long nextstop_offset;
      bfd_boolean insns;
      bfd_boolean insns;
 
 
      addr = section->vma + addr_offset;
      addr = section->vma + addr_offset;
      addr = ((addr & ((sign_adjust << 1) - 1)) ^ sign_adjust) - sign_adjust;
      addr = ((addr & ((sign_adjust << 1) - 1)) ^ sign_adjust) - sign_adjust;
 
 
      if (sym != NULL && bfd_asymbol_value (sym) <= addr)
      if (sym != NULL && bfd_asymbol_value (sym) <= addr)
        {
        {
          int x;
          int x;
 
 
          for (x = place;
          for (x = place;
               (x < sorted_symcount
               (x < sorted_symcount
                && (bfd_asymbol_value (sorted_syms[x]) <= addr));
                && (bfd_asymbol_value (sorted_syms[x]) <= addr));
               ++x)
               ++x)
            continue;
            continue;
 
 
          pinfo->symbols = sorted_syms + place;
          pinfo->symbols = sorted_syms + place;
          pinfo->num_symbols = x - place;
          pinfo->num_symbols = x - place;
          pinfo->symtab_pos = place;
          pinfo->symtab_pos = place;
        }
        }
      else
      else
        {
        {
          pinfo->symbols = NULL;
          pinfo->symbols = NULL;
          pinfo->num_symbols = 0;
          pinfo->num_symbols = 0;
          pinfo->symtab_pos = -1;
          pinfo->symtab_pos = -1;
        }
        }
 
 
      if (! prefix_addresses)
      if (! prefix_addresses)
        {
        {
          pinfo->fprintf_func (pinfo->stream, "\n");
          pinfo->fprintf_func (pinfo->stream, "\n");
          objdump_print_addr_with_sym (abfd, section, sym, addr,
          objdump_print_addr_with_sym (abfd, section, sym, addr,
                                       pinfo, FALSE);
                                       pinfo, FALSE);
          pinfo->fprintf_func (pinfo->stream, ":\n");
          pinfo->fprintf_func (pinfo->stream, ":\n");
        }
        }
 
 
      if (sym != NULL && bfd_asymbol_value (sym) > addr)
      if (sym != NULL && bfd_asymbol_value (sym) > addr)
        nextsym = sym;
        nextsym = sym;
      else if (sym == NULL)
      else if (sym == NULL)
        nextsym = NULL;
        nextsym = NULL;
      else
      else
        {
        {
#define is_valid_next_sym(SYM) \
#define is_valid_next_sym(SYM) \
  ((SYM)->section == section \
  ((SYM)->section == section \
   && (bfd_asymbol_value (SYM) > bfd_asymbol_value (sym)) \
   && (bfd_asymbol_value (SYM) > bfd_asymbol_value (sym)) \
   && pinfo->symbol_is_valid (SYM, pinfo))
   && pinfo->symbol_is_valid (SYM, pinfo))
 
 
          /* Search forward for the next appropriate symbol in
          /* Search forward for the next appropriate symbol in
             SECTION.  Note that all the symbols are sorted
             SECTION.  Note that all the symbols are sorted
             together into one big array, and that some sections
             together into one big array, and that some sections
             may have overlapping addresses.  */
             may have overlapping addresses.  */
          while (place < sorted_symcount
          while (place < sorted_symcount
                 && ! is_valid_next_sym (sorted_syms [place]))
                 && ! is_valid_next_sym (sorted_syms [place]))
            ++place;
            ++place;
 
 
          if (place >= sorted_symcount)
          if (place >= sorted_symcount)
            nextsym = NULL;
            nextsym = NULL;
          else
          else
            nextsym = sorted_syms[place];
            nextsym = sorted_syms[place];
        }
        }
 
 
      if (sym != NULL && bfd_asymbol_value (sym) > addr)
      if (sym != NULL && bfd_asymbol_value (sym) > addr)
        nextstop_offset = bfd_asymbol_value (sym) - section->vma;
        nextstop_offset = bfd_asymbol_value (sym) - section->vma;
      else if (nextsym == NULL)
      else if (nextsym == NULL)
        nextstop_offset = stop_offset;
        nextstop_offset = stop_offset;
      else
      else
        nextstop_offset = bfd_asymbol_value (nextsym) - section->vma;
        nextstop_offset = bfd_asymbol_value (nextsym) - section->vma;
 
 
      if (nextstop_offset > stop_offset
      if (nextstop_offset > stop_offset
          || nextstop_offset <= addr_offset)
          || nextstop_offset <= addr_offset)
        nextstop_offset = stop_offset;
        nextstop_offset = stop_offset;
 
 
      /* If a symbol is explicitly marked as being an object
      /* If a symbol is explicitly marked as being an object
         rather than a function, just dump the bytes without
         rather than a function, just dump the bytes without
         disassembling them.  */
         disassembling them.  */
      if (disassemble_all
      if (disassemble_all
          || sym == NULL
          || sym == NULL
          || sym->section != section
          || sym->section != section
          || bfd_asymbol_value (sym) > addr
          || bfd_asymbol_value (sym) > addr
          || ((sym->flags & BSF_OBJECT) == 0
          || ((sym->flags & BSF_OBJECT) == 0
              && (strstr (bfd_asymbol_name (sym), "gnu_compiled")
              && (strstr (bfd_asymbol_name (sym), "gnu_compiled")
                  == NULL)
                  == NULL)
              && (strstr (bfd_asymbol_name (sym), "gcc2_compiled")
              && (strstr (bfd_asymbol_name (sym), "gcc2_compiled")
                  == NULL))
                  == NULL))
          || (sym->flags & BSF_FUNCTION) != 0)
          || (sym->flags & BSF_FUNCTION) != 0)
        insns = TRUE;
        insns = TRUE;
      else
      else
        insns = FALSE;
        insns = FALSE;
 
 
      disassemble_bytes (pinfo, paux->disassemble_fn, insns, data,
      disassemble_bytes (pinfo, paux->disassemble_fn, insns, data,
                         addr_offset, nextstop_offset,
                         addr_offset, nextstop_offset,
                         rel_offset, &rel_pp, rel_ppend);
                         rel_offset, &rel_pp, rel_ppend);
 
 
      addr_offset = nextstop_offset;
      addr_offset = nextstop_offset;
      sym = nextsym;
      sym = nextsym;
    }
    }
 
 
  free (data);
  free (data);
 
 
  if (rel_ppstart != NULL)
  if (rel_ppstart != NULL)
    free (rel_ppstart);
    free (rel_ppstart);
}
}
 
 
/* Disassemble the contents of an object file.  */
/* Disassemble the contents of an object file.  */
 
 
static void
static void
disassemble_data (bfd *abfd)
disassemble_data (bfd *abfd)
{
{
  struct disassemble_info disasm_info;
  struct disassemble_info disasm_info;
  struct objdump_disasm_info aux;
  struct objdump_disasm_info aux;
  long i;
  long i;
 
 
  print_files = NULL;
  print_files = NULL;
  prev_functionname = NULL;
  prev_functionname = NULL;
  prev_line = -1;
  prev_line = -1;
 
 
  /* We make a copy of syms to sort.  We don't want to sort syms
  /* We make a copy of syms to sort.  We don't want to sort syms
     because that will screw up the relocs.  */
     because that will screw up the relocs.  */
  sorted_symcount = symcount ? symcount : dynsymcount;
  sorted_symcount = symcount ? symcount : dynsymcount;
  sorted_syms = (asymbol **) xmalloc ((sorted_symcount + synthcount)
  sorted_syms = (asymbol **) xmalloc ((sorted_symcount + synthcount)
                                      * sizeof (asymbol *));
                                      * sizeof (asymbol *));
  memcpy (sorted_syms, symcount ? syms : dynsyms,
  memcpy (sorted_syms, symcount ? syms : dynsyms,
          sorted_symcount * sizeof (asymbol *));
          sorted_symcount * sizeof (asymbol *));
 
 
  sorted_symcount = remove_useless_symbols (sorted_syms, sorted_symcount);
  sorted_symcount = remove_useless_symbols (sorted_syms, sorted_symcount);
 
 
  for (i = 0; i < synthcount; ++i)
  for (i = 0; i < synthcount; ++i)
    {
    {
      sorted_syms[sorted_symcount] = synthsyms + i;
      sorted_syms[sorted_symcount] = synthsyms + i;
      ++sorted_symcount;
      ++sorted_symcount;
    }
    }
 
 
  /* Sort the symbols into section and symbol order.  */
  /* Sort the symbols into section and symbol order.  */
  qsort (sorted_syms, sorted_symcount, sizeof (asymbol *), compare_symbols);
  qsort (sorted_syms, sorted_symcount, sizeof (asymbol *), compare_symbols);
 
 
  init_disassemble_info (&disasm_info, stdout, (fprintf_ftype) fprintf);
  init_disassemble_info (&disasm_info, stdout, (fprintf_ftype) fprintf);
 
 
  disasm_info.application_data = (void *) &aux;
  disasm_info.application_data = (void *) &aux;
  aux.abfd = abfd;
  aux.abfd = abfd;
  aux.require_sec = FALSE;
  aux.require_sec = FALSE;
  aux.dynrelbuf = NULL;
  aux.dynrelbuf = NULL;
  aux.dynrelcount = 0;
  aux.dynrelcount = 0;
  aux.reloc = NULL;
  aux.reloc = NULL;
 
 
  disasm_info.print_address_func = objdump_print_address;
  disasm_info.print_address_func = objdump_print_address;
  disasm_info.symbol_at_address_func = objdump_symbol_at_address;
  disasm_info.symbol_at_address_func = objdump_symbol_at_address;
 
 
  if (machine != NULL)
  if (machine != NULL)
    {
    {
      const bfd_arch_info_type *info = bfd_scan_arch (machine);
      const bfd_arch_info_type *info = bfd_scan_arch (machine);
 
 
      if (info == NULL)
      if (info == NULL)
        fatal (_("Can't use supplied machine %s"), machine);
        fatal (_("Can't use supplied machine %s"), machine);
 
 
      abfd->arch_info = info;
      abfd->arch_info = info;
    }
    }
 
 
  if (endian != BFD_ENDIAN_UNKNOWN)
  if (endian != BFD_ENDIAN_UNKNOWN)
    {
    {
      struct bfd_target *xvec;
      struct bfd_target *xvec;
 
 
      xvec = (struct bfd_target *) xmalloc (sizeof (struct bfd_target));
      xvec = (struct bfd_target *) xmalloc (sizeof (struct bfd_target));
      memcpy (xvec, abfd->xvec, sizeof (struct bfd_target));
      memcpy (xvec, abfd->xvec, sizeof (struct bfd_target));
      xvec->byteorder = endian;
      xvec->byteorder = endian;
      abfd->xvec = xvec;
      abfd->xvec = xvec;
    }
    }
 
 
  /* Use libopcodes to locate a suitable disassembler.  */
  /* Use libopcodes to locate a suitable disassembler.  */
  aux.disassemble_fn = disassembler (abfd);
  aux.disassemble_fn = disassembler (abfd);
  if (!aux.disassemble_fn)
  if (!aux.disassemble_fn)
    {
    {
      non_fatal (_("Can't disassemble for architecture %s\n"),
      non_fatal (_("Can't disassemble for architecture %s\n"),
                 bfd_printable_arch_mach (bfd_get_arch (abfd), 0));
                 bfd_printable_arch_mach (bfd_get_arch (abfd), 0));
      exit_status = 1;
      exit_status = 1;
      return;
      return;
    }
    }
 
 
  disasm_info.flavour = bfd_get_flavour (abfd);
  disasm_info.flavour = bfd_get_flavour (abfd);
  disasm_info.arch = bfd_get_arch (abfd);
  disasm_info.arch = bfd_get_arch (abfd);
  disasm_info.mach = bfd_get_mach (abfd);
  disasm_info.mach = bfd_get_mach (abfd);
  disasm_info.disassembler_options = disassembler_options;
  disasm_info.disassembler_options = disassembler_options;
  disasm_info.octets_per_byte = bfd_octets_per_byte (abfd);
  disasm_info.octets_per_byte = bfd_octets_per_byte (abfd);
  disasm_info.skip_zeroes = DEFAULT_SKIP_ZEROES;
  disasm_info.skip_zeroes = DEFAULT_SKIP_ZEROES;
  disasm_info.skip_zeroes_at_end = DEFAULT_SKIP_ZEROES_AT_END;
  disasm_info.skip_zeroes_at_end = DEFAULT_SKIP_ZEROES_AT_END;
  disasm_info.disassembler_needs_relocs = FALSE;
  disasm_info.disassembler_needs_relocs = FALSE;
 
 
  if (bfd_big_endian (abfd))
  if (bfd_big_endian (abfd))
    disasm_info.display_endian = disasm_info.endian = BFD_ENDIAN_BIG;
    disasm_info.display_endian = disasm_info.endian = BFD_ENDIAN_BIG;
  else if (bfd_little_endian (abfd))
  else if (bfd_little_endian (abfd))
    disasm_info.display_endian = disasm_info.endian = BFD_ENDIAN_LITTLE;
    disasm_info.display_endian = disasm_info.endian = BFD_ENDIAN_LITTLE;
  else
  else
    /* ??? Aborting here seems too drastic.  We could default to big or little
    /* ??? Aborting here seems too drastic.  We could default to big or little
       instead.  */
       instead.  */
    disasm_info.endian = BFD_ENDIAN_UNKNOWN;
    disasm_info.endian = BFD_ENDIAN_UNKNOWN;
 
 
  /* Allow the target to customize the info structure.  */
  /* Allow the target to customize the info structure.  */
  disassemble_init_for_target (& disasm_info);
  disassemble_init_for_target (& disasm_info);
 
 
  /* Pre-load the dynamic relocs if we are going
  /* Pre-load the dynamic relocs if we are going
     to be dumping them along with the disassembly.  */
     to be dumping them along with the disassembly.  */
  if (dump_dynamic_reloc_info)
  if (dump_dynamic_reloc_info)
    {
    {
      long relsize = bfd_get_dynamic_reloc_upper_bound (abfd);
      long relsize = bfd_get_dynamic_reloc_upper_bound (abfd);
 
 
      if (relsize < 0)
      if (relsize < 0)
        bfd_fatal (bfd_get_filename (abfd));
        bfd_fatal (bfd_get_filename (abfd));
 
 
      if (relsize > 0)
      if (relsize > 0)
        {
        {
          aux.dynrelbuf = (arelent **) xmalloc (relsize);
          aux.dynrelbuf = (arelent **) xmalloc (relsize);
          aux.dynrelcount = bfd_canonicalize_dynamic_reloc (abfd,
          aux.dynrelcount = bfd_canonicalize_dynamic_reloc (abfd,
                                                            aux.dynrelbuf,
                                                            aux.dynrelbuf,
                                                            dynsyms);
                                                            dynsyms);
          if (aux.dynrelcount < 0)
          if (aux.dynrelcount < 0)
            bfd_fatal (bfd_get_filename (abfd));
            bfd_fatal (bfd_get_filename (abfd));
 
 
          /* Sort the relocs by address.  */
          /* Sort the relocs by address.  */
          qsort (aux.dynrelbuf, aux.dynrelcount, sizeof (arelent *),
          qsort (aux.dynrelbuf, aux.dynrelcount, sizeof (arelent *),
                 compare_relocs);
                 compare_relocs);
        }
        }
    }
    }
  disasm_info.symtab = sorted_syms;
  disasm_info.symtab = sorted_syms;
  disasm_info.symtab_size = sorted_symcount;
  disasm_info.symtab_size = sorted_symcount;
 
 
  bfd_map_over_sections (abfd, disassemble_section, & disasm_info);
  bfd_map_over_sections (abfd, disassemble_section, & disasm_info);
 
 
  if (aux.dynrelbuf != NULL)
  if (aux.dynrelbuf != NULL)
    free (aux.dynrelbuf);
    free (aux.dynrelbuf);
  free (sorted_syms);
  free (sorted_syms);
}
}


static int
static int
load_specific_debug_section (enum dwarf_section_display_enum debug,
load_specific_debug_section (enum dwarf_section_display_enum debug,
                             asection *sec, void *file)
                             asection *sec, void *file)
{
{
  struct dwarf_section *section = &debug_displays [debug].section;
  struct dwarf_section *section = &debug_displays [debug].section;
  bfd *abfd = (bfd *) file;
  bfd *abfd = (bfd *) file;
  bfd_boolean ret;
  bfd_boolean ret;
  int section_is_compressed;
  int section_is_compressed;
 
 
  /* If it is already loaded, do nothing.  */
  /* If it is already loaded, do nothing.  */
  if (section->start != NULL)
  if (section->start != NULL)
    return 1;
    return 1;
 
 
  section_is_compressed = section->name == section->compressed_name;
  section_is_compressed = section->name == section->compressed_name;
 
 
  section->address = 0;
  section->address = 0;
  section->size = bfd_get_section_size (sec);
  section->size = bfd_get_section_size (sec);
  section->start = (unsigned char *) xmalloc (section->size);
  section->start = (unsigned char *) xmalloc (section->size);
 
 
  if (is_relocatable && debug_displays [debug].relocate)
  if (is_relocatable && debug_displays [debug].relocate)
    ret = bfd_simple_get_relocated_section_contents (abfd,
    ret = bfd_simple_get_relocated_section_contents (abfd,
                                                     sec,
                                                     sec,
                                                     section->start,
                                                     section->start,
                                                     syms) != NULL;
                                                     syms) != NULL;
  else
  else
    ret = bfd_get_section_contents (abfd, sec, section->start, 0,
    ret = bfd_get_section_contents (abfd, sec, section->start, 0,
                                    section->size);
                                    section->size);
 
 
  if (! ret)
  if (! ret)
    {
    {
      free_debug_section (debug);
      free_debug_section (debug);
      printf (_("\nCan't get contents for section '%s'.\n"),
      printf (_("\nCan't get contents for section '%s'.\n"),
              section->name);
              section->name);
      return 0;
      return 0;
    }
    }
 
 
  if (section_is_compressed)
  if (section_is_compressed)
    {
    {
      bfd_size_type size = section->size;
      bfd_size_type size = section->size;
      if (! bfd_uncompress_section_contents (&section->start, &size))
      if (! bfd_uncompress_section_contents (&section->start, &size))
        {
        {
          free_debug_section (debug);
          free_debug_section (debug);
          printf (_("\nCan't uncompress section '%s'.\n"), section->name);
          printf (_("\nCan't uncompress section '%s'.\n"), section->name);
          return 0;
          return 0;
        }
        }
      section->size = size;
      section->size = size;
    }
    }
 
 
  return 1;
  return 1;
}
}
 
 
int
int
load_debug_section (enum dwarf_section_display_enum debug, void *file)
load_debug_section (enum dwarf_section_display_enum debug, void *file)
{
{
  struct dwarf_section *section = &debug_displays [debug].section;
  struct dwarf_section *section = &debug_displays [debug].section;
  bfd *abfd = (bfd *) file;
  bfd *abfd = (bfd *) file;
  asection *sec;
  asection *sec;
 
 
  /* If it is already loaded, do nothing.  */
  /* If it is already loaded, do nothing.  */
  if (section->start != NULL)
  if (section->start != NULL)
    return 1;
    return 1;
 
 
  /* Locate the debug section.  */
  /* Locate the debug section.  */
  sec = bfd_get_section_by_name (abfd, section->uncompressed_name);
  sec = bfd_get_section_by_name (abfd, section->uncompressed_name);
  if (sec != NULL)
  if (sec != NULL)
    section->name = section->uncompressed_name;
    section->name = section->uncompressed_name;
  else
  else
    {
    {
      sec = bfd_get_section_by_name (abfd, section->compressed_name);
      sec = bfd_get_section_by_name (abfd, section->compressed_name);
      if (sec != NULL)
      if (sec != NULL)
        section->name = section->compressed_name;
        section->name = section->compressed_name;
    }
    }
  if (sec == NULL)
  if (sec == NULL)
    return 0;
    return 0;
 
 
  return load_specific_debug_section (debug, sec, file);
  return load_specific_debug_section (debug, sec, file);
}
}
 
 
void
void
free_debug_section (enum dwarf_section_display_enum debug)
free_debug_section (enum dwarf_section_display_enum debug)
{
{
  struct dwarf_section *section = &debug_displays [debug].section;
  struct dwarf_section *section = &debug_displays [debug].section;
 
 
  if (section->start == NULL)
  if (section->start == NULL)
    return;
    return;
 
 
  free ((char *) section->start);
  free ((char *) section->start);
  section->start = NULL;
  section->start = NULL;
  section->address = 0;
  section->address = 0;
  section->size = 0;
  section->size = 0;
}
}
 
 
static void
static void
dump_dwarf_section (bfd *abfd, asection *section,
dump_dwarf_section (bfd *abfd, asection *section,
                    void *arg ATTRIBUTE_UNUSED)
                    void *arg ATTRIBUTE_UNUSED)
{
{
  const char *name = bfd_get_section_name (abfd, section);
  const char *name = bfd_get_section_name (abfd, section);
  const char *match;
  const char *match;
  int i;
  int i;
 
 
  if (CONST_STRNEQ (name, ".gnu.linkonce.wi."))
  if (CONST_STRNEQ (name, ".gnu.linkonce.wi."))
    match = ".debug_info";
    match = ".debug_info";
  else
  else
    match = name;
    match = name;
 
 
  for (i = 0; i < max; i++)
  for (i = 0; i < max; i++)
    if ((strcmp (debug_displays [i].section.uncompressed_name, match) == 0
    if ((strcmp (debug_displays [i].section.uncompressed_name, match) == 0
         || strcmp (debug_displays [i].section.compressed_name, match) == 0)
         || strcmp (debug_displays [i].section.compressed_name, match) == 0)
        && debug_displays [i].enabled != NULL
        && debug_displays [i].enabled != NULL
        && *debug_displays [i].enabled)
        && *debug_displays [i].enabled)
      {
      {
        struct dwarf_section *sec = &debug_displays [i].section;
        struct dwarf_section *sec = &debug_displays [i].section;
 
 
        if (strcmp (sec->uncompressed_name, match) == 0)
        if (strcmp (sec->uncompressed_name, match) == 0)
          sec->name = sec->uncompressed_name;
          sec->name = sec->uncompressed_name;
        else
        else
          sec->name = sec->compressed_name;
          sec->name = sec->compressed_name;
        if (load_specific_debug_section ((enum dwarf_section_display_enum) i,
        if (load_specific_debug_section ((enum dwarf_section_display_enum) i,
                                         section, abfd))
                                         section, abfd))
          {
          {
            debug_displays [i].display (sec, abfd);
            debug_displays [i].display (sec, abfd);
 
 
            if (i != info && i != abbrev)
            if (i != info && i != abbrev)
              free_debug_section ((enum dwarf_section_display_enum) i);
              free_debug_section ((enum dwarf_section_display_enum) i);
          }
          }
        break;
        break;
      }
      }
}
}
 
 
/* Dump the dwarf debugging information.  */
/* Dump the dwarf debugging information.  */
 
 
static void
static void
dump_dwarf (bfd *abfd)
dump_dwarf (bfd *abfd)
{
{
  is_relocatable = (abfd->flags & (EXEC_P | DYNAMIC)) == 0;
  is_relocatable = (abfd->flags & (EXEC_P | DYNAMIC)) == 0;
 
 
  /* FIXME: bfd_get_arch_size may return -1.  We assume that 64bit
  /* FIXME: bfd_get_arch_size may return -1.  We assume that 64bit
     targets will return 64.  */
     targets will return 64.  */
  eh_addr_size = bfd_get_arch_size (abfd) == 64 ? 8 : 4;
  eh_addr_size = bfd_get_arch_size (abfd) == 64 ? 8 : 4;
 
 
  if (bfd_big_endian (abfd))
  if (bfd_big_endian (abfd))
    byte_get = byte_get_big_endian;
    byte_get = byte_get_big_endian;
  else if (bfd_little_endian (abfd))
  else if (bfd_little_endian (abfd))
    byte_get = byte_get_little_endian;
    byte_get = byte_get_little_endian;
  else
  else
    abort ();
    abort ();
 
 
  if (bfd_get_flavour (abfd) == bfd_target_elf_flavour)
  if (bfd_get_flavour (abfd) == bfd_target_elf_flavour)
    {
    {
      const struct elf_backend_data *bed = get_elf_backend_data (abfd);
      const struct elf_backend_data *bed = get_elf_backend_data (abfd);
      init_dwarf_regnames (bed->elf_machine_code);
      init_dwarf_regnames (bed->elf_machine_code);
    }
    }
 
 
  bfd_map_over_sections (abfd, dump_dwarf_section, NULL);
  bfd_map_over_sections (abfd, dump_dwarf_section, NULL);
 
 
  free_debug_memory ();
  free_debug_memory ();
}
}


/* Read ABFD's stabs section STABSECT_NAME, and return a pointer to
/* Read ABFD's stabs section STABSECT_NAME, and return a pointer to
   it.  Return NULL on failure.   */
   it.  Return NULL on failure.   */
 
 
static char *
static char *
read_section_stabs (bfd *abfd, const char *sect_name, bfd_size_type *size_ptr)
read_section_stabs (bfd *abfd, const char *sect_name, bfd_size_type *size_ptr)
{
{
  asection *stabsect;
  asection *stabsect;
  bfd_size_type size;
  bfd_size_type size;
  char *contents;
  char *contents;
 
 
  stabsect = bfd_get_section_by_name (abfd, sect_name);
  stabsect = bfd_get_section_by_name (abfd, sect_name);
  if (stabsect == NULL)
  if (stabsect == NULL)
    {
    {
      printf (_("No %s section present\n\n"), sect_name);
      printf (_("No %s section present\n\n"), sect_name);
      return FALSE;
      return FALSE;
    }
    }
 
 
  size = bfd_section_size (abfd, stabsect);
  size = bfd_section_size (abfd, stabsect);
  contents  = (char *) xmalloc (size);
  contents  = (char *) xmalloc (size);
 
 
  if (! bfd_get_section_contents (abfd, stabsect, contents, 0, size))
  if (! bfd_get_section_contents (abfd, stabsect, contents, 0, size))
    {
    {
      non_fatal (_("Reading %s section of %s failed: %s"),
      non_fatal (_("Reading %s section of %s failed: %s"),
                 sect_name, bfd_get_filename (abfd),
                 sect_name, bfd_get_filename (abfd),
                 bfd_errmsg (bfd_get_error ()));
                 bfd_errmsg (bfd_get_error ()));
      free (contents);
      free (contents);
      exit_status = 1;
      exit_status = 1;
      return NULL;
      return NULL;
    }
    }
 
 
  *size_ptr = size;
  *size_ptr = size;
 
 
  return contents;
  return contents;
}
}
 
 
/* Stabs entries use a 12 byte format:
/* Stabs entries use a 12 byte format:
     4 byte string table index
     4 byte string table index
     1 byte stab type
     1 byte stab type
     1 byte stab other field
     1 byte stab other field
     2 byte stab desc field
     2 byte stab desc field
     4 byte stab value
     4 byte stab value
   FIXME: This will have to change for a 64 bit object format.  */
   FIXME: This will have to change for a 64 bit object format.  */
 
 
#define STRDXOFF  (0)
#define STRDXOFF  (0)
#define TYPEOFF   (4)
#define TYPEOFF   (4)
#define OTHEROFF  (5)
#define OTHEROFF  (5)
#define DESCOFF   (6)
#define DESCOFF   (6)
#define VALOFF    (8)
#define VALOFF    (8)
#define STABSIZE (12)
#define STABSIZE (12)
 
 
/* Print ABFD's stabs section STABSECT_NAME (in `stabs'),
/* Print ABFD's stabs section STABSECT_NAME (in `stabs'),
   using string table section STRSECT_NAME (in `strtab').  */
   using string table section STRSECT_NAME (in `strtab').  */
 
 
static void
static void
print_section_stabs (bfd *abfd,
print_section_stabs (bfd *abfd,
                     const char *stabsect_name,
                     const char *stabsect_name,
                     unsigned *string_offset_ptr)
                     unsigned *string_offset_ptr)
{
{
  int i;
  int i;
  unsigned file_string_table_offset = 0;
  unsigned file_string_table_offset = 0;
  unsigned next_file_string_table_offset = *string_offset_ptr;
  unsigned next_file_string_table_offset = *string_offset_ptr;
  bfd_byte *stabp, *stabs_end;
  bfd_byte *stabp, *stabs_end;
 
 
  stabp = stabs;
  stabp = stabs;
  stabs_end = stabp + stab_size;
  stabs_end = stabp + stab_size;
 
 
  printf (_("Contents of %s section:\n\n"), stabsect_name);
  printf (_("Contents of %s section:\n\n"), stabsect_name);
  printf ("Symnum n_type n_othr n_desc n_value  n_strx String\n");
  printf ("Symnum n_type n_othr n_desc n_value  n_strx String\n");
 
 
  /* Loop through all symbols and print them.
  /* Loop through all symbols and print them.
 
 
     We start the index at -1 because there is a dummy symbol on
     We start the index at -1 because there is a dummy symbol on
     the front of stabs-in-{coff,elf} sections that supplies sizes.  */
     the front of stabs-in-{coff,elf} sections that supplies sizes.  */
  for (i = -1; stabp < stabs_end; stabp += STABSIZE, i++)
  for (i = -1; stabp < stabs_end; stabp += STABSIZE, i++)
    {
    {
      const char *name;
      const char *name;
      unsigned long strx;
      unsigned long strx;
      unsigned char type, other;
      unsigned char type, other;
      unsigned short desc;
      unsigned short desc;
      bfd_vma value;
      bfd_vma value;
 
 
      strx = bfd_h_get_32 (abfd, stabp + STRDXOFF);
      strx = bfd_h_get_32 (abfd, stabp + STRDXOFF);
      type = bfd_h_get_8 (abfd, stabp + TYPEOFF);
      type = bfd_h_get_8 (abfd, stabp + TYPEOFF);
      other = bfd_h_get_8 (abfd, stabp + OTHEROFF);
      other = bfd_h_get_8 (abfd, stabp + OTHEROFF);
      desc = bfd_h_get_16 (abfd, stabp + DESCOFF);
      desc = bfd_h_get_16 (abfd, stabp + DESCOFF);
      value = bfd_h_get_32 (abfd, stabp + VALOFF);
      value = bfd_h_get_32 (abfd, stabp + VALOFF);
 
 
      printf ("\n%-6d ", i);
      printf ("\n%-6d ", i);
      /* Either print the stab name, or, if unnamed, print its number
      /* Either print the stab name, or, if unnamed, print its number
         again (makes consistent formatting for tools like awk).  */
         again (makes consistent formatting for tools like awk).  */
      name = bfd_get_stab_name (type);
      name = bfd_get_stab_name (type);
      if (name != NULL)
      if (name != NULL)
        printf ("%-6s", name);
        printf ("%-6s", name);
      else if (type == N_UNDF)
      else if (type == N_UNDF)
        printf ("HdrSym");
        printf ("HdrSym");
      else
      else
        printf ("%-6d", type);
        printf ("%-6d", type);
      printf (" %-6d %-6d ", other, desc);
      printf (" %-6d %-6d ", other, desc);
      bfd_printf_vma (abfd, value);
      bfd_printf_vma (abfd, value);
      printf (" %-6lu", strx);
      printf (" %-6lu", strx);
 
 
      /* Symbols with type == 0 (N_UNDF) specify the length of the
      /* Symbols with type == 0 (N_UNDF) specify the length of the
         string table associated with this file.  We use that info
         string table associated with this file.  We use that info
         to know how to relocate the *next* file's string table indices.  */
         to know how to relocate the *next* file's string table indices.  */
      if (type == N_UNDF)
      if (type == N_UNDF)
        {
        {
          file_string_table_offset = next_file_string_table_offset;
          file_string_table_offset = next_file_string_table_offset;
          next_file_string_table_offset += value;
          next_file_string_table_offset += value;
        }
        }
      else
      else
        {
        {
          /* Using the (possibly updated) string table offset, print the
          /* Using the (possibly updated) string table offset, print the
             string (if any) associated with this symbol.  */
             string (if any) associated with this symbol.  */
          if ((strx + file_string_table_offset) < stabstr_size)
          if ((strx + file_string_table_offset) < stabstr_size)
            printf (" %s", &strtab[strx + file_string_table_offset]);
            printf (" %s", &strtab[strx + file_string_table_offset]);
          else
          else
            printf (" *");
            printf (" *");
        }
        }
    }
    }
  printf ("\n\n");
  printf ("\n\n");
  *string_offset_ptr = next_file_string_table_offset;
  *string_offset_ptr = next_file_string_table_offset;
}
}
 
 
typedef struct
typedef struct
{
{
  const char * section_name;
  const char * section_name;
  const char * string_section_name;
  const char * string_section_name;
  unsigned string_offset;
  unsigned string_offset;
}
}
stab_section_names;
stab_section_names;
 
 
static void
static void
find_stabs_section (bfd *abfd, asection *section, void *names)
find_stabs_section (bfd *abfd, asection *section, void *names)
{
{
  int len;
  int len;
  stab_section_names * sought = (stab_section_names *) names;
  stab_section_names * sought = (stab_section_names *) names;
 
 
  /* Check for section names for which stabsect_name is a prefix, to
  /* Check for section names for which stabsect_name is a prefix, to
     handle .stab.N, etc.  */
     handle .stab.N, etc.  */
  len = strlen (sought->section_name);
  len = strlen (sought->section_name);
 
 
  /* If the prefix matches, and the files section name ends with a
  /* If the prefix matches, and the files section name ends with a
     nul or a digit, then we match.  I.e., we want either an exact
     nul or a digit, then we match.  I.e., we want either an exact
     match or a section followed by a number.  */
     match or a section followed by a number.  */
  if (strncmp (sought->section_name, section->name, len) == 0
  if (strncmp (sought->section_name, section->name, len) == 0
      && (section->name[len] == 0
      && (section->name[len] == 0
          || (section->name[len] == '.' && ISDIGIT (section->name[len + 1]))))
          || (section->name[len] == '.' && ISDIGIT (section->name[len + 1]))))
    {
    {
      if (strtab == NULL)
      if (strtab == NULL)
        strtab = read_section_stabs (abfd, sought->string_section_name,
        strtab = read_section_stabs (abfd, sought->string_section_name,
                                     &stabstr_size);
                                     &stabstr_size);
 
 
      if (strtab)
      if (strtab)
        {
        {
          stabs = (bfd_byte *) read_section_stabs (abfd, section->name,
          stabs = (bfd_byte *) read_section_stabs (abfd, section->name,
                                                   &stab_size);
                                                   &stab_size);
          if (stabs)
          if (stabs)
            print_section_stabs (abfd, section->name, &sought->string_offset);
            print_section_stabs (abfd, section->name, &sought->string_offset);
        }
        }
    }
    }
}
}
 
 
static void
static void
dump_stabs_section (bfd *abfd, char *stabsect_name, char *strsect_name)
dump_stabs_section (bfd *abfd, char *stabsect_name, char *strsect_name)
{
{
  stab_section_names s;
  stab_section_names s;
 
 
  s.section_name = stabsect_name;
  s.section_name = stabsect_name;
  s.string_section_name = strsect_name;
  s.string_section_name = strsect_name;
  s.string_offset = 0;
  s.string_offset = 0;
 
 
  bfd_map_over_sections (abfd, find_stabs_section, & s);
  bfd_map_over_sections (abfd, find_stabs_section, & s);
 
 
  free (strtab);
  free (strtab);
  strtab = NULL;
  strtab = NULL;
}
}
 
 
/* Dump the any sections containing stabs debugging information.  */
/* Dump the any sections containing stabs debugging information.  */
 
 
static void
static void
dump_stabs (bfd *abfd)
dump_stabs (bfd *abfd)
{
{
  dump_stabs_section (abfd, ".stab", ".stabstr");
  dump_stabs_section (abfd, ".stab", ".stabstr");
  dump_stabs_section (abfd, ".stab.excl", ".stab.exclstr");
  dump_stabs_section (abfd, ".stab.excl", ".stab.exclstr");
  dump_stabs_section (abfd, ".stab.index", ".stab.indexstr");
  dump_stabs_section (abfd, ".stab.index", ".stab.indexstr");
 
 
  /* For Darwin.  */
  /* For Darwin.  */
  dump_stabs_section (abfd, "LC_SYMTAB.stabs", "LC_SYMTAB.stabstr");
  dump_stabs_section (abfd, "LC_SYMTAB.stabs", "LC_SYMTAB.stabstr");
 
 
  dump_stabs_section (abfd, "$GDB_SYMBOLS$", "$GDB_STRINGS$");
  dump_stabs_section (abfd, "$GDB_SYMBOLS$", "$GDB_STRINGS$");
}
}


static void
static void
dump_bfd_header (bfd *abfd)
dump_bfd_header (bfd *abfd)
{
{
  char *comma = "";
  char *comma = "";
 
 
  printf (_("architecture: %s, "),
  printf (_("architecture: %s, "),
          bfd_printable_arch_mach (bfd_get_arch (abfd),
          bfd_printable_arch_mach (bfd_get_arch (abfd),
                                   bfd_get_mach (abfd)));
                                   bfd_get_mach (abfd)));
  printf (_("flags 0x%08x:\n"), abfd->flags);
  printf (_("flags 0x%08x:\n"), abfd->flags);
 
 
#define PF(x, y)    if (abfd->flags & x) {printf("%s%s", comma, y); comma=", ";}
#define PF(x, y)    if (abfd->flags & x) {printf("%s%s", comma, y); comma=", ";}
  PF (HAS_RELOC, "HAS_RELOC");
  PF (HAS_RELOC, "HAS_RELOC");
  PF (EXEC_P, "EXEC_P");
  PF (EXEC_P, "EXEC_P");
  PF (HAS_LINENO, "HAS_LINENO");
  PF (HAS_LINENO, "HAS_LINENO");
  PF (HAS_DEBUG, "HAS_DEBUG");
  PF (HAS_DEBUG, "HAS_DEBUG");
  PF (HAS_SYMS, "HAS_SYMS");
  PF (HAS_SYMS, "HAS_SYMS");
  PF (HAS_LOCALS, "HAS_LOCALS");
  PF (HAS_LOCALS, "HAS_LOCALS");
  PF (DYNAMIC, "DYNAMIC");
  PF (DYNAMIC, "DYNAMIC");
  PF (WP_TEXT, "WP_TEXT");
  PF (WP_TEXT, "WP_TEXT");
  PF (D_PAGED, "D_PAGED");
  PF (D_PAGED, "D_PAGED");
  PF (BFD_IS_RELAXABLE, "BFD_IS_RELAXABLE");
  PF (BFD_IS_RELAXABLE, "BFD_IS_RELAXABLE");
  PF (HAS_LOAD_PAGE, "HAS_LOAD_PAGE");
  PF (HAS_LOAD_PAGE, "HAS_LOAD_PAGE");
  printf (_("\nstart address 0x"));
  printf (_("\nstart address 0x"));
  bfd_printf_vma (abfd, abfd->start_address);
  bfd_printf_vma (abfd, abfd->start_address);
  printf ("\n");
  printf ("\n");
}
}
 
 


static void
static void
dump_bfd_private_header (bfd *abfd)
dump_bfd_private_header (bfd *abfd)
{
{
  bfd_print_private_bfd_data (abfd, stdout);
  bfd_print_private_bfd_data (abfd, stdout);
}
}
 
 


/* Display a section in hexadecimal format with associated characters.
/* Display a section in hexadecimal format with associated characters.
   Each line prefixed by the zero padded address.  */
   Each line prefixed by the zero padded address.  */
 
 
static void
static void
dump_section (bfd *abfd, asection *section, void *dummy ATTRIBUTE_UNUSED)
dump_section (bfd *abfd, asection *section, void *dummy ATTRIBUTE_UNUSED)
{
{
  bfd_byte *data = 0;
  bfd_byte *data = 0;
  bfd_size_type datasize;
  bfd_size_type datasize;
  bfd_size_type addr_offset;
  bfd_size_type addr_offset;
  bfd_size_type start_offset;
  bfd_size_type start_offset;
  bfd_size_type stop_offset;
  bfd_size_type stop_offset;
  unsigned int opb = bfd_octets_per_byte (abfd);
  unsigned int opb = bfd_octets_per_byte (abfd);
  /* Bytes per line.  */
  /* Bytes per line.  */
  const int onaline = 16;
  const int onaline = 16;
  char buf[64];
  char buf[64];
  int count;
  int count;
  int width;
  int width;
 
 
  if ((section->flags & SEC_HAS_CONTENTS) == 0)
  if ((section->flags & SEC_HAS_CONTENTS) == 0)
    return;
    return;
 
 
  if (! process_section_p (section))
  if (! process_section_p (section))
    return;
    return;
 
 
  if ((datasize = bfd_section_size (abfd, section)) == 0)
  if ((datasize = bfd_section_size (abfd, section)) == 0)
    return;
    return;
 
 
  /* Compute the address range to display.  */
  /* Compute the address range to display.  */
  if (start_address == (bfd_vma) -1
  if (start_address == (bfd_vma) -1
      || start_address < section->vma)
      || start_address < section->vma)
    start_offset = 0;
    start_offset = 0;
  else
  else
    start_offset = start_address - section->vma;
    start_offset = start_address - section->vma;
 
 
  if (stop_address == (bfd_vma) -1)
  if (stop_address == (bfd_vma) -1)
    stop_offset = datasize / opb;
    stop_offset = datasize / opb;
  else
  else
    {
    {
      if (stop_address < section->vma)
      if (stop_address < section->vma)
        stop_offset = 0;
        stop_offset = 0;
      else
      else
        stop_offset = stop_address - section->vma;
        stop_offset = stop_address - section->vma;
 
 
      if (stop_offset > datasize / opb)
      if (stop_offset > datasize / opb)
        stop_offset = datasize / opb;
        stop_offset = datasize / opb;
    }
    }
 
 
  if (start_offset >= stop_offset)
  if (start_offset >= stop_offset)
    return;
    return;
 
 
  printf (_("Contents of section %s:"), section->name);
  printf (_("Contents of section %s:"), section->name);
  if (display_file_offsets)
  if (display_file_offsets)
    printf (_("  (Starting at file offset: 0x%lx)"),
    printf (_("  (Starting at file offset: 0x%lx)"),
            (unsigned long) (section->filepos + start_offset));
            (unsigned long) (section->filepos + start_offset));
  printf ("\n");
  printf ("\n");
 
 
  data = (bfd_byte *) xmalloc (datasize);
  data = (bfd_byte *) xmalloc (datasize);
 
 
  bfd_get_section_contents (abfd, section, data, 0, datasize);
  bfd_get_section_contents (abfd, section, data, 0, datasize);
 
 
  width = 4;
  width = 4;
 
 
  bfd_sprintf_vma (abfd, buf, start_offset + section->vma);
  bfd_sprintf_vma (abfd, buf, start_offset + section->vma);
  if (strlen (buf) >= sizeof (buf))
  if (strlen (buf) >= sizeof (buf))
    abort ();
    abort ();
 
 
  count = 0;
  count = 0;
  while (buf[count] == '0' && buf[count+1] != '\0')
  while (buf[count] == '0' && buf[count+1] != '\0')
    count++;
    count++;
  count = strlen (buf) - count;
  count = strlen (buf) - count;
  if (count > width)
  if (count > width)
    width = count;
    width = count;
 
 
  bfd_sprintf_vma (abfd, buf, stop_offset + section->vma - 1);
  bfd_sprintf_vma (abfd, buf, stop_offset + section->vma - 1);
  if (strlen (buf) >= sizeof (buf))
  if (strlen (buf) >= sizeof (buf))
    abort ();
    abort ();
 
 
  count = 0;
  count = 0;
  while (buf[count] == '0' && buf[count+1] != '\0')
  while (buf[count] == '0' && buf[count+1] != '\0')
    count++;
    count++;
  count = strlen (buf) - count;
  count = strlen (buf) - count;
  if (count > width)
  if (count > width)
    width = count;
    width = count;
 
 
  for (addr_offset = start_offset;
  for (addr_offset = start_offset;
       addr_offset < stop_offset; addr_offset += onaline / opb)
       addr_offset < stop_offset; addr_offset += onaline / opb)
    {
    {
      bfd_size_type j;
      bfd_size_type j;
 
 
      bfd_sprintf_vma (abfd, buf, (addr_offset + section->vma));
      bfd_sprintf_vma (abfd, buf, (addr_offset + section->vma));
      count = strlen (buf);
      count = strlen (buf);
      if ((size_t) count >= sizeof (buf))
      if ((size_t) count >= sizeof (buf))
        abort ();
        abort ();
 
 
      putchar (' ');
      putchar (' ');
      while (count < width)
      while (count < width)
        {
        {
          putchar ('0');
          putchar ('0');
          count++;
          count++;
        }
        }
      fputs (buf + count - width, stdout);
      fputs (buf + count - width, stdout);
      putchar (' ');
      putchar (' ');
 
 
      for (j = addr_offset * opb;
      for (j = addr_offset * opb;
           j < addr_offset * opb + onaline; j++)
           j < addr_offset * opb + onaline; j++)
        {
        {
          if (j < stop_offset * opb)
          if (j < stop_offset * opb)
            printf ("%02x", (unsigned) (data[j]));
            printf ("%02x", (unsigned) (data[j]));
          else
          else
            printf ("  ");
            printf ("  ");
          if ((j & 3) == 3)
          if ((j & 3) == 3)
            printf (" ");
            printf (" ");
        }
        }
 
 
      printf (" ");
      printf (" ");
      for (j = addr_offset * opb;
      for (j = addr_offset * opb;
           j < addr_offset * opb + onaline; j++)
           j < addr_offset * opb + onaline; j++)
        {
        {
          if (j >= stop_offset * opb)
          if (j >= stop_offset * opb)
            printf (" ");
            printf (" ");
          else
          else
            printf ("%c", ISPRINT (data[j]) ? data[j] : '.');
            printf ("%c", ISPRINT (data[j]) ? data[j] : '.');
        }
        }
      putchar ('\n');
      putchar ('\n');
    }
    }
  free (data);
  free (data);
}
}
 
 
/* Actually display the various requested regions.  */
/* Actually display the various requested regions.  */
 
 
static void
static void
dump_data (bfd *abfd)
dump_data (bfd *abfd)
{
{
  bfd_map_over_sections (abfd, dump_section, NULL);
  bfd_map_over_sections (abfd, dump_section, NULL);
}
}
 
 
/* Should perhaps share code and display with nm?  */
/* Should perhaps share code and display with nm?  */
 
 
static void
static void
dump_symbols (bfd *abfd ATTRIBUTE_UNUSED, bfd_boolean dynamic)
dump_symbols (bfd *abfd ATTRIBUTE_UNUSED, bfd_boolean dynamic)
{
{
  asymbol **current;
  asymbol **current;
  long max;
  long max;
  long count;
  long count;
 
 
  if (dynamic)
  if (dynamic)
    {
    {
      current = dynsyms;
      current = dynsyms;
      max = dynsymcount;
      max = dynsymcount;
      printf ("DYNAMIC SYMBOL TABLE:\n");
      printf ("DYNAMIC SYMBOL TABLE:\n");
    }
    }
  else
  else
    {
    {
      current = syms;
      current = syms;
      max = symcount;
      max = symcount;
      printf ("SYMBOL TABLE:\n");
      printf ("SYMBOL TABLE:\n");
    }
    }
 
 
  if (max == 0)
  if (max == 0)
    printf (_("no symbols\n"));
    printf (_("no symbols\n"));
 
 
  for (count = 0; count < max; count++)
  for (count = 0; count < max; count++)
    {
    {
      bfd *cur_bfd;
      bfd *cur_bfd;
 
 
      if (*current == NULL)
      if (*current == NULL)
        printf (_("no information for symbol number %ld\n"), count);
        printf (_("no information for symbol number %ld\n"), count);
 
 
      else if ((cur_bfd = bfd_asymbol_bfd (*current)) == NULL)
      else if ((cur_bfd = bfd_asymbol_bfd (*current)) == NULL)
        printf (_("could not determine the type of symbol number %ld\n"),
        printf (_("could not determine the type of symbol number %ld\n"),
                count);
                count);
 
 
      else if (process_section_p ((* current)->section)
      else if (process_section_p ((* current)->section)
               && (dump_special_syms
               && (dump_special_syms
                   || !bfd_is_target_special_symbol (cur_bfd, *current)))
                   || !bfd_is_target_special_symbol (cur_bfd, *current)))
        {
        {
          const char *name = (*current)->name;
          const char *name = (*current)->name;
 
 
          if (do_demangle && name != NULL && *name != '\0')
          if (do_demangle && name != NULL && *name != '\0')
            {
            {
              char *alloc;
              char *alloc;
 
 
              /* If we want to demangle the name, we demangle it
              /* If we want to demangle the name, we demangle it
                 here, and temporarily clobber it while calling
                 here, and temporarily clobber it while calling
                 bfd_print_symbol.  FIXME: This is a gross hack.  */
                 bfd_print_symbol.  FIXME: This is a gross hack.  */
              alloc = bfd_demangle (cur_bfd, name, DMGL_ANSI | DMGL_PARAMS);
              alloc = bfd_demangle (cur_bfd, name, DMGL_ANSI | DMGL_PARAMS);
              if (alloc != NULL)
              if (alloc != NULL)
                (*current)->name = alloc;
                (*current)->name = alloc;
              bfd_print_symbol (cur_bfd, stdout, *current,
              bfd_print_symbol (cur_bfd, stdout, *current,
                                bfd_print_symbol_all);
                                bfd_print_symbol_all);
              if (alloc != NULL)
              if (alloc != NULL)
                {
                {
                  (*current)->name = name;
                  (*current)->name = name;
                  free (alloc);
                  free (alloc);
                }
                }
            }
            }
          else
          else
            bfd_print_symbol (cur_bfd, stdout, *current,
            bfd_print_symbol (cur_bfd, stdout, *current,
                              bfd_print_symbol_all);
                              bfd_print_symbol_all);
          printf ("\n");
          printf ("\n");
        }
        }
 
 
      current++;
      current++;
    }
    }
  printf ("\n\n");
  printf ("\n\n");
}
}


static void
static void
dump_reloc_set (bfd *abfd, asection *sec, arelent **relpp, long relcount)
dump_reloc_set (bfd *abfd, asection *sec, arelent **relpp, long relcount)
{
{
  arelent **p;
  arelent **p;
  char *last_filename, *last_functionname;
  char *last_filename, *last_functionname;
  unsigned int last_line;
  unsigned int last_line;
 
 
  /* Get column headers lined up reasonably.  */
  /* Get column headers lined up reasonably.  */
  {
  {
    static int width;
    static int width;
 
 
    if (width == 0)
    if (width == 0)
      {
      {
        char buf[30];
        char buf[30];
 
 
        bfd_sprintf_vma (abfd, buf, (bfd_vma) -1);
        bfd_sprintf_vma (abfd, buf, (bfd_vma) -1);
        width = strlen (buf) - 7;
        width = strlen (buf) - 7;
      }
      }
    printf ("OFFSET %*s TYPE %*s VALUE \n", width, "", 12, "");
    printf ("OFFSET %*s TYPE %*s VALUE \n", width, "", 12, "");
  }
  }
 
 
  last_filename = NULL;
  last_filename = NULL;
  last_functionname = NULL;
  last_functionname = NULL;
  last_line = 0;
  last_line = 0;
 
 
  for (p = relpp; relcount && *p != NULL; p++, relcount--)
  for (p = relpp; relcount && *p != NULL; p++, relcount--)
    {
    {
      arelent *q = *p;
      arelent *q = *p;
      const char *filename, *functionname;
      const char *filename, *functionname;
      unsigned int line;
      unsigned int line;
      const char *sym_name;
      const char *sym_name;
      const char *section_name;
      const char *section_name;
 
 
      if (start_address != (bfd_vma) -1
      if (start_address != (bfd_vma) -1
          && q->address < start_address)
          && q->address < start_address)
        continue;
        continue;
      if (stop_address != (bfd_vma) -1
      if (stop_address != (bfd_vma) -1
          && q->address > stop_address)
          && q->address > stop_address)
        continue;
        continue;
 
 
      if (with_line_numbers
      if (with_line_numbers
          && sec != NULL
          && sec != NULL
          && bfd_find_nearest_line (abfd, sec, syms, q->address,
          && bfd_find_nearest_line (abfd, sec, syms, q->address,
                                    &filename, &functionname, &line))
                                    &filename, &functionname, &line))
        {
        {
          if (functionname != NULL
          if (functionname != NULL
              && (last_functionname == NULL
              && (last_functionname == NULL
                  || strcmp (functionname, last_functionname) != 0))
                  || strcmp (functionname, last_functionname) != 0))
            {
            {
              printf ("%s():\n", functionname);
              printf ("%s():\n", functionname);
              if (last_functionname != NULL)
              if (last_functionname != NULL)
                free (last_functionname);
                free (last_functionname);
              last_functionname = xstrdup (functionname);
              last_functionname = xstrdup (functionname);
            }
            }
 
 
          if (line > 0
          if (line > 0
              && (line != last_line
              && (line != last_line
                  || (filename != NULL
                  || (filename != NULL
                      && last_filename != NULL
                      && last_filename != NULL
                      && strcmp (filename, last_filename) != 0)))
                      && strcmp (filename, last_filename) != 0)))
            {
            {
              printf ("%s:%u\n", filename == NULL ? "???" : filename, line);
              printf ("%s:%u\n", filename == NULL ? "???" : filename, line);
              last_line = line;
              last_line = line;
              if (last_filename != NULL)
              if (last_filename != NULL)
                free (last_filename);
                free (last_filename);
              if (filename == NULL)
              if (filename == NULL)
                last_filename = NULL;
                last_filename = NULL;
              else
              else
                last_filename = xstrdup (filename);
                last_filename = xstrdup (filename);
            }
            }
        }
        }
 
 
      if (q->sym_ptr_ptr && *q->sym_ptr_ptr)
      if (q->sym_ptr_ptr && *q->sym_ptr_ptr)
        {
        {
          sym_name = (*(q->sym_ptr_ptr))->name;
          sym_name = (*(q->sym_ptr_ptr))->name;
          section_name = (*(q->sym_ptr_ptr))->section->name;
          section_name = (*(q->sym_ptr_ptr))->section->name;
        }
        }
      else
      else
        {
        {
          sym_name = NULL;
          sym_name = NULL;
          section_name = NULL;
          section_name = NULL;
        }
        }
 
 
      bfd_printf_vma (abfd, q->address);
      bfd_printf_vma (abfd, q->address);
      if (q->howto == NULL)
      if (q->howto == NULL)
        printf (" *unknown*         ");
        printf (" *unknown*         ");
      else if (q->howto->name)
      else if (q->howto->name)
        printf (" %-16s  ", q->howto->name);
        printf (" %-16s  ", q->howto->name);
      else
      else
        printf (" %-16d  ", q->howto->type);
        printf (" %-16d  ", q->howto->type);
 
 
      if (sym_name)
      if (sym_name)
        {
        {
          objdump_print_symname (abfd, NULL, *q->sym_ptr_ptr);
          objdump_print_symname (abfd, NULL, *q->sym_ptr_ptr);
        }
        }
      else
      else
        {
        {
          if (section_name == NULL)
          if (section_name == NULL)
            section_name = "*unknown*";
            section_name = "*unknown*";
          printf ("[%s]", section_name);
          printf ("[%s]", section_name);
        }
        }
 
 
      if (q->addend)
      if (q->addend)
        {
        {
          printf ("+0x");
          printf ("+0x");
          bfd_printf_vma (abfd, q->addend);
          bfd_printf_vma (abfd, q->addend);
        }
        }
 
 
      printf ("\n");
      printf ("\n");
    }
    }
}
}
 
 
static void
static void
dump_relocs_in_section (bfd *abfd,
dump_relocs_in_section (bfd *abfd,
                        asection *section,
                        asection *section,
                        void *dummy ATTRIBUTE_UNUSED)
                        void *dummy ATTRIBUTE_UNUSED)
{
{
  arelent **relpp;
  arelent **relpp;
  long relcount;
  long relcount;
  long relsize;
  long relsize;
 
 
  if (   bfd_is_abs_section (section)
  if (   bfd_is_abs_section (section)
      || bfd_is_und_section (section)
      || bfd_is_und_section (section)
      || bfd_is_com_section (section)
      || bfd_is_com_section (section)
      || (! process_section_p (section))
      || (! process_section_p (section))
      || ((section->flags & SEC_RELOC) == 0))
      || ((section->flags & SEC_RELOC) == 0))
    return;
    return;
 
 
  relsize = bfd_get_reloc_upper_bound (abfd, section);
  relsize = bfd_get_reloc_upper_bound (abfd, section);
  if (relsize < 0)
  if (relsize < 0)
    bfd_fatal (bfd_get_filename (abfd));
    bfd_fatal (bfd_get_filename (abfd));
 
 
  printf ("RELOCATION RECORDS FOR [%s]:", section->name);
  printf ("RELOCATION RECORDS FOR [%s]:", section->name);
 
 
  if (relsize == 0)
  if (relsize == 0)
    {
    {
      printf (" (none)\n\n");
      printf (" (none)\n\n");
      return;
      return;
    }
    }
 
 
  relpp = (arelent **) xmalloc (relsize);
  relpp = (arelent **) xmalloc (relsize);
  relcount = bfd_canonicalize_reloc (abfd, section, relpp, syms);
  relcount = bfd_canonicalize_reloc (abfd, section, relpp, syms);
 
 
  if (relcount < 0)
  if (relcount < 0)
    bfd_fatal (bfd_get_filename (abfd));
    bfd_fatal (bfd_get_filename (abfd));
  else if (relcount == 0)
  else if (relcount == 0)
    printf (" (none)\n\n");
    printf (" (none)\n\n");
  else
  else
    {
    {
      printf ("\n");
      printf ("\n");
      dump_reloc_set (abfd, section, relpp, relcount);
      dump_reloc_set (abfd, section, relpp, relcount);
      printf ("\n\n");
      printf ("\n\n");
    }
    }
  free (relpp);
  free (relpp);
}
}
 
 
static void
static void
dump_relocs (bfd *abfd)
dump_relocs (bfd *abfd)
{
{
  bfd_map_over_sections (abfd, dump_relocs_in_section, NULL);
  bfd_map_over_sections (abfd, dump_relocs_in_section, NULL);
}
}
 
 
static void
static void
dump_dynamic_relocs (bfd *abfd)
dump_dynamic_relocs (bfd *abfd)
{
{
  long relsize;
  long relsize;
  arelent **relpp;
  arelent **relpp;
  long relcount;
  long relcount;
 
 
  relsize = bfd_get_dynamic_reloc_upper_bound (abfd);
  relsize = bfd_get_dynamic_reloc_upper_bound (abfd);
  if (relsize < 0)
  if (relsize < 0)
    bfd_fatal (bfd_get_filename (abfd));
    bfd_fatal (bfd_get_filename (abfd));
 
 
  printf ("DYNAMIC RELOCATION RECORDS");
  printf ("DYNAMIC RELOCATION RECORDS");
 
 
  if (relsize == 0)
  if (relsize == 0)
    printf (" (none)\n\n");
    printf (" (none)\n\n");
  else
  else
    {
    {
      relpp = (arelent **) xmalloc (relsize);
      relpp = (arelent **) xmalloc (relsize);
      relcount = bfd_canonicalize_dynamic_reloc (abfd, relpp, dynsyms);
      relcount = bfd_canonicalize_dynamic_reloc (abfd, relpp, dynsyms);
 
 
      if (relcount < 0)
      if (relcount < 0)
        bfd_fatal (bfd_get_filename (abfd));
        bfd_fatal (bfd_get_filename (abfd));
      else if (relcount == 0)
      else if (relcount == 0)
        printf (" (none)\n\n");
        printf (" (none)\n\n");
      else
      else
        {
        {
          printf ("\n");
          printf ("\n");
          dump_reloc_set (abfd, NULL, relpp, relcount);
          dump_reloc_set (abfd, NULL, relpp, relcount);
          printf ("\n\n");
          printf ("\n\n");
        }
        }
      free (relpp);
      free (relpp);
    }
    }
}
}
 
 
/* Creates a table of paths, to search for source files.  */
/* Creates a table of paths, to search for source files.  */
 
 
static void
static void
add_include_path (const char *path)
add_include_path (const char *path)
{
{
  if (path[0] == 0)
  if (path[0] == 0)
    return;
    return;
  include_path_count++;
  include_path_count++;
  include_paths = (const char **)
  include_paths = (const char **)
      xrealloc (include_paths, include_path_count * sizeof (*include_paths));
      xrealloc (include_paths, include_path_count * sizeof (*include_paths));
#ifdef HAVE_DOS_BASED_FILE_SYSTEM
#ifdef HAVE_DOS_BASED_FILE_SYSTEM
  if (path[1] == ':' && path[2] == 0)
  if (path[1] == ':' && path[2] == 0)
    path = concat (path, ".", (const char *) 0);
    path = concat (path, ".", (const char *) 0);
#endif
#endif
  include_paths[include_path_count - 1] = path;
  include_paths[include_path_count - 1] = path;
}
}
 
 
static void
static void
adjust_addresses (bfd *abfd ATTRIBUTE_UNUSED,
adjust_addresses (bfd *abfd ATTRIBUTE_UNUSED,
                  asection *section,
                  asection *section,
                  void *arg)
                  void *arg)
{
{
  if ((section->flags & SEC_DEBUGGING) == 0)
  if ((section->flags & SEC_DEBUGGING) == 0)
    {
    {
      bfd_boolean *has_reloc_p = (bfd_boolean *) arg;
      bfd_boolean *has_reloc_p = (bfd_boolean *) arg;
      section->vma += adjust_section_vma;
      section->vma += adjust_section_vma;
      if (*has_reloc_p)
      if (*has_reloc_p)
        section->lma += adjust_section_vma;
        section->lma += adjust_section_vma;
    }
    }
}
}
 
 
/* Dump selected contents of ABFD.  */
/* Dump selected contents of ABFD.  */
 
 
static void
static void
dump_bfd (bfd *abfd)
dump_bfd (bfd *abfd)
{
{
  /* If we are adjusting section VMA's, change them all now.  Changing
  /* If we are adjusting section VMA's, change them all now.  Changing
     the BFD information is a hack.  However, we must do it, or
     the BFD information is a hack.  However, we must do it, or
     bfd_find_nearest_line will not do the right thing.  */
     bfd_find_nearest_line will not do the right thing.  */
  if (adjust_section_vma != 0)
  if (adjust_section_vma != 0)
    {
    {
      bfd_boolean has_reloc = (abfd->flags & HAS_RELOC);
      bfd_boolean has_reloc = (abfd->flags & HAS_RELOC);
      bfd_map_over_sections (abfd, adjust_addresses, &has_reloc);
      bfd_map_over_sections (abfd, adjust_addresses, &has_reloc);
    }
    }
 
 
  if (! dump_debugging_tags)
  if (! dump_debugging_tags)
    printf (_("\n%s:     file format %s\n"), bfd_get_filename (abfd),
    printf (_("\n%s:     file format %s\n"), bfd_get_filename (abfd),
            abfd->xvec->name);
            abfd->xvec->name);
  if (dump_ar_hdrs)
  if (dump_ar_hdrs)
    print_arelt_descr (stdout, abfd, TRUE);
    print_arelt_descr (stdout, abfd, TRUE);
  if (dump_file_header)
  if (dump_file_header)
    dump_bfd_header (abfd);
    dump_bfd_header (abfd);
  if (dump_private_headers)
  if (dump_private_headers)
    dump_bfd_private_header (abfd);
    dump_bfd_private_header (abfd);
  if (! dump_debugging_tags)
  if (! dump_debugging_tags)
    putchar ('\n');
    putchar ('\n');
  if (dump_section_headers)
  if (dump_section_headers)
    dump_headers (abfd);
    dump_headers (abfd);
 
 
  if (dump_symtab
  if (dump_symtab
      || dump_reloc_info
      || dump_reloc_info
      || disassemble
      || disassemble
      || dump_debugging
      || dump_debugging
      || dump_dwarf_section_info)
      || dump_dwarf_section_info)
    syms = slurp_symtab (abfd);
    syms = slurp_symtab (abfd);
  if (dump_dynamic_symtab || dump_dynamic_reloc_info
  if (dump_dynamic_symtab || dump_dynamic_reloc_info
      || (disassemble && bfd_get_dynamic_symtab_upper_bound (abfd) > 0))
      || (disassemble && bfd_get_dynamic_symtab_upper_bound (abfd) > 0))
    dynsyms = slurp_dynamic_symtab (abfd);
    dynsyms = slurp_dynamic_symtab (abfd);
  if (disassemble)
  if (disassemble)
    {
    {
      synthcount = bfd_get_synthetic_symtab (abfd, symcount, syms,
      synthcount = bfd_get_synthetic_symtab (abfd, symcount, syms,
                                             dynsymcount, dynsyms, &synthsyms);
                                             dynsymcount, dynsyms, &synthsyms);
      if (synthcount < 0)
      if (synthcount < 0)
        synthcount = 0;
        synthcount = 0;
    }
    }
 
 
  if (dump_symtab)
  if (dump_symtab)
    dump_symbols (abfd, FALSE);
    dump_symbols (abfd, FALSE);
  if (dump_dynamic_symtab)
  if (dump_dynamic_symtab)
    dump_symbols (abfd, TRUE);
    dump_symbols (abfd, TRUE);
  if (dump_dwarf_section_info)
  if (dump_dwarf_section_info)
    dump_dwarf (abfd);
    dump_dwarf (abfd);
  if (dump_stab_section_info)
  if (dump_stab_section_info)
    dump_stabs (abfd);
    dump_stabs (abfd);
  if (dump_reloc_info && ! disassemble)
  if (dump_reloc_info && ! disassemble)
    dump_relocs (abfd);
    dump_relocs (abfd);
  if (dump_dynamic_reloc_info && ! disassemble)
  if (dump_dynamic_reloc_info && ! disassemble)
    dump_dynamic_relocs (abfd);
    dump_dynamic_relocs (abfd);
  if (dump_section_contents)
  if (dump_section_contents)
    dump_data (abfd);
    dump_data (abfd);
  if (disassemble)
  if (disassemble)
    disassemble_data (abfd);
    disassemble_data (abfd);
 
 
  if (dump_debugging)
  if (dump_debugging)
    {
    {
      void *dhandle;
      void *dhandle;
 
 
      dhandle = read_debugging_info (abfd, syms, symcount, TRUE);
      dhandle = read_debugging_info (abfd, syms, symcount, TRUE);
      if (dhandle != NULL)
      if (dhandle != NULL)
        {
        {
          if (!print_debugging_info (stdout, dhandle, abfd, syms,
          if (!print_debugging_info (stdout, dhandle, abfd, syms,
                                     bfd_demangle,
                                     bfd_demangle,
                                     dump_debugging_tags ? TRUE : FALSE))
                                     dump_debugging_tags ? TRUE : FALSE))
            {
            {
              non_fatal (_("%s: printing debugging information failed"),
              non_fatal (_("%s: printing debugging information failed"),
                         bfd_get_filename (abfd));
                         bfd_get_filename (abfd));
              exit_status = 1;
              exit_status = 1;
            }
            }
        }
        }
      /* PR 6483: If there was no STABS or IEEE debug
      /* PR 6483: If there was no STABS or IEEE debug
         info in the file, try DWARF instead.  */
         info in the file, try DWARF instead.  */
      else if (! dump_dwarf_section_info)
      else if (! dump_dwarf_section_info)
        {
        {
          dump_dwarf (abfd);
          dump_dwarf (abfd);
        }
        }
    }
    }
 
 
  if (syms)
  if (syms)
    {
    {
      free (syms);
      free (syms);
      syms = NULL;
      syms = NULL;
    }
    }
 
 
  if (dynsyms)
  if (dynsyms)
    {
    {
      free (dynsyms);
      free (dynsyms);
      dynsyms = NULL;
      dynsyms = NULL;
    }
    }
 
 
  if (synthsyms)
  if (synthsyms)
    {
    {
      free (synthsyms);
      free (synthsyms);
      synthsyms = NULL;
      synthsyms = NULL;
    }
    }
 
 
  symcount = 0;
  symcount = 0;
  dynsymcount = 0;
  dynsymcount = 0;
  synthcount = 0;
  synthcount = 0;
}
}
 
 
static void
static void
display_bfd (bfd *abfd)
display_bfd (bfd *abfd)
{
{
  char **matching;
  char **matching;
 
 
  if (bfd_check_format_matches (abfd, bfd_object, &matching))
  if (bfd_check_format_matches (abfd, bfd_object, &matching))
    {
    {
      dump_bfd (abfd);
      dump_bfd (abfd);
      return;
      return;
    }
    }
 
 
  if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
  if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
    {
    {
      nonfatal (bfd_get_filename (abfd));
      nonfatal (bfd_get_filename (abfd));
      list_matching_formats (matching);
      list_matching_formats (matching);
      free (matching);
      free (matching);
      return;
      return;
    }
    }
 
 
  if (bfd_get_error () != bfd_error_file_not_recognized)
  if (bfd_get_error () != bfd_error_file_not_recognized)
    {
    {
      nonfatal (bfd_get_filename (abfd));
      nonfatal (bfd_get_filename (abfd));
      return;
      return;
    }
    }
 
 
  if (bfd_check_format_matches (abfd, bfd_core, &matching))
  if (bfd_check_format_matches (abfd, bfd_core, &matching))
    {
    {
      dump_bfd (abfd);
      dump_bfd (abfd);
      return;
      return;
    }
    }
 
 
  nonfatal (bfd_get_filename (abfd));
  nonfatal (bfd_get_filename (abfd));
 
 
  if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
  if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
    {
    {
      list_matching_formats (matching);
      list_matching_formats (matching);
      free (matching);
      free (matching);
    }
    }
}
}
 
 
static void
static void
display_file (char *filename, char *target)
display_file (char *filename, char *target)
{
{
  bfd *file;
  bfd *file;
  bfd *arfile = NULL;
  bfd *arfile = NULL;
 
 
  if (get_file_size (filename) < 1)
  if (get_file_size (filename) < 1)
    {
    {
      exit_status = 1;
      exit_status = 1;
      return;
      return;
    }
    }
 
 
  file = bfd_openr (filename, target);
  file = bfd_openr (filename, target);
  if (file == NULL)
  if (file == NULL)
    {
    {
      nonfatal (filename);
      nonfatal (filename);
      return;
      return;
    }
    }
 
 
  /* If the file is an archive, process all of its elements.  */
  /* If the file is an archive, process all of its elements.  */
  if (bfd_check_format (file, bfd_archive))
  if (bfd_check_format (file, bfd_archive))
    {
    {
      bfd *last_arfile = NULL;
      bfd *last_arfile = NULL;
 
 
      printf (_("In archive %s:\n"), bfd_get_filename (file));
      printf (_("In archive %s:\n"), bfd_get_filename (file));
      for (;;)
      for (;;)
        {
        {
          bfd_set_error (bfd_error_no_error);
          bfd_set_error (bfd_error_no_error);
 
 
          arfile = bfd_openr_next_archived_file (file, arfile);
          arfile = bfd_openr_next_archived_file (file, arfile);
          if (arfile == NULL)
          if (arfile == NULL)
            {
            {
              if (bfd_get_error () != bfd_error_no_more_archived_files)
              if (bfd_get_error () != bfd_error_no_more_archived_files)
                nonfatal (bfd_get_filename (file));
                nonfatal (bfd_get_filename (file));
              break;
              break;
            }
            }
 
 
          display_bfd (arfile);
          display_bfd (arfile);
 
 
          if (last_arfile != NULL)
          if (last_arfile != NULL)
            bfd_close (last_arfile);
            bfd_close (last_arfile);
          last_arfile = arfile;
          last_arfile = arfile;
        }
        }
 
 
      if (last_arfile != NULL)
      if (last_arfile != NULL)
        bfd_close (last_arfile);
        bfd_close (last_arfile);
    }
    }
  else
  else
    display_bfd (file);
    display_bfd (file);
 
 
  bfd_close (file);
  bfd_close (file);
}
}


int
int
main (int argc, char **argv)
main (int argc, char **argv)
{
{
  int c;
  int c;
  char *target = default_target;
  char *target = default_target;
  bfd_boolean seenflag = FALSE;
  bfd_boolean seenflag = FALSE;
 
 
#if defined (HAVE_SETLOCALE)
#if defined (HAVE_SETLOCALE)
#if defined (HAVE_LC_MESSAGES)
#if defined (HAVE_LC_MESSAGES)
  setlocale (LC_MESSAGES, "");
  setlocale (LC_MESSAGES, "");
#endif
#endif
  setlocale (LC_CTYPE, "");
  setlocale (LC_CTYPE, "");
#endif
#endif
 
 
  bindtextdomain (PACKAGE, LOCALEDIR);
  bindtextdomain (PACKAGE, LOCALEDIR);
  textdomain (PACKAGE);
  textdomain (PACKAGE);
 
 
  program_name = *argv;
  program_name = *argv;
  xmalloc_set_program_name (program_name);
  xmalloc_set_program_name (program_name);
 
 
  START_PROGRESS (program_name, 0);
  START_PROGRESS (program_name, 0);
 
 
  expandargv (&argc, &argv);
  expandargv (&argc, &argv);
 
 
  bfd_init ();
  bfd_init ();
  set_default_bfd_target ();
  set_default_bfd_target ();
 
 
  while ((c = getopt_long (argc, argv,
  while ((c = getopt_long (argc, argv,
                           "pib:m:M:VvCdDlfFaHhrRtTxsSI:j:wE:zgeGW::",
                           "pib:m:M:VvCdDlfFaHhrRtTxsSI:j:wE:zgeGW::",
                           long_options, (int *) 0))
                           long_options, (int *) 0))
         != EOF)
         != EOF)
    {
    {
      switch (c)
      switch (c)
        {
        {
        case 0:
        case 0:
          break;                /* We've been given a long option.  */
          break;                /* We've been given a long option.  */
        case 'm':
        case 'm':
          machine = optarg;
          machine = optarg;
          break;
          break;
        case 'M':
        case 'M':
          if (disassembler_options)
          if (disassembler_options)
            /* Ignore potential memory leak for now.  */
            /* Ignore potential memory leak for now.  */
            disassembler_options = concat (disassembler_options, ",",
            disassembler_options = concat (disassembler_options, ",",
                                           optarg, (const char *) NULL);
                                           optarg, (const char *) NULL);
          else
          else
            disassembler_options = optarg;
            disassembler_options = optarg;
          break;
          break;
        case 'j':
        case 'j':
          if (only_used == only_size)
          if (only_used == only_size)
            {
            {
              only_size += 8;
              only_size += 8;
              only = (char **) xrealloc (only, only_size * sizeof (char *));
              only = (char **) xrealloc (only, only_size * sizeof (char *));
            }
            }
          only [only_used++] = optarg;
          only [only_used++] = optarg;
          break;
          break;
        case 'F':
        case 'F':
          display_file_offsets = TRUE;
          display_file_offsets = TRUE;
          break;
          break;
        case 'l':
        case 'l':
          with_line_numbers = TRUE;
          with_line_numbers = TRUE;
          break;
          break;
        case 'b':
        case 'b':
          target = optarg;
          target = optarg;
          break;
          break;
        case 'C':
        case 'C':
          do_demangle = TRUE;
          do_demangle = TRUE;
          if (optarg != NULL)
          if (optarg != NULL)
            {
            {
              enum demangling_styles style;
              enum demangling_styles style;
 
 
              style = cplus_demangle_name_to_style (optarg);
              style = cplus_demangle_name_to_style (optarg);
              if (style == unknown_demangling)
              if (style == unknown_demangling)
                fatal (_("unknown demangling style `%s'"),
                fatal (_("unknown demangling style `%s'"),
                       optarg);
                       optarg);
 
 
              cplus_demangle_set_style (style);
              cplus_demangle_set_style (style);
            }
            }
          break;
          break;
        case 'w':
        case 'w':
          wide_output = TRUE;
          wide_output = TRUE;
          break;
          break;
        case OPTION_ADJUST_VMA:
        case OPTION_ADJUST_VMA:
          adjust_section_vma = parse_vma (optarg, "--adjust-vma");
          adjust_section_vma = parse_vma (optarg, "--adjust-vma");
          break;
          break;
        case OPTION_START_ADDRESS:
        case OPTION_START_ADDRESS:
          start_address = parse_vma (optarg, "--start-address");
          start_address = parse_vma (optarg, "--start-address");
          if ((stop_address != (bfd_vma) -1) && stop_address <= start_address)
          if ((stop_address != (bfd_vma) -1) && stop_address <= start_address)
            fatal (_("error: the start address should be before the end address"));
            fatal (_("error: the start address should be before the end address"));
          break;
          break;
        case OPTION_STOP_ADDRESS:
        case OPTION_STOP_ADDRESS:
          stop_address = parse_vma (optarg, "--stop-address");
          stop_address = parse_vma (optarg, "--stop-address");
          if ((start_address != (bfd_vma) -1) && stop_address <= start_address)
          if ((start_address != (bfd_vma) -1) && stop_address <= start_address)
            fatal (_("error: the stop address should be after the start address"));
            fatal (_("error: the stop address should be after the start address"));
          break;
          break;
        case OPTION_PREFIX:
        case OPTION_PREFIX:
          prefix = optarg;
          prefix = optarg;
          prefix_length = strlen (prefix);
          prefix_length = strlen (prefix);
          /* Remove an unnecessary trailing '/' */
          /* Remove an unnecessary trailing '/' */
          while (IS_DIR_SEPARATOR (prefix[prefix_length - 1]))
          while (IS_DIR_SEPARATOR (prefix[prefix_length - 1]))
            prefix_length--;
            prefix_length--;
          break;
          break;
        case OPTION_PREFIX_STRIP:
        case OPTION_PREFIX_STRIP:
          prefix_strip = atoi (optarg);
          prefix_strip = atoi (optarg);
          if (prefix_strip < 0)
          if (prefix_strip < 0)
            fatal (_("error: prefix strip must be non-negative"));
            fatal (_("error: prefix strip must be non-negative"));
          break;
          break;
        case OPTION_INSN_WIDTH:
        case OPTION_INSN_WIDTH:
          insn_width = strtoul (optarg, NULL, 0);
          insn_width = strtoul (optarg, NULL, 0);
          if (insn_width <= 0)
          if (insn_width <= 0)
            fatal (_("error: instruction width must be positive"));
            fatal (_("error: instruction width must be positive"));
          break;
          break;
        case 'E':
        case 'E':
          if (strcmp (optarg, "B") == 0)
          if (strcmp (optarg, "B") == 0)
            endian = BFD_ENDIAN_BIG;
            endian = BFD_ENDIAN_BIG;
          else if (strcmp (optarg, "L") == 0)
          else if (strcmp (optarg, "L") == 0)
            endian = BFD_ENDIAN_LITTLE;
            endian = BFD_ENDIAN_LITTLE;
          else
          else
            {
            {
              non_fatal (_("unrecognized -E option"));
              non_fatal (_("unrecognized -E option"));
              usage (stderr, 1);
              usage (stderr, 1);
            }
            }
          break;
          break;
        case OPTION_ENDIAN:
        case OPTION_ENDIAN:
          if (strncmp (optarg, "big", strlen (optarg)) == 0)
          if (strncmp (optarg, "big", strlen (optarg)) == 0)
            endian = BFD_ENDIAN_BIG;
            endian = BFD_ENDIAN_BIG;
          else if (strncmp (optarg, "little", strlen (optarg)) == 0)
          else if (strncmp (optarg, "little", strlen (optarg)) == 0)
            endian = BFD_ENDIAN_LITTLE;
            endian = BFD_ENDIAN_LITTLE;
          else
          else
            {
            {
              non_fatal (_("unrecognized --endian type `%s'"), optarg);
              non_fatal (_("unrecognized --endian type `%s'"), optarg);
              usage (stderr, 1);
              usage (stderr, 1);
            }
            }
          break;
          break;
 
 
        case 'f':
        case 'f':
          dump_file_header = TRUE;
          dump_file_header = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'i':
        case 'i':
          formats_info = TRUE;
          formats_info = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'I':
        case 'I':
          add_include_path (optarg);
          add_include_path (optarg);
          break;
          break;
        case 'p':
        case 'p':
          dump_private_headers = TRUE;
          dump_private_headers = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'x':
        case 'x':
          dump_private_headers = TRUE;
          dump_private_headers = TRUE;
          dump_symtab = TRUE;
          dump_symtab = TRUE;
          dump_reloc_info = TRUE;
          dump_reloc_info = TRUE;
          dump_file_header = TRUE;
          dump_file_header = TRUE;
          dump_ar_hdrs = TRUE;
          dump_ar_hdrs = TRUE;
          dump_section_headers = TRUE;
          dump_section_headers = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 't':
        case 't':
          dump_symtab = TRUE;
          dump_symtab = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'T':
        case 'T':
          dump_dynamic_symtab = TRUE;
          dump_dynamic_symtab = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'd':
        case 'd':
          disassemble = TRUE;
          disassemble = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'z':
        case 'z':
          disassemble_zeroes = TRUE;
          disassemble_zeroes = TRUE;
          break;
          break;
        case 'D':
        case 'D':
          disassemble = TRUE;
          disassemble = TRUE;
          disassemble_all = TRUE;
          disassemble_all = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'S':
        case 'S':
          disassemble = TRUE;
          disassemble = TRUE;
          with_source_code = TRUE;
          with_source_code = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'g':
        case 'g':
          dump_debugging = 1;
          dump_debugging = 1;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'e':
        case 'e':
          dump_debugging = 1;
          dump_debugging = 1;
          dump_debugging_tags = 1;
          dump_debugging_tags = 1;
          do_demangle = TRUE;
          do_demangle = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'W':
        case 'W':
          dump_dwarf_section_info = TRUE;
          dump_dwarf_section_info = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          if (optarg)
          if (optarg)
            dwarf_select_sections_by_letters (optarg);
            dwarf_select_sections_by_letters (optarg);
          else
          else
            dwarf_select_sections_all ();
            dwarf_select_sections_all ();
          break;
          break;
        case OPTION_DWARF:
        case OPTION_DWARF:
          dump_dwarf_section_info = TRUE;
          dump_dwarf_section_info = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          if (optarg)
          if (optarg)
            dwarf_select_sections_by_names (optarg);
            dwarf_select_sections_by_names (optarg);
          else
          else
            dwarf_select_sections_all ();
            dwarf_select_sections_all ();
          break;
          break;
        case 'G':
        case 'G':
          dump_stab_section_info = TRUE;
          dump_stab_section_info = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 's':
        case 's':
          dump_section_contents = TRUE;
          dump_section_contents = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'r':
        case 'r':
          dump_reloc_info = TRUE;
          dump_reloc_info = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'R':
        case 'R':
          dump_dynamic_reloc_info = TRUE;
          dump_dynamic_reloc_info = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'a':
        case 'a':
          dump_ar_hdrs = TRUE;
          dump_ar_hdrs = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'h':
        case 'h':
          dump_section_headers = TRUE;
          dump_section_headers = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
        case 'H':
        case 'H':
          usage (stdout, 0);
          usage (stdout, 0);
          seenflag = TRUE;
          seenflag = TRUE;
        case 'v':
        case 'v':
        case 'V':
        case 'V':
          show_version = TRUE;
          show_version = TRUE;
          seenflag = TRUE;
          seenflag = TRUE;
          break;
          break;
 
 
        default:
        default:
          usage (stderr, 1);
          usage (stderr, 1);
        }
        }
    }
    }
 
 
  if (show_version)
  if (show_version)
    print_version ("objdump");
    print_version ("objdump");
 
 
  if (!seenflag)
  if (!seenflag)
    usage (stderr, 2);
    usage (stderr, 2);
 
 
  if (formats_info)
  if (formats_info)
    exit_status = display_info ();
    exit_status = display_info ();
  else
  else
    {
    {
      if (optind == argc)
      if (optind == argc)
        display_file ("a.out", target);
        display_file ("a.out", target);
      else
      else
        for (; optind < argc;)
        for (; optind < argc;)
          display_file (argv[optind++], target);
          display_file (argv[optind++], target);
    }
    }
 
 
  END_PROGRESS (program_name);
  END_PROGRESS (program_name);
 
 
  return exit_status;
  return exit_status;
}
}
 
 

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.