1991-10-24 19:28:54 +08:00
|
|
|
|
/* Read ELF (Executable and Linking Format) object files for GDB.
|
|
|
|
|
Copyright (C) 1991 Free Software Foundation, Inc.
|
|
|
|
|
Written by Fred Fish at Cygnus Support.
|
|
|
|
|
|
|
|
|
|
This file is part of GDB.
|
|
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
|
|
|
|
|
|
|
|
|
|
/************************************************************************
|
|
|
|
|
* *
|
|
|
|
|
* NOTICE *
|
|
|
|
|
* *
|
|
|
|
|
* This file is still under construction. When it is complete, this *
|
|
|
|
|
* notice will be removed. Until then, direct any questions or changes *
|
|
|
|
|
* to Fred Fish at Cygnus Support (fnf@cygint) *
|
|
|
|
|
* *
|
|
|
|
|
* FIXME Still needs support for shared libraries. *
|
|
|
|
|
* FIXME Still needs support for core files. *
|
|
|
|
|
* FIXME The ".debug" and ".line" section names are hardwired. *
|
|
|
|
|
* FIXME Still needs support ELF symbol tables (as distinct *
|
|
|
|
|
* from DWARF support). Can use them to build the misc *
|
|
|
|
|
* function vector at least. This is fairly trivial once *
|
|
|
|
|
* bfd is extended to handle ELF symbol tables. *
|
|
|
|
|
* *
|
|
|
|
|
************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
|
|
#include "defs.h"
|
1991-12-01 14:47:23 +08:00
|
|
|
|
#include "elf/common.h"
|
|
|
|
|
#include "elf/external.h"
|
|
|
|
|
#include "elf/internal.h"
|
1991-10-24 19:28:54 +08:00
|
|
|
|
#include "bfd.h"
|
|
|
|
|
#include "symfile.h"
|
|
|
|
|
#include "symtab.h"
|
|
|
|
|
#include "ansidecl.h"
|
|
|
|
|
|
1991-11-09 18:50:07 +08:00
|
|
|
|
extern int EXFUN (strcmp, (CONST char *a, CONST char *b));
|
|
|
|
|
extern int EXFUN (dwarf_build_psymtabs,
|
|
|
|
|
(int desc, char *filename, CORE_ADDR addr, int mainline,
|
|
|
|
|
unsigned int dbfoff, unsigned int dbsize, unsigned int lnoffset,
|
|
|
|
|
unsigned int lnsize, struct objfile *objfile));
|
|
|
|
|
|
1991-10-24 19:28:54 +08:00
|
|
|
|
#define STREQ(a,b) (strcmp((a),(b))==0)
|
|
|
|
|
|
|
|
|
|
struct elfinfo {
|
|
|
|
|
unsigned int dboffset; /* Offset to dwarf debug section */
|
|
|
|
|
unsigned int dbsize; /* Size of dwarf debug section */
|
|
|
|
|
unsigned int lnoffset; /* Offset to dwarf line number section */
|
|
|
|
|
unsigned int lnsize; /* Size of dwarf line number section */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* We are called once per section from elf_symfile_read. We
|
|
|
|
|
need to examine each section we are passed, check to see
|
|
|
|
|
if it is something we are interested in processing, and
|
|
|
|
|
if so, stash away some access information for the section.
|
|
|
|
|
|
|
|
|
|
For now we recognize the dwarf debug information sections and
|
|
|
|
|
line number sections from matching their section names. The
|
|
|
|
|
ELF definition is no real help here since it has no direct
|
|
|
|
|
knowledge of DWARF (by design, so any debugging format can be
|
|
|
|
|
used).
|
|
|
|
|
|
|
|
|
|
FIXME: The section names should not be hardwired strings. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
DEFUN(elf_locate_sections, (abfd, sectp, ei),
|
|
|
|
|
bfd *abfd AND
|
|
|
|
|
asection *sectp AND
|
|
|
|
|
struct elfinfo *ei)
|
|
|
|
|
{
|
|
|
|
|
if (STREQ (sectp -> name, ".debug"))
|
|
|
|
|
{
|
|
|
|
|
ei -> dboffset = sectp -> filepos;
|
1992-01-25 07:13:14 +08:00
|
|
|
|
ei -> dbsize = bfd_get_section_size_before_reloc (sectp);
|
1991-10-24 19:28:54 +08:00
|
|
|
|
}
|
|
|
|
|
else if (STREQ (sectp -> name, ".line"))
|
|
|
|
|
{
|
|
|
|
|
ei -> lnoffset = sectp -> filepos;
|
1992-01-25 07:13:14 +08:00
|
|
|
|
ei -> lnsize = bfd_get_section_size_before_reloc (sectp);
|
1991-10-24 19:28:54 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1991-12-17 04:57:28 +08:00
|
|
|
|
char *
|
|
|
|
|
DEFUN(elf_interpreter, (abfd),
|
|
|
|
|
bfd *abfd)
|
|
|
|
|
{
|
|
|
|
|
sec_ptr interp_sec;
|
|
|
|
|
unsigned size;
|
|
|
|
|
char *interp = NULL;
|
|
|
|
|
|
|
|
|
|
interp_sec = bfd_get_section_by_name (abfd, ".interp");
|
|
|
|
|
if (interp_sec)
|
|
|
|
|
{
|
|
|
|
|
size = bfd_section_size (abfd, interp_sec);
|
|
|
|
|
interp = alloca (size);
|
|
|
|
|
if (bfd_get_section_contents (abfd, interp_sec, interp, (file_ptr)0,
|
|
|
|
|
size))
|
|
|
|
|
{
|
|
|
|
|
interp = savestring (interp, size - 1);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
interp = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return (interp);
|
|
|
|
|
}
|
|
|
|
|
|
1991-11-15 11:36:45 +08:00
|
|
|
|
/*
|
|
|
|
|
|
|
|
|
|
LOCAL FUNCTION
|
|
|
|
|
|
|
|
|
|
record_misc_function -- add entry to miscellaneous function vector
|
|
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
|
|
|
|
|
|
static void record_misc_function (char *name, CORE_ADDR address)
|
|
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
|
|
|
|
|
|
Given a pointer to the name of a symbol that should be added to the
|
|
|
|
|
miscellaneous function vector, and the address associated with that
|
|
|
|
|
symbol, records this information for later use in building the
|
|
|
|
|
miscellaneous function vector.
|
|
|
|
|
|
|
|
|
|
NOTES
|
|
|
|
|
|
|
|
|
|
FIXME: For now we just use mf_unknown as the type. This should be
|
|
|
|
|
fixed.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
static void
|
1991-12-17 04:57:28 +08:00
|
|
|
|
DEFUN(record_misc_function, (name, address, mf_type),
|
|
|
|
|
char *name AND CORE_ADDR address AND enum misc_function_type mf_type)
|
1991-11-15 11:36:45 +08:00
|
|
|
|
{
|
|
|
|
|
prim_record_misc_function (obsavestring (name, strlen (name)), address,
|
1991-12-17 04:57:28 +08:00
|
|
|
|
mf_type);
|
1991-11-15 11:36:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
1991-12-17 04:57:28 +08:00
|
|
|
|
/*
|
|
|
|
|
|
|
|
|
|
LOCAL FUNCTION
|
|
|
|
|
|
|
|
|
|
elf_symtab_read -- read the symbol table of an ELF file
|
|
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
|
|
|
|
|
|
void elf_symtab_read (bfd *abfd, CORE_ADDR addr, int mainline)
|
|
|
|
|
|
|
|
|
|
DESCRIPTION
|
|
|
|
|
|
|
|
|
|
Given an open bfd, a base address to relocate symbols to, and a
|
|
|
|
|
flag that specifies whether or not this bfd is for an executable
|
|
|
|
|
or not (may be shared library for example), add all the global
|
|
|
|
|
function and data symbols to the miscellaneous function vector.
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
1991-11-15 11:36:45 +08:00
|
|
|
|
static void
|
1991-12-17 04:57:28 +08:00
|
|
|
|
DEFUN (elf_symtab_read, (abfd, addr, mainline),
|
1991-11-15 11:36:45 +08:00
|
|
|
|
bfd *abfd AND
|
1991-12-17 04:57:28 +08:00
|
|
|
|
CORE_ADDR addr AND
|
|
|
|
|
int mainline)
|
1991-11-15 11:36:45 +08:00
|
|
|
|
{
|
|
|
|
|
unsigned int storage_needed;
|
|
|
|
|
asymbol *sym;
|
|
|
|
|
asymbol **symbol_table;
|
|
|
|
|
unsigned int number_of_symbols;
|
|
|
|
|
unsigned int i;
|
|
|
|
|
struct cleanup *back_to;
|
1991-12-17 04:57:28 +08:00
|
|
|
|
CORE_ADDR symaddr;
|
|
|
|
|
enum misc_function_type mf_type;
|
1991-11-15 11:36:45 +08:00
|
|
|
|
|
|
|
|
|
storage_needed = get_symtab_upper_bound (abfd);
|
|
|
|
|
|
|
|
|
|
if (storage_needed > 0)
|
|
|
|
|
{
|
|
|
|
|
symbol_table = (asymbol **) bfd_xmalloc (storage_needed);
|
|
|
|
|
back_to = make_cleanup (free, symbol_table);
|
|
|
|
|
number_of_symbols = bfd_canonicalize_symtab (abfd, symbol_table);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < number_of_symbols; i++)
|
|
|
|
|
{
|
|
|
|
|
sym = *symbol_table++;
|
1992-02-14 09:22:12 +08:00
|
|
|
|
/* Select global/weak symbols that are defined in a specific section.
|
|
|
|
|
Note that bfd now puts abs symbols in their own section, so
|
|
|
|
|
all symbols we are interested in will have a section. */
|
|
|
|
|
if ((sym -> flags & (BSF_GLOBAL | BSF_WEAK))
|
|
|
|
|
&& (sym -> section != NULL))
|
1991-11-15 11:36:45 +08:00
|
|
|
|
{
|
1991-12-17 04:57:28 +08:00
|
|
|
|
symaddr = sym -> value;
|
1992-02-14 09:22:12 +08:00
|
|
|
|
/* Relocate all non-absolute symbols by base address.
|
|
|
|
|
FIXME: Can we eliminate the check for mainline now,
|
|
|
|
|
since shouldn't addr be 0 in this case? */
|
|
|
|
|
if (!mainline && (sym -> section != &bfd_abs_section))
|
1991-12-17 04:57:28 +08:00
|
|
|
|
{
|
|
|
|
|
symaddr += addr;
|
|
|
|
|
}
|
|
|
|
|
/* For non-absolute symbols, use the type of the section
|
|
|
|
|
they are relative to, to intuit text/data. Bfd provides
|
|
|
|
|
no way of figuring this out for absolute symbols. */
|
1992-02-14 09:22:12 +08:00
|
|
|
|
if (sym -> section -> flags & SEC_CODE)
|
1991-12-17 04:57:28 +08:00
|
|
|
|
{
|
|
|
|
|
mf_type = mf_text;
|
|
|
|
|
}
|
1992-02-14 09:22:12 +08:00
|
|
|
|
else if (sym -> section -> flags & SEC_DATA)
|
1991-12-17 04:57:28 +08:00
|
|
|
|
{
|
|
|
|
|
mf_type = mf_data;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
mf_type = mf_unknown;
|
|
|
|
|
}
|
|
|
|
|
record_misc_function ((char *) sym -> name, symaddr, mf_type);
|
1991-11-15 11:36:45 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
do_cleanups (back_to);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1991-10-24 19:28:54 +08:00
|
|
|
|
/* Scan and build partial symbols for a symbol file.
|
|
|
|
|
We have been initialized by a call to elf_symfile_init, which
|
|
|
|
|
currently does nothing.
|
|
|
|
|
|
|
|
|
|
ADDR is the address relative to which the symbols in it are (e.g.
|
|
|
|
|
the base address of the text segment).
|
|
|
|
|
|
|
|
|
|
MAINLINE is true if we are reading the main symbol
|
|
|
|
|
table (as opposed to a shared lib or dynamically loaded file).
|
|
|
|
|
|
|
|
|
|
This function only does the minimum work necessary for letting the
|
|
|
|
|
user "name" things symbolically; it does not read the entire symtab.
|
|
|
|
|
Instead, it reads the external and static symbols and puts them in partial
|
|
|
|
|
symbol tables. When more extensive information is requested of a
|
|
|
|
|
file, the corresponding partial symbol table is mutated into a full
|
|
|
|
|
fledged symbol table by going back and reading the symbols
|
|
|
|
|
for real. The function dwarf_psymtab_to_symtab() is the function that
|
1991-11-15 11:36:45 +08:00
|
|
|
|
does this for DWARF symbols.
|
|
|
|
|
|
|
|
|
|
Note that ELF files have a "minimal" symbol table, which looks a lot
|
|
|
|
|
like a COFF symbol table, but has only the minimal information necessary
|
|
|
|
|
for linking. We process this also, and just use the information to
|
|
|
|
|
add to the misc function vector. This gives us some minimal debugging
|
|
|
|
|
capability even for files compiled without -g.
|
|
|
|
|
*/
|
1991-10-24 19:28:54 +08:00
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
DEFUN(elf_symfile_read, (sf, addr, mainline),
|
|
|
|
|
struct sym_fns *sf AND
|
|
|
|
|
CORE_ADDR addr AND
|
|
|
|
|
int mainline)
|
|
|
|
|
{
|
1991-11-09 18:50:07 +08:00
|
|
|
|
bfd *abfd = sf->objfile->obfd;
|
1991-10-24 19:28:54 +08:00
|
|
|
|
struct elfinfo ei;
|
1991-11-15 11:36:45 +08:00
|
|
|
|
struct cleanup *back_to;
|
1991-10-24 19:28:54 +08:00
|
|
|
|
|
1991-11-15 11:36:45 +08:00
|
|
|
|
init_misc_bunches ();
|
|
|
|
|
back_to = make_cleanup (discard_misc_bunches, 0);
|
|
|
|
|
|
|
|
|
|
/* Process the normal ELF symbol table first. */
|
|
|
|
|
|
1991-12-17 04:57:28 +08:00
|
|
|
|
elf_symtab_read (abfd, addr, mainline);
|
1991-11-15 11:36:45 +08:00
|
|
|
|
|
|
|
|
|
/* Now process the DWARF debugging information, which is contained in
|
|
|
|
|
special ELF sections. We first have to find them... */
|
|
|
|
|
|
|
|
|
|
(void) memset ((char *) &ei, 0, sizeof (ei));
|
1991-10-24 19:28:54 +08:00
|
|
|
|
bfd_map_over_sections (abfd, elf_locate_sections, &ei);
|
|
|
|
|
if (ei.dboffset && ei.lnoffset)
|
|
|
|
|
{
|
|
|
|
|
dwarf_build_psymtabs (fileno ((FILE *)(abfd -> iostream)),
|
|
|
|
|
bfd_get_filename (abfd),
|
|
|
|
|
addr, mainline,
|
|
|
|
|
ei.dboffset, ei.dbsize,
|
1991-11-09 18:50:07 +08:00
|
|
|
|
ei.lnoffset, ei.lnsize, sf->objfile);
|
1991-10-24 19:28:54 +08:00
|
|
|
|
}
|
1991-11-15 11:36:45 +08:00
|
|
|
|
|
1991-10-24 19:28:54 +08:00
|
|
|
|
if (!partial_symtab_list)
|
|
|
|
|
{
|
|
|
|
|
wrap_here ("");
|
|
|
|
|
printf_filtered ("(no debugging symbols found)...");
|
|
|
|
|
wrap_here ("");
|
|
|
|
|
}
|
1991-11-15 11:36:45 +08:00
|
|
|
|
|
|
|
|
|
/* Go over the miscellaneous functions and install them in the
|
|
|
|
|
miscellaneous function vector. */
|
|
|
|
|
|
|
|
|
|
condense_misc_bunches (!mainline);
|
|
|
|
|
do_cleanups (back_to);
|
1991-10-24 19:28:54 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Initialize anything that needs initializing when a completely new symbol
|
|
|
|
|
file is specified (not just adding some symbols from another file, e.g. a
|
|
|
|
|
shared library).
|
|
|
|
|
|
|
|
|
|
For now at least, we have nothing in particular to do, so this function is
|
|
|
|
|
just a stub. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
DEFUN_VOID (elf_new_init)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ELF specific initialization routine for reading symbols.
|
|
|
|
|
|
|
|
|
|
It is passed a pointer to a struct sym_fns which contains, among other
|
|
|
|
|
things, the BFD for the file whose symbols are being read, and a slot for
|
|
|
|
|
a pointer to "private data" which we can fill with goodies.
|
|
|
|
|
|
|
|
|
|
For now at least, we have nothing in particular to do, so this function is
|
|
|
|
|
just a stub. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
DEFUN(elf_symfile_init, (sf),
|
|
|
|
|
struct sym_fns *sf)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Register that we are able to handle ELF object file formats and DWARF
|
|
|
|
|
debugging formats.
|
|
|
|
|
|
|
|
|
|
Unlike other object file formats, where the debugging information format
|
|
|
|
|
is implied by the object file format, the ELF object file format and the
|
|
|
|
|
DWARF debugging information format are two distinct, and potentially
|
|
|
|
|
separate entities. I.E. it is perfectly possible to have ELF objects
|
|
|
|
|
with debugging formats other than DWARF. And it is conceivable that the
|
|
|
|
|
DWARF debugging format might be used with another object file format,
|
|
|
|
|
like COFF, by simply using COFF's custom section feature.
|
|
|
|
|
|
|
|
|
|
GDB, and to a lesser extent BFD, should support the notion of separate
|
|
|
|
|
object file formats and debugging information formats. For now, we just
|
|
|
|
|
use "elf" in the same sense as "a.out" or "coff", to imply both the ELF
|
|
|
|
|
object file format and the DWARF debugging format. */
|
|
|
|
|
|
|
|
|
|
static struct sym_fns elf_sym_fns = {
|
|
|
|
|
"elf", /* sym_name: name or name prefix of BFD target type */
|
|
|
|
|
3, /* sym_namelen: number of significant sym_name chars */
|
|
|
|
|
elf_new_init, /* sym_new_init: init anything gbl to entire symtab */
|
|
|
|
|
elf_symfile_init, /* sym_init: read initial info, setup for sym_read() */
|
|
|
|
|
elf_symfile_read, /* sym_read: read a symbol file into symtab */
|
|
|
|
|
NULL, /* sym_bfd: accessor for symbol file being read */
|
|
|
|
|
NULL, /* sym_private: sym_init & sym_read shared info */
|
|
|
|
|
NULL /* next: pointer to next struct sym_fns */
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
DEFUN_VOID (_initialize_elfread)
|
|
|
|
|
{
|
|
|
|
|
add_symtab_fns (&elf_sym_fns);
|
|
|
|
|
}
|