1991-03-21 22:11:25 +01:00
|
|
|
|
/* Copyright (C) 1990, 1991 Free Software Foundation, Inc.
|
|
|
|
|
|
|
|
|
|
This file is part of BFD, the Binary File Diddler.
|
|
|
|
|
|
|
|
|
|
BFD 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 1, or (at your option)
|
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
|
|
BFD 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 BFD; see the file COPYING. If not, write to
|
|
|
|
|
the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
|
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
/* $Id$ */
|
1991-07-04 18:52:56 +02:00
|
|
|
|
|
|
|
|
|
/*proto*
|
|
|
|
|
@section typedef bfd
|
|
|
|
|
|
|
|
|
|
Pointers to bfd structs are the cornerstone of any application using
|
|
|
|
|
libbfd. References though the bfd and to data in the bfd give the
|
|
|
|
|
entire bfd functionality.
|
|
|
|
|
|
|
|
|
|
Finally! The BFD struct itself. This contains the major data about
|
|
|
|
|
the file, and contains pointers to the rest of the data.
|
|
|
|
|
|
|
|
|
|
*+++
|
|
|
|
|
|
|
|
|
|
$struct _bfd
|
|
|
|
|
${
|
|
|
|
|
The filename the application opened the bfd with.
|
|
|
|
|
|
|
|
|
|
$ CONST char *filename;
|
|
|
|
|
|
|
|
|
|
A pointer to the target jump table.
|
|
|
|
|
|
|
|
|
|
$ struct bfd_target *xvec;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
To avoid dragging too many header files into every file that
|
|
|
|
|
includes bfd.h, IOSTREAM has been declared as a "char *", and MTIME
|
|
|
|
|
as a "long". Their correct types, to which they are cast when used,
|
|
|
|
|
are "FILE *" and "time_t".
|
|
|
|
|
|
|
|
|
|
The iostream is the result of an fopen on the filename.
|
|
|
|
|
|
|
|
|
|
$ char *iostream;
|
|
|
|
|
|
|
|
|
|
Is the file being cached @xref{File Caching}.
|
|
|
|
|
|
|
|
|
|
$ boolean cacheable;
|
|
|
|
|
|
|
|
|
|
Marks whether there was a default target specified when the bfd was
|
|
|
|
|
opened. This is used to select what matching algorithm to use to chose
|
|
|
|
|
the back end.
|
|
|
|
|
|
|
|
|
|
$ boolean target_defaulted;
|
|
|
|
|
|
|
|
|
|
The caching routines use these to maintain an LRU list of bfds.
|
|
|
|
|
|
|
|
|
|
$ struct _bfd *lru_prev, *lru_next;
|
|
|
|
|
|
|
|
|
|
When a file is closed by the caching routines, it retains the state
|
|
|
|
|
here:
|
|
|
|
|
|
|
|
|
|
$ file_ptr where;
|
|
|
|
|
|
|
|
|
|
and here:
|
|
|
|
|
|
|
|
|
|
$ boolean opened_once;
|
|
|
|
|
|
|
|
|
|
$ boolean mtime_set;
|
|
|
|
|
File modified time
|
|
|
|
|
|
|
|
|
|
$ long mtime;
|
|
|
|
|
|
|
|
|
|
For output files, channel we locked (is this used?).
|
|
|
|
|
|
|
|
|
|
$int ifd;
|
|
|
|
|
|
|
|
|
|
The format which belongs to the bfd.
|
|
|
|
|
|
|
|
|
|
$ bfd_format format;
|
|
|
|
|
|
|
|
|
|
The direction the bfd was opened with
|
|
|
|
|
|
|
|
|
|
$ enum bfd_direction {no_direction = 0,
|
|
|
|
|
$ read_direction = 1,
|
|
|
|
|
$ write_direction = 2,
|
|
|
|
|
$ both_direction = 3} direction;
|
|
|
|
|
|
|
|
|
|
Format_specific flags
|
|
|
|
|
|
|
|
|
|
$ flagword flags;
|
|
|
|
|
|
|
|
|
|
Currently my_archive is tested before adding origin to anything. I
|
|
|
|
|
believe that this can become always an add of origin, with origin set
|
|
|
|
|
to 0 for non archive files.
|
|
|
|
|
|
|
|
|
|
$ file_ptr origin;
|
|
|
|
|
|
|
|
|
|
Remember when output has begun, to stop strange things happening.
|
|
|
|
|
|
|
|
|
|
$ boolean output_has_begun;
|
|
|
|
|
|
|
|
|
|
Pointer to linked list of sections
|
|
|
|
|
|
|
|
|
|
$ struct sec *sections;
|
|
|
|
|
|
|
|
|
|
The number of sections
|
|
|
|
|
|
|
|
|
|
$ unsigned int section_count;
|
|
|
|
|
|
|
|
|
|
Stuff only usefull for object files:
|
|
|
|
|
The start address.
|
|
|
|
|
|
|
|
|
|
$ bfd_vma start_address;
|
|
|
|
|
Used for input and output
|
|
|
|
|
|
|
|
|
|
$ unsigned int symcount;
|
|
|
|
|
Symtab for output bfd
|
|
|
|
|
|
|
|
|
|
$ struct symbol_cache_entry **outsymbols;
|
|
|
|
|
|
|
|
|
|
Architecture of object machine, eg m68k
|
|
|
|
|
|
|
|
|
|
$ enum bfd_architecture obj_arch;
|
|
|
|
|
|
|
|
|
|
Particular machine within arch, e.g. 68010
|
|
|
|
|
|
|
|
|
|
$ unsigned long obj_machine;
|
|
|
|
|
|
|
|
|
|
Stuff only usefull for archives:
|
|
|
|
|
|
|
|
|
|
$ PTR arelt_data;
|
|
|
|
|
$ struct _bfd *my_archive;
|
|
|
|
|
$ struct _bfd *next;
|
|
|
|
|
$ struct _bfd *archive_head;
|
|
|
|
|
$ boolean has_armap;
|
|
|
|
|
|
|
|
|
|
Used by the back end to hold private data.
|
|
|
|
|
|
|
|
|
|
$ PTR tdata;
|
|
|
|
|
|
|
|
|
|
Used by the application to hold private data
|
|
|
|
|
|
|
|
|
|
$ PTR usrdata;
|
|
|
|
|
|
|
|
|
|
Where all the allocated stuff under this BFD goes
|
|
|
|
|
|
|
|
|
|
$ struct obstack memory;
|
|
|
|
|
$};
|
|
|
|
|
|
|
|
|
|
*---
|
|
|
|
|
|
|
|
|
|
*/
|
1991-05-21 02:14:16 +02:00
|
|
|
|
#include <sysdep.h>
|
1991-03-21 22:11:25 +01:00
|
|
|
|
#include "bfd.h"
|
|
|
|
|
#include "libbfd.h"
|
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
short _bfd_host_big_endian = 0x0100;
|
1991-07-04 18:52:56 +02:00
|
|
|
|
/* Accessing the above as (*(char*)&_bfd_host_big_endian), will
|
|
|
|
|
return 1 if the host is big-endian, 0 otherwise.
|
|
|
|
|
(assuming that a short is two bytes long!!! FIXME)
|
|
|
|
|
(See HOST_IS_BIG_ENDIAN_P in bfd.h.) */
|
1991-03-21 22:11:25 +01:00
|
|
|
|
|
|
|
|
|
/** Error handling
|
|
|
|
|
o - Most functions return nonzero on success (check doc for
|
1991-07-04 18:52:56 +02:00
|
|
|
|
precise semantics); 0 or NULL on error.
|
1991-03-21 22:11:25 +01:00
|
|
|
|
o - Internal errors are documented by the value of bfd_error.
|
1991-07-04 18:52:56 +02:00
|
|
|
|
If that is system_call_error then check errno.
|
1991-03-21 22:11:25 +01:00
|
|
|
|
o - The easiest way to report this to the user is to use bfd_perror.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
bfd_ec bfd_error = no_error;
|
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
char *bfd_errmsgs[] = { "No error",
|
|
|
|
|
"System call error",
|
|
|
|
|
"Invalid target",
|
|
|
|
|
"File in wrong format",
|
|
|
|
|
"Invalid operation",
|
|
|
|
|
"Memory exhausted",
|
|
|
|
|
"No symbols",
|
|
|
|
|
"No relocation info",
|
|
|
|
|
"No more archived files",
|
|
|
|
|
"Malformed archive",
|
|
|
|
|
"Symbol not found",
|
|
|
|
|
"File format not recognized",
|
|
|
|
|
"File format is ambiguous",
|
|
|
|
|
"Section has no contents",
|
|
|
|
|
"Nonrepresentable section on output",
|
|
|
|
|
"#<Invalid error code>"
|
|
|
|
|
};
|
1991-03-21 22:11:25 +01:00
|
|
|
|
|
1991-04-15 17:25:16 +02:00
|
|
|
|
static
|
|
|
|
|
void
|
|
|
|
|
DEFUN(bfd_nonrepresentable_section,(abfd, name),
|
1991-07-04 18:52:56 +02:00
|
|
|
|
CONST bfd * CONST abfd AND
|
|
|
|
|
CONST char * CONST name)
|
1991-04-15 17:25:16 +02:00
|
|
|
|
{
|
1991-04-17 02:56:02 +02:00
|
|
|
|
printf("bfd error writing file %s, format %s can't represent section %s\n",
|
1991-07-04 18:52:56 +02:00
|
|
|
|
abfd->filename,
|
|
|
|
|
abfd->xvec->name,
|
|
|
|
|
name);
|
1991-04-15 17:25:16 +02:00
|
|
|
|
exit(1);
|
|
|
|
|
}
|
1991-05-02 06:11:40 +02:00
|
|
|
|
|
1991-04-15 17:25:16 +02:00
|
|
|
|
bfd_error_vector_type bfd_error_vector =
|
|
|
|
|
{
|
|
|
|
|
bfd_nonrepresentable_section
|
|
|
|
|
};
|
|
|
|
|
|
1991-05-21 02:14:16 +02:00
|
|
|
|
#if 1 || !defined(ANSI_LIBRARIES) && !defined(__STDC__)
|
1991-03-21 22:11:25 +01:00
|
|
|
|
char *
|
|
|
|
|
strerror (code)
|
|
|
|
|
int code;
|
|
|
|
|
{
|
|
|
|
|
extern int sys_nerr;
|
|
|
|
|
extern char *sys_errlist[];
|
|
|
|
|
|
|
|
|
|
return (((code < 0) || (code >= sys_nerr)) ? "(unknown error)" :
|
1991-07-04 18:52:56 +02:00
|
|
|
|
sys_errlist [code]);
|
1991-03-21 22:11:25 +01:00
|
|
|
|
}
|
|
|
|
|
#endif /* not ANSI_LIBRARIES */
|
|
|
|
|
|
1991-04-15 17:25:16 +02:00
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
char *
|
|
|
|
|
bfd_errmsg (error_tag)
|
|
|
|
|
bfd_ec error_tag;
|
|
|
|
|
{
|
1991-05-21 02:14:16 +02:00
|
|
|
|
#ifndef errno
|
|
|
|
|
extern int errno;
|
|
|
|
|
#endif
|
1991-03-21 22:11:25 +01:00
|
|
|
|
if (error_tag == system_call_error)
|
|
|
|
|
return strerror (errno);
|
|
|
|
|
|
|
|
|
|
if ((((int)error_tag <(int) no_error) ||
|
|
|
|
|
((int)error_tag > (int)invalid_error_code)))
|
|
|
|
|
error_tag = invalid_error_code;/* sanity check */
|
|
|
|
|
|
|
|
|
|
return bfd_errmsgs [(int)error_tag];
|
|
|
|
|
}
|
|
|
|
|
|
1991-04-15 17:25:16 +02:00
|
|
|
|
|
|
|
|
|
void bfd_default_error_trap(error_tag)
|
|
|
|
|
bfd_ec error_tag;
|
|
|
|
|
{
|
|
|
|
|
printf("bfd assert fail (%s)\n", bfd_errmsg(error_tag));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void (*bfd_error_trap)() = bfd_default_error_trap;
|
|
|
|
|
void (*bfd_error_nonrepresentabltrap)() = bfd_default_error_trap;
|
1991-05-02 06:11:40 +02:00
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
void
|
1991-04-17 02:56:02 +02:00
|
|
|
|
DEFUN(bfd_perror,(message),
|
|
|
|
|
CONST char *message)
|
1991-03-21 22:11:25 +01:00
|
|
|
|
{
|
|
|
|
|
if (bfd_error == system_call_error)
|
1991-07-04 18:52:56 +02:00
|
|
|
|
perror((char *)message); /* must be system error then... */
|
1991-03-21 22:11:25 +01:00
|
|
|
|
else {
|
|
|
|
|
if (message == NULL || *message == '\0')
|
|
|
|
|
fprintf (stderr, "%s\n", bfd_errmsg (bfd_error));
|
|
|
|
|
else
|
|
|
|
|
fprintf (stderr, "%s: %s\n", message, bfd_errmsg (bfd_error));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
/** Symbols */
|
|
|
|
|
|
|
|
|
|
/* returns the number of octets of storage required */
|
1991-07-04 18:52:56 +02:00
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
unsigned int
|
|
|
|
|
get_reloc_upper_bound (abfd, asect)
|
|
|
|
|
bfd *abfd;
|
|
|
|
|
sec_ptr asect;
|
|
|
|
|
{
|
|
|
|
|
if (abfd->format != bfd_object) {
|
|
|
|
|
bfd_error = invalid_operation;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return BFD_SEND (abfd, _get_reloc_upper_bound, (abfd, asect));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unsigned int
|
|
|
|
|
bfd_canonicalize_reloc (abfd, asect, location, symbols)
|
|
|
|
|
bfd *abfd;
|
|
|
|
|
sec_ptr asect;
|
|
|
|
|
arelent **location;
|
|
|
|
|
asymbol **symbols;
|
|
|
|
|
{
|
|
|
|
|
if (abfd->format != bfd_object) {
|
|
|
|
|
bfd_error = invalid_operation;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return BFD_SEND (abfd, _bfd_canonicalize_reloc, (abfd, asect, location, symbols));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
boolean
|
|
|
|
|
bfd_set_file_flags (abfd, flags)
|
|
|
|
|
bfd *abfd;
|
|
|
|
|
flagword flags;
|
|
|
|
|
{
|
|
|
|
|
if (abfd->format != bfd_object) {
|
|
|
|
|
bfd_error = wrong_format;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (bfd_read_p (abfd)) {
|
|
|
|
|
bfd_error = invalid_operation;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & bfd_applicable_file_flags (abfd)) != flags) {
|
|
|
|
|
bfd_error = invalid_operation;
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bfd_get_file_flags (abfd) = flags;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
bfd_set_reloc (ignore_abfd, asect, location, count)
|
|
|
|
|
bfd *ignore_abfd;
|
|
|
|
|
sec_ptr asect;
|
|
|
|
|
arelent **location;
|
|
|
|
|
unsigned int count;
|
|
|
|
|
{
|
|
|
|
|
asect->orelocation = location;
|
|
|
|
|
asect->reloc_count = count;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
bfd_assert(file, line)
|
|
|
|
|
char *file;
|
|
|
|
|
int line;
|
|
|
|
|
{
|
|
|
|
|
printf("bfd assertion fail %s:%d\n",file,line);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
/*proto* bfd_set_start_address
|
|
|
|
|
|
|
|
|
|
Marks the entry point of an output bfd. Returns @code{true} on
|
|
|
|
|
success, @code{false} otherwise.
|
|
|
|
|
|
|
|
|
|
*; PROTO(boolean, bfd_set_start_address,(bfd *, bfd_vma));
|
|
|
|
|
*/
|
|
|
|
|
|
1991-03-21 22:11:25 +01:00
|
|
|
|
boolean
|
|
|
|
|
bfd_set_start_address(abfd, vma)
|
|
|
|
|
bfd *abfd;
|
|
|
|
|
bfd_vma vma;
|
|
|
|
|
{
|
|
|
|
|
abfd->start_address = vma;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
/*proto* bfd_get_mtime
|
|
|
|
|
|
|
|
|
|
Return cached file modification time (e.g. as read from archive header
|
|
|
|
|
for archive members, or from file system if we have been called
|
|
|
|
|
before); else determine modify time, cache it, and return it.
|
1991-03-21 22:11:25 +01:00
|
|
|
|
|
1991-07-04 18:52:56 +02:00
|
|
|
|
*;PROTO(long, bfd_get_mtime, (bfd *));
|
|
|
|
|
|
|
|
|
|
*/
|
1991-03-21 22:11:25 +01:00
|
|
|
|
|
|
|
|
|
long
|
|
|
|
|
bfd_get_mtime (abfd)
|
|
|
|
|
bfd *abfd;
|
|
|
|
|
{
|
|
|
|
|
FILE *fp;
|
|
|
|
|
struct stat buf;
|
|
|
|
|
|
|
|
|
|
if (abfd->mtime_set)
|
|
|
|
|
return abfd->mtime;
|
|
|
|
|
|
|
|
|
|
fp = bfd_cache_lookup (abfd);
|
|
|
|
|
if (0 != fstat (fileno (fp), &buf))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
abfd->mtime_set = true;
|
|
|
|
|
abfd->mtime = buf.st_mtime;
|
|
|
|
|
return abfd->mtime;
|
|
|
|
|
}
|
1991-07-04 18:52:56 +02:00
|
|
|
|
|
|
|
|
|
/*proto*
|
|
|
|
|
*i stuff
|
|
|
|
|
*+
|
|
|
|
|
#define bfd_sizeof_headers(abfd, reloc) \
|
|
|
|
|
BFD_SEND (abfd, _bfd_sizeof_headers, (abfd, reloc))
|
|
|
|
|
|
|
|
|
|
#define bfd_find_nearest_line(abfd, section, symbols, offset, filename_ptr, func, line_ptr) \
|
|
|
|
|
BFD_SEND (abfd, _bfd_find_nearest_line, (abfd, section, symbols, offset, filename_ptr, func, line_ptr))
|
|
|
|
|
|
|
|
|
|
#define bfd_debug_info_start(abfd) \
|
|
|
|
|
BFD_SEND (abfd, _bfd_debug_info_start, (abfd))
|
|
|
|
|
|
|
|
|
|
#define bfd_debug_info_end(abfd) \
|
|
|
|
|
BFD_SEND (abfd, _bfd_debug_info_end, (abfd))
|
|
|
|
|
|
|
|
|
|
#define bfd_debug_info_accumulate(abfd, section) \
|
|
|
|
|
BFD_SEND (abfd, _bfd_debug_info_accumulate, (abfd, section))
|
|
|
|
|
|
|
|
|
|
#define bfd_stat_arch_elt(abfd, stat) \
|
|
|
|
|
BFD_SEND (abfd, _bfd_stat_arch_elt,(abfd, stat))
|
|
|
|
|
*-
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|