binutils-gdb/binutils/objdump.c

1144 lines
26 KiB
C
Raw Normal View History

/* objdump.c -- dump information about an object file.
Copyright 1990, 1991, 1992, 1993 Free Software Foundation, Inc.
1991-03-22 05:29:06 +08:00
This file is part of GNU Binutils.
1991-03-22 05:29:06 +08:00
This program is free software; you can redistribute it and/or modify
1991-03-22 05:29:06 +08:00
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
1991-03-22 05:29:06 +08:00
any later version.
This program is distributed in the hope that it will be useful,
1991-03-22 05:29:06 +08:00
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
1991-03-22 05:29:06 +08:00
#include "bfd.h"
#include "sysdep.h"
1991-03-22 05:29:06 +08:00
#include "getopt.h"
#include <stdio.h>
#include <ctype.h>
#include "dis-asm.h"
1991-03-22 05:29:06 +08:00
#define ELF_STAB_DISPLAY /* This code works, but uses internal
bfd and elf stuff. Flip this define
off if you need to just use generic
BFD interfaces. */
#ifdef ELF_STAB_DISPLAY
/* Internal headers for the ELF .stab-dump code - sorry. */
#define BYTES_IN_WORD 32
#include "aout/aout64.h"
#include "elf/internal.h"
extern Elf_Internal_Shdr *bfd_elf_find_section();
#endif /* ELF_STAB_DISPLAY */
1991-04-19 06:14:53 +09:00
extern char *xmalloc ();
1993-04-27 09:39:38 +08:00
extern int fprintf PARAMS ((FILE *, CONST char *, ...));
1991-03-22 05:29:06 +08:00
char *default_target = NULL; /* default at runtime */
extern *program_version;
1991-03-22 05:29:06 +08:00
char *program_name = NULL;
int show_version = 0; /* show the version number */
1991-03-22 05:29:06 +08:00
int dump_section_contents; /* -s */
int dump_section_headers; /* -h */
boolean dump_file_header; /* -f */
int dump_symtab; /* -t */
int dump_reloc_info; /* -r */
int dump_ar_hdrs; /* -a */
int with_line_numbers; /* -l */
int dump_stab_section_info; /* -stabs */
boolean disassemble; /* -d */
boolean info; /* -i */
1991-03-22 05:29:06 +08:00
char *only;
char *machine = (char *) NULL;
asymbol **syms;
1991-03-22 05:29:06 +08:00
unsigned int storage;
unsigned int symcount = 0;
/* Forward declarations. */
static void
display_file PARAMS ((char *filename, char *target));
static void
dump_data PARAMS ((bfd *abfd));
static void
dump_relocs PARAMS ((bfd *abfd));
static void
dump_symbols PARAMS ((bfd *abfd));
1991-03-22 05:29:06 +08:00
void
usage (stream, status)
FILE *stream;
int status;
1991-03-22 05:29:06 +08:00
{
fprintf (stream, "\
Usage: %s [-ahifdrtxsl] [-m machine] [-j section_name] [-b bfdname]\n\
[--syms] [--reloc] [--header] [--version] [--help] objfile...\n",
program_name);
exit (status);
1991-03-22 05:29:06 +08:00
}
static struct option long_options[]=
{
{"syms", no_argument, &dump_symtab, 1},
{"reloc", no_argument, &dump_reloc_info, 1},
{"header", no_argument, &dump_section_headers, 1},
{"version", no_argument, &show_version, 1},
{"help", no_argument, 0, 'H'},
#ifdef ELF_STAB_DISPLAY
{"stabs", no_argument, &dump_stab_section_info, 1},
#endif
{0, no_argument, 0, 0}
};
1991-03-22 05:29:06 +08:00
static void
dump_headers (abfd)
bfd *abfd;
1991-03-22 05:29:06 +08:00
{
asection *section;
1991-03-22 05:29:06 +08:00
for (section = abfd->sections;
section != (asection *) NULL;
section = section->next)
{
char *comma = "";
1991-03-22 05:29:06 +08:00
#define PF(x,y) \
if (section->flags & x) { printf("%s%s",comma,y); comma = ", "; }
printf ("SECTION %d [%s]\t: size %08x",
section->index,
section->name,
(unsigned) bfd_get_section_size_before_reloc (section));
printf (" vma ");
printf_vma (section->vma);
printf (" align 2**%u\n ",
section->alignment_power);
PF (SEC_ALLOC, "ALLOC");
PF (SEC_CONSTRUCTOR, "CONSTRUCTOR");
PF (SEC_CONSTRUCTOR_TEXT, "CONSTRUCTOR TEXT");
PF (SEC_CONSTRUCTOR_DATA, "CONSTRUCTOR DATA");
PF (SEC_CONSTRUCTOR_BSS, "CONSTRUCTOR BSS");
PF (SEC_LOAD, "LOAD");
PF (SEC_RELOC, "RELOC");
PF (SEC_BALIGN, "BALIGN");
PF (SEC_READONLY, "READONLY");
PF (SEC_CODE, "CODE");
PF (SEC_DATA, "DATA");
PF (SEC_ROM, "ROM");
printf ("\n");
1991-03-22 05:29:06 +08:00
#undef PF
}
1991-03-22 05:29:06 +08:00
}
static asymbol **
DEFUN (slurp_symtab, (abfd),
bfd * abfd)
1991-03-22 05:29:06 +08:00
{
asymbol **sy = (asymbol **) NULL;
1991-03-22 05:29:06 +08:00
if (!(bfd_get_file_flags (abfd) & HAS_SYMS))
{
(void) printf ("No symbols in \"%s\".\n", bfd_get_filename (abfd));
return (NULL);
}
storage = get_symtab_upper_bound (abfd);
if (storage)
{
sy = (asymbol **) malloc (storage);
if (sy == NULL)
{
fprintf (stderr, "%s: out of memory.\n", program_name);
exit (1);
}
}
symcount = bfd_canonicalize_symtab (abfd, sy);
if (symcount <= 0)
{
fprintf (stderr, "%s: Bad symbol table in \"%s\".\n",
program_name, bfd_get_filename (abfd));
exit (1);
}
return sy;
1991-03-22 05:29:06 +08:00
}
/* Filter out (in place) symbols that are useless for dis-assemble.
Return count of useful symbols. */
int remove_useless_symbols (syms, count)
asymbol **syms;
int count;
{
register asymbol **in_ptr = syms;
register asymbol **out_ptr = syms;
while ( --count >= 0 )
{
asymbol *sym = *in_ptr++;
if (sym->name == NULL || sym->name[0] == '\0')
continue;
if (sym->flags & (BSF_DEBUGGING))
continue;
if (sym->section == &bfd_und_section
|| bfd_is_com_section (sym->section))
continue;
*out_ptr++ = sym;
}
return out_ptr - syms;
}
1991-03-22 05:29:06 +08:00
/* Sort symbols into value order */
static int
comp (ap, bp)
1992-06-13 13:24:22 +08:00
PTR ap;
PTR bp;
1991-03-22 05:29:06 +08:00
{
1992-06-13 13:24:22 +08:00
asymbol *a = *(asymbol **)ap;
asymbol *b = *(asymbol **)bp;
1991-03-22 05:29:06 +08:00
if (a->value > b->value)
return 1;
else if (a->value < b->value)
return -1;
1991-03-22 05:29:06 +08:00
if (a->section > b->section)
return 1;
else if (a->section < b->section)
return -1;
return 0;
1991-03-22 05:29:06 +08:00
}
/* Print the supplied address symbolically if possible */
void
objdump_print_address (vma, info)
bfd_vma vma;
struct disassemble_info *info;
1991-03-22 05:29:06 +08:00
{
/* Perform a binary search looking for the closest symbol to
the required value */
unsigned int min = 0;
unsigned int max = symcount;
unsigned int thisplace = 1;
unsigned int oldthisplace;
1991-03-22 05:29:06 +08:00
int vardiff;
fprintf_vma (info->stream, vma);
if (symcount > 0)
{
while (true)
{
asymbol *sym; asection *sym_sec;
oldthisplace = thisplace;
thisplace = (max + min) / 2;
if (thisplace == oldthisplace)
break;
sym = syms[thisplace];
vardiff = sym->value - vma;
sym_sec = sym->section;
if (vardiff > 0)
max = thisplace;
else if (vardiff < 0)
min = thisplace;
else
goto found;
}
/* We've run out of places to look, print the symbol before this one
see if this or the symbol before describes this location the best */
if (thisplace != 0)
{
if (syms[thisplace - 1]->value - vma >
syms[thisplace]->value - vma)
{
/* Previous symbol is in correct section and is closer */
thisplace--;
}
}
found:
fprintf (info->stream, " <%s", syms[thisplace]->name);
if (syms[thisplace]->value > vma)
{
char buf[30], *p = buf;
sprintf_vma (buf, syms[thisplace]->value - vma);
while (*p == '0')
p++;
fprintf (info->stream, "-%s", p);
}
else if (vma > syms[thisplace]->value)
{
char buf[30], *p = buf;
sprintf_vma (buf, vma - syms[thisplace]->value);
while (*p == '0')
p++;
fprintf (info->stream, "+%s", p);
}
fprintf (info->stream, ">");
1991-03-22 05:29:06 +08:00
}
}
void
disassemble_data (abfd)
bfd *abfd;
1991-03-22 05:29:06 +08:00
{
bfd_byte *data = NULL;
bfd_arch_info_type *info;
bfd_size_type datasize = 0;
bfd_size_type i;
unsigned int (*print) ()= 0; /* Old style */
disassembler_ftype disassemble = 0; /* New style */
1991-03-22 05:29:06 +08:00
enum bfd_architecture a;
struct disassemble_info disasm_info;
int prevline;
CONST char *prev_function = "";
1991-03-22 05:29:06 +08:00
asection *section;
1991-03-22 05:29:06 +08:00
/* Replace symbol section relative values with abs values */
boolean done_dot = false;
INIT_DISASSEMBLE_INFO(disasm_info, stdout);
disasm_info.print_address_func = objdump_print_address;
for (i = 0; i < symcount; i++)
{
1991-03-22 05:29:06 +08:00
syms[i]->value += syms[i]->section->vma;
}
1991-03-22 05:29:06 +08:00
symcount = remove_useless_symbols (syms, symcount);
1991-03-22 05:29:06 +08:00
/* Sort the symbols into section and symbol order */
(void) qsort (syms, symcount, sizeof (asymbol *), comp);
1991-03-22 05:29:06 +08:00
if (machine != (char *) NULL)
{
info = bfd_scan_arch (machine);
if (info == 0)
{
fprintf (stderr, "%s: Can't use supplied machine %s\n",
program_name,
machine);
exit (1);
}
abfd->arch_info = info;
1991-03-22 05:29:06 +08:00
}
/* See if we can disassemble using bfd */
if (abfd->arch_info->disassemble)
{
print = abfd->arch_info->disassemble;
}
else
{
a = bfd_get_arch (abfd);
switch (a)
{
case bfd_arch_sparc:
disassemble = print_insn_sparc;
break;
case bfd_arch_z8k:
if (bfd_get_mach(abfd) == bfd_mach_z8001)
disassemble = print_insn_z8001;
else
disassemble = print_insn_z8002;
break;
case bfd_arch_i386:
disassemble = print_insn_i386;
break;
case bfd_arch_h8500:
disassemble = print_insn_h8500;
break;
case bfd_arch_h8300:
if (bfd_get_mach(abfd) == bfd_mach_h8300h)
disassemble = print_insn_h8300h;
else
disassemble = print_insn_h8300;
break;
1993-04-27 09:39:38 +08:00
case bfd_arch_sh:
disassemble = print_insn_sh;
break;
case bfd_arch_alpha:
disassemble = print_insn_alpha;
break;
case bfd_arch_m68k:
disassemble = print_insn_m68k;
break;
case bfd_arch_a29k:
/* As far as I know we only handle big-endian 29k objects. */
disassemble = print_insn_big_a29k;
break;
case bfd_arch_i960:
disassemble = print_insn_i960;
break;
case bfd_arch_m88k:
disassemble = print_insn_m88k;
break;
case bfd_arch_mips:
if (abfd->xvec->byteorder_big_p)
disassemble = print_insn_big_mips;
else
disassemble = print_insn_little_mips;
break;
default:
fprintf (stderr, "%s: Can't disassemble for architecture %s\n",
program_name,
bfd_printable_arch_mach (bfd_get_arch (abfd), 0));
exit (1);
}
1991-03-22 05:29:06 +08:00
}
1991-03-22 05:29:06 +08:00
for (section = abfd->sections;
section != (asection *) NULL;
section = section->next)
{
1991-03-22 05:29:06 +08:00
if ((section->flags & SEC_LOAD)
&& (only == (char *) NULL || strcmp (only, section->name) == 0))
{
printf ("Disassembly of section %s:\n", section->name);
1991-03-22 05:29:06 +08:00
if (bfd_get_section_size_before_reloc (section) == 0)
continue;
1991-03-22 05:29:06 +08:00
data = (bfd_byte *) malloc (bfd_get_section_size_before_reloc (section));
1991-03-22 05:29:06 +08:00
if (data == (bfd_byte *) NULL)
{
fprintf (stderr, "%s: memory exhausted.\n", program_name);
exit (1);
}
datasize = bfd_get_section_size_before_reloc (section);
1991-03-22 05:29:06 +08:00
bfd_get_section_contents (abfd, section, data, 0, bfd_get_section_size_before_reloc (section));
1991-03-22 05:29:06 +08:00
disasm_info.buffer = data;
disasm_info.buffer_vma = section->vma;
disasm_info.buffer_length =
bfd_get_section_size_before_reloc (section);
i = 0;
while (i < disasm_info.buffer_length)
{
if (data[i] == 0 && data[i + 1] == 0 && data[i + 2] == 0 &&
data[i + 3] == 0)
{
if (done_dot == false)
{
printf ("...\n");
done_dot = true;
}
i += 4;
}
else
{
done_dot = false;
if (with_line_numbers)
{
CONST char *filename;
CONST char *functionname;
unsigned int line;
if (bfd_find_nearest_line (abfd,
section,
syms,
section->vma + i,
&filename,
&functionname,
&line))
{
if (functionname && *functionname
&& strcmp(functionname, prev_function))
{
printf ("%s():\n", functionname);
prev_function = functionname;
}
if (!filename)
filename = "???";
if (line && line != prevline)
{
printf ("%s:%u\n", filename, line);
prevline = line;
}
}
}
objdump_print_address (section->vma + i, &disasm_info);
printf (" ");
if (disassemble) /* New style */
{
int bytes = (*disassemble)(section->vma + i,
&disasm_info);
if (bytes < 0)
break;
i += bytes;
}
else /* Old style */
i += print (section->vma + i,
data + i,
stdout);
putchar ('\n');
}
}
free (data);
}
1991-03-22 05:29:06 +08:00
}
}
#ifdef ELF_STAB_DISPLAY
/* Define a table of stab values and print-strings. We wish the initializer
could be a direct-mapped table, but instead we build one the first
time we need it. */
#define STAB_STRING_LENGTH 6
char stab_name[256][STAB_STRING_LENGTH];
struct stab_print {
int value;
char string[STAB_STRING_LENGTH];
};
struct stab_print stab_print[] = {
#define __define_stab(NAME, CODE, STRING) {CODE, STRING},
#include "aout/stab.def"
#undef __define_stab
{0, 0}
};
void dump_elf_stabs_1 ();
/* This is a kludge for dumping the stabs section from an ELF file that
uses Sun stabs encoding. It has to use some hooks into BFD because
string table sections are not normally visible to BFD callers. */
void
dump_elf_stabs (abfd)
bfd *abfd;
{
int i;
/* Initialize stab name array if first time. */
if (stab_name[0][0] == 0)
{
/* Fill in numeric values for all possible strings. */
for (i = 0; i < 256; i++)
{
sprintf (stab_name[i], "%d", i);
}
for (i = 0; stab_print[i].string[0]; i++)
strcpy (stab_name[stab_print[i].value], stab_print[i].string);
}
if (0 != strncmp ("elf", abfd->xvec->name, 3))
{
fprintf (stderr, "%s: %s is not in ELF format.\n", program_name,
abfd->filename);
return;
}
dump_elf_stabs_1 (abfd, ".stab", ".stabstr");
dump_elf_stabs_1 (abfd, ".stab.excl", ".stab.exclstr");
dump_elf_stabs_1 (abfd, ".stab.index", ".stab.indexstr");
}
void
dump_elf_stabs_1 (abfd, name1, name2)
bfd *abfd;
char *name1; /* Section name of .stab */
char *name2; /* Section name of its string section */
{
Elf_Internal_Shdr *stab_hdr, *stabstr_hdr;
char *strtab;
struct internal_nlist *stabs, *stabs_end;
int i;
unsigned file_string_table_offset, next_file_string_table_offset;
stab_hdr = bfd_elf_find_section (abfd, name1);
if (0 == stab_hdr)
{
printf ("Contents of %s section: none.\n\n", name1);
return;
}
stabstr_hdr = bfd_elf_find_section (abfd, name2);
if (0 == stabstr_hdr)
{
fprintf (stderr, "%s: %s has no %s section.\n", program_name,
abfd->filename, name2);
return;
}
stabs = (struct internal_nlist *) xmalloc (stab_hdr ->sh_size);
strtab = (char *) xmalloc (stabstr_hdr->sh_size);
stabs_end = (struct internal_nlist *) (stab_hdr->sh_size + (char *)stabs);
if (bfd_seek (abfd, stab_hdr->sh_offset, SEEK_SET) < 0 ||
stab_hdr->sh_size != bfd_read ((PTR)stabs, stab_hdr->sh_size, 1, abfd))
{
fprintf (stderr, "%s: reading %s section of %s failed.\n",
program_name, name1,
abfd->filename);
return;
}
1991-03-22 05:29:06 +08:00
if (bfd_seek (abfd, stabstr_hdr->sh_offset, SEEK_SET) < 0 ||
stabstr_hdr->sh_size != bfd_read ((PTR)strtab, stabstr_hdr->sh_size,
1, abfd))
{
fprintf (stderr, "%s: reading %s section of %s failed.\n",
program_name, name2,
abfd->filename);
return;
}
#define SWAP_SYMBOL(symp, abfd) \
{ \
(symp)->n_strx = bfd_h_get_32(abfd, \
(unsigned char *)&(symp)->n_strx); \
(symp)->n_desc = bfd_h_get_16 (abfd, \
(unsigned char *)&(symp)->n_desc); \
(symp)->n_value = bfd_h_get_32 (abfd, \
(unsigned char *)&(symp)->n_value); \
}
printf ("Contents of %s section:\n\n", name1);
printf ("Symnum n_type n_othr n_desc n_value n_strx String\n");
file_string_table_offset = 0;
next_file_string_table_offset = 0;
/* Loop through all symbols and print them.
We start the index at -1 because there is a dummy symbol on
the front of Sun's stabs-in-elf sections. */
for (i = -1; stabs < stabs_end; stabs++, i++)
{
SWAP_SYMBOL (stabs, abfd);
printf ("\n%-6d %-6s %-6d %-6d %08x %-6d", i,
stab_name [stabs->n_type],
stabs->n_other, stabs->n_desc, stabs->n_value,
stabs->n_strx);
/* Symbols with type == 0 (N_UNDF) specify the length of the
string table associated with this file. We use that info
to know how to relocate the *next* file's string table indices. */
if (stabs->n_type == N_UNDF)
{
file_string_table_offset = next_file_string_table_offset;
next_file_string_table_offset += stabs->n_value;
}
/* Now, using the possibly updated string table offset, print the
string (if any) associated with this symbol. */
if ((stabs->n_strx + file_string_table_offset) < stabstr_hdr->sh_size)
printf (" %s", &strtab[stabs->n_strx + file_string_table_offset]);
else
printf (" *");
}
printf ("\n\n");
}
#endif /* ELF_STAB_DISPLAY */
1991-03-22 05:29:06 +08:00
display_bfd (abfd)
bfd *abfd;
{
if (!bfd_check_format (abfd, bfd_object))
{
fprintf (stderr, "%s:%s: %s\n", program_name, abfd->filename,
bfd_errmsg (bfd_error));
return;
}
1991-03-22 05:29:06 +08:00
printf ("\n%s: file format %s\n", abfd->filename, abfd->xvec->name);
if (dump_ar_hdrs)
print_arelt_descr (stdout, abfd, true);
1991-03-22 05:29:06 +08:00
if (dump_file_header)
{
char *comma = "";
printf ("architecture: %s, ",
bfd_printable_arch_mach (bfd_get_arch (abfd),
bfd_get_mach (abfd)));
printf ("flags 0x%08x:\n", abfd->flags);
1991-03-22 05:29:06 +08:00
#define PF(x, y) if (abfd->flags & x) {printf("%s%s", comma, y); comma=", ";}
PF (HAS_RELOC, "HAS_RELOC");
PF (EXEC_P, "EXEC_P");
PF (HAS_LINENO, "HAS_LINENO");
PF (HAS_DEBUG, "HAS_DEBUG");
PF (HAS_SYMS, "HAS_SYMS");
PF (HAS_LOCALS, "HAS_LOCALS");
PF (DYNAMIC, "DYNAMIC");
PF (WP_TEXT, "WP_TEXT");
PF (D_PAGED, "D_PAGED");
PF (BFD_IS_RELAXABLE, "BFD_IS_RELAXABLE");
printf ("\nstart address 0x");
printf_vma (abfd->start_address);
}
printf ("\n");
1991-03-22 05:29:06 +08:00
if (dump_section_headers)
dump_headers (abfd);
if (dump_symtab || dump_reloc_info || disassemble)
{
syms = slurp_symtab (abfd);
}
if (dump_symtab)
dump_symbols (abfd);
#ifdef ELF_STAB_DISPLAY
if (dump_stab_section_info)
dump_elf_stabs (abfd);
#endif
if (dump_reloc_info)
dump_relocs (abfd);
if (dump_section_contents)
dump_data (abfd);
/* Note that disassemble_data re-orders the syms table, but that is
safe - as long as it is done last! */
if (disassemble)
disassemble_data (abfd);
1991-03-22 05:29:06 +08:00
}
static void
1991-03-22 05:29:06 +08:00
display_file (filename, target)
char *filename;
char *target;
{
bfd *file, *arfile = (bfd *) NULL;
file = bfd_openr (filename, target);
if (file == NULL)
{
fprintf (stderr, "%s: ", program_name);
bfd_perror (filename);
return;
}
1991-03-22 05:29:06 +08:00
if (bfd_check_format (file, bfd_archive) == true)
{
printf ("In archive %s:\n", bfd_get_filename (file));
for (;;)
{
bfd_error = no_error;
arfile = bfd_openr_next_archived_file (file, arfile);
if (arfile == NULL)
{
if (bfd_error != no_more_archived_files)
{
fprintf (stderr, "%s: ", program_name);
bfd_perror (bfd_get_filename (file));
}
return;
}
1991-03-22 05:29:06 +08:00
display_bfd (arfile);
/* Don't close the archive elements; we need them for next_archive */
}
1991-03-22 05:29:06 +08:00
}
else
display_bfd (file);
1991-03-22 05:29:06 +08:00
bfd_close (file);
1991-03-22 05:29:06 +08:00
}
/* Actually display the various requested regions */
static void
1991-03-22 05:29:06 +08:00
dump_data (abfd)
bfd *abfd;
{
asection *section;
bfd_byte *data = 0;
bfd_size_type datasize = 0;
bfd_size_type i;
1991-03-22 05:29:06 +08:00
for (section = abfd->sections; section != NULL; section =
section->next)
{
int onaline = 16;
1991-03-22 05:29:06 +08:00
if (only == (char *) NULL ||
strcmp (only, section->name) == 0)
1992-05-02 06:45:45 +08:00
{
if (section->flags & SEC_HAS_CONTENTS)
{
printf ("Contents of section %s:\n", section->name);
if (bfd_get_section_size_before_reloc (section) == 0)
continue;
data = (bfd_byte *) malloc (bfd_get_section_size_before_reloc (section));
if (data == (bfd_byte *) NULL)
{
fprintf (stderr, "%s: memory exhausted.\n", program_name);
exit (1);
}
datasize = bfd_get_section_size_before_reloc (section);
1991-03-22 05:29:06 +08:00
bfd_get_section_contents (abfd, section, (PTR) data, 0, bfd_get_section_size_before_reloc (section));
1991-03-22 05:29:06 +08:00
for (i = 0; i < bfd_get_section_size_before_reloc (section); i += onaline)
{
bfd_size_type j;
printf (" %04lx ", (unsigned long int) (i + section->vma));
for (j = i; j < i + onaline; j++)
{
if (j < bfd_get_section_size_before_reloc (section))
printf ("%02x", (unsigned) (data[j]));
else
printf (" ");
if ((j & 3) == 3)
printf (" ");
}
1991-03-22 05:29:06 +08:00
printf (" ");
for (j = i; j < i + onaline; j++)
{
if (j >= bfd_get_section_size_before_reloc (section))
printf (" ");
else
printf ("%c", isprint (data[j]) ? data[j] : '.');
}
putchar ('\n');
}
free (data);
1992-05-02 06:45:45 +08:00
}
1991-03-22 05:29:06 +08:00
}
}
}
/* Should perhaps share code and display with nm? */
static void
1991-03-22 05:29:06 +08:00
dump_symbols (abfd)
bfd *abfd;
{
unsigned int count;
asymbol **current = syms;
printf ("SYMBOL TABLE:\n");
for (count = 0; count < symcount; count++)
{
1991-03-22 05:29:06 +08:00
if (*current)
{
bfd *cur_bfd = bfd_asymbol_bfd(*current);
if (cur_bfd)
{
bfd_print_symbol (cur_bfd,
stdout,
*current, bfd_print_symbol_all);
printf ("\n");
}
}
current++;
1991-03-22 05:29:06 +08:00
}
printf ("\n");
printf ("\n");
1991-03-22 05:29:06 +08:00
}
static void
dump_relocs (abfd)
bfd *abfd;
1991-03-22 05:29:06 +08:00
{
arelent **relpp;
unsigned int relcount;
asection *a;
for (a = abfd->sections; a != (asection *) NULL; a = a->next)
{
if (a == &bfd_abs_section)
continue;
if (a == &bfd_und_section)
continue;
if (bfd_is_com_section (a))
continue;
printf ("RELOCATION RECORDS FOR [%s]:", a->name);
if (bfd_get_reloc_upper_bound (abfd, a) == 0)
{
printf (" (none)\n\n");
}
else
{
arelent **p;
relpp = (arelent **) xmalloc (bfd_get_reloc_upper_bound (abfd, a));
/* Note that this must be done *before* we sort the syms table. */
relcount = bfd_canonicalize_reloc (abfd, a, relpp, syms);
if (relcount == 0)
{
printf (" (none)\n\n");
}
else
{
printf ("\n");
printf ("OFFSET TYPE VALUE \n");
for (p = relpp; relcount && *p != (arelent *) NULL; p++,
relcount--)
{
arelent *q = *p;
CONST char *sym_name;
/* CONST char *section_name = q->section == (asection *)NULL ? "*abs" :*/
/* q->section->name;*/
CONST char *section_name = (*(q->sym_ptr_ptr))->section->name;
if (q->sym_ptr_ptr && *q->sym_ptr_ptr)
{
sym_name = (*(q->sym_ptr_ptr))->name;
}
else
{
sym_name = 0;
}
if (sym_name)
{
printf_vma (q->address);
printf (" %-8s %s",
q->howto->name,
sym_name);
}
else
{
printf_vma (q->address);
printf (" %-8s [%s]",
q->howto->name,
section_name);
}
if (q->addend)
{
printf ("+0x");
printf_vma (q->addend);
}
printf ("\n");
}
printf ("\n\n");
free (relpp);
}
1991-03-22 05:29:06 +08:00
}
}
1991-03-22 05:29:06 +08:00
}
#ifdef unix
#define _DUMMY_NAME_ "/dev/null"
#else
#define _DUMMY_NAME_ "##dummy"
#endif
1991-04-24 01:02:09 +09:00
static void
DEFUN (display_info_table, (first, last),
int first AND int last)
1991-04-24 01:02:09 +09:00
{
unsigned int i, j;
1991-04-24 01:02:09 +09:00
extern bfd_target *target_vector[];
printf ("\n%12s", " ");
for (i = first; i++ < last && target_vector[i];)
printf ("%s ", target_vector[i]->name);
printf ("\n");
1991-04-24 01:02:09 +09:00
for (j = (int) bfd_arch_obscure + 1; (int) j < (int) bfd_arch_last; j++)
if (strcmp (bfd_printable_arch_mach (j, 0), "UNKNOWN!") != 0)
{
printf ("%11s ", bfd_printable_arch_mach (j, 0));
for (i = first; i++ < last && target_vector[i];)
{
bfd_target *p = target_vector[i];
bfd *abfd = bfd_openw (_DUMMY_NAME_, p->name);
int l = strlen (p->name);
int ok;
bfd_set_format (abfd, bfd_object);
ok = bfd_set_arch_mach (abfd, j, 0);
if (ok)
printf ("%s ", p->name);
else
{
while (l--)
printf ("%c", ok ? '*' : '-');
printf (" ");
}
}
printf ("\n");
}
1991-04-24 01:02:09 +09:00
}
static void
DEFUN_VOID (display_info)
{
char *colum;
unsigned int i, j, columns;
extern bfd_target *target_vector[];
extern char *getenv ();
printf ("BFD header file version %s\n", BFD_VERSION);
for (i = 0; target_vector[i]; i++)
{
bfd_target *p = target_vector[i];
bfd *abfd = bfd_openw (_DUMMY_NAME_, p->name);
bfd_set_format (abfd, bfd_object);
printf ("%s\n (header %s, data %s)\n", p->name,
p->header_byteorder_big_p ? "big endian" : "little endian",
p->byteorder_big_p ? "big endian" : "little endian");
for (j = (int) bfd_arch_obscure + 1; j < (int) bfd_arch_last; j++)
if (bfd_set_arch_mach (abfd, (enum bfd_architecture) j, 0))
printf (" %s\n",
bfd_printable_arch_mach ((enum bfd_architecture) j, 0));
}
columns = 0;
if (colum = getenv ("COLUMNS"))
columns = atoi (colum);
if (!columns)
columns = 80;
for (i = 0; target_vector[i];)
{
int old;
old = i;
for (j = 12; target_vector[i] && j < columns; i++)
j += strlen (target_vector[i]->name) + 1;
i--;
if (old == i)
break;
display_info_table (old, i);
}
}
1991-03-22 05:29:06 +08:00
/** main and like trivia */
int
main (argc, argv)
int argc;
char **argv;
{
int c;
extern int optind;
extern char *optarg;
char *target = default_target;
boolean seenflag = false;
bfd_init ();
1991-03-22 05:29:06 +08:00
program_name = *argv;
while ((c = getopt_long (argc, argv, "ib:m:Vdlfahrtxsj:", long_options,
(int *) 0))
!= EOF)
{
seenflag = true;
switch (c)
{
case 0:
break; /* we've been given a long option */
case 'm':
machine = optarg;
break;
case 'j':
only = optarg;
break;
case 'l':
with_line_numbers = 1;
break;
case 'b':
target = optarg;
break;
case 'f':
dump_file_header = true;
break;
case 'i':
info = true;
break;
case 'x':
dump_symtab = 1;
dump_reloc_info = 1;
dump_file_header = true;
dump_ar_hdrs = 1;
dump_section_headers = 1;
break;
case 't':
dump_symtab = 1;
break;
case 'd':
disassemble = true;
break;
case 's':
dump_section_contents = 1;
break;
case 'r':
dump_reloc_info = 1;
break;
case 'a':
dump_ar_hdrs = 1;
break;
case 'h':
dump_section_headers = 1;
break;
case 'H':
usage (stdout, 0);
case 'V':
show_version = 1;
break;
default:
usage (stderr, 1);
}
1991-03-22 05:29:06 +08:00
}
if (show_version)
{
printf ("GNU %s version %s\n", program_name, program_version);
exit (0);
}
1991-03-22 05:29:06 +08:00
if (seenflag == false)
usage (stderr, 1);
1991-03-22 05:29:06 +08:00
if (info)
{
display_info ();
}
else
{
if (optind == argc)
display_file ("a.out", target);
else
for (; optind < argc;)
display_file (argv[optind++], target);
}
1991-03-22 05:29:06 +08:00
return 0;
}