9783e04a64
check the result for NULL. Most set bfd_error to no_memory and return in that case; a few are harder to fix, and are marked with "FIXME <return type>". * elf32-hppa.c (hppa_elf_build_arg_reloc_stub hppa_elf_build_long_branch_stub): Check bfd_make_empty_symbol return. * linker.c (_bfd_generic_link_output_symbols _bfd_generic_link_write_global_symbol): Ditto * section.c (bfd_make_section_anyway): Ditto. * tekhex.c (find_chunk tekhex_mkobject): Check bfd_alloc. (first_phase): Ditto. FIXME void (tekhex_make_empty_symbol): Check bfd_zalloc. * sunos.c (sunos_read_dynamic_info): Check bfd_zalloc. (MY(read_dynamic_symbols) MY(read_dynamic_relocs)): Check bfd_alloc. * stringhash.c (_bfd_stringtab_hash_newfunc): Check bfd_hash_allocate. * srec.c: Indent. (fillup_symbols): Check bfd_alloc. FIXME void (srec_mkobject srec_get_section_contents srec_set_section_contents): Check bfd_alloc. (srec_make_empty_symbol): Check bfd_zalloc. * som.c (hppa_som_gen_reloc_type): Check bfd_alloc_by_size_t. (make_unique_section): Check bfd_alloc. (som_new_section_hook): Check bfd_zalloc. (bfd_som_attach_aux_hdr): Ditto. FIXME void * rs6000-core.c (rs6000coff_core_p): Check bfd_zalloc. * osf-core.c (osf_core_make_empty_symbol): Check bfd_zalloc. (osf_core_core_file_p): Check bfd_alloc. * oasys.c (oasys_slurp_symbol_table oasys_archive_p oasys_mkobject oasys_object_p oasys_new_section_hook oasys_set_section_contents): Check bfd_alloc. (oasys_slurp_section_data): Check bfd_zalloc and bfd_alloc. (oasys_make_empty_symbol): Check bfd_zalloc. * nlmcode.h (nlm_make_empty_symbol): Check bfd_zalloc. (nlm_slurp_symbol_table): Check bfd_zalloc and bfd_alloc. * nlm32-sparc.c (nlm_sparc_read_import): Check bfd_alloc. * nlm32-i386.c (nlm_i386_read_import): Check bfd_alloc. * nlm32-alpha.c (nlm_alpha_read_import): Check bfd_alloc. * linker.c (_bfd_link_hash_newfunc (generic_link_hash_newfunc (archive_hash_newfunc (_bfd_generic_link_add_one_symbol): Check bfd_hash_allocate. (_bfd_generic_final_link (_bfd_generic_link_output_symbols (default_indirect_link_order): Check bfd_alloc. (bfd_new_link_order): Check bfd_alloc_by_size_t. * irix-core.c (irix_core_make_empty_symbol): Check bfd_zalloc. * ieee.c: Indent. (read_id get_symbol get_section_entry ieee_archive_p ieee_object_p ieee_slurp_section_data ieee_new_section_hook): Check bfd_alloc. (do_one): Check bfd_alloc. Return a boolean. (ieee_slurp_section_data): Check it. (init_for_output): Check bfd_alloc. Return a boolean. (ieee_set_section_contents): Check it. (do_with_relocs): Check bfd_alloc. Return a boolean. (ieee_bfd_debug_info_accumulate): Ditto. FIXME void. (ieee_mkobject): Check bfd_zalloc. (ieee_make_empty_symbol): Check bfd_zmalloc. * hpux-core.c (hpux_core_make_empty_symbol): Check bfd_zalloc. * hppabsd-core.c (hppabsd_core_make_empty_symbol): Check bfd_zalloc. (hppabsd_core_core_file_p): Check bfd_zalloc. * hp300hpux.c (MY(slurp_symbol_table)): Check bfd_alloc. * elfcode.h (elf_new_section_hook): Check bfd_alloc. (bfd_section_from_phdr): Ditto. (write_relocs): Ditto. FIXME void (elf_map_symbols assign_section_numbers map_program_segments): Ditto. Return a boolean. (swap_out_syms): Ditto. Check elf_map_symbols. (elf_slurp_symbol_table): Check bfd_zalloc. (elf_slurp_reloca_table): Check bfd_alloc. (elf_slurp_reloc_table): Ditto. (elf_compute_section_file_positions): Check assign_section_numbers. (assign_file_positions_except_relocs): Return a boolean. Check map_program_segments. (elf_compute_section_file_positions): Check it. * elf32-mips.c (mips_elf_final_link): Check bfd_alloc. * elf32-hppa.c (hppa_elf_stub_branch_reloc): Check bfd_zmalloc and realloc. (hppa_elf_stub_reloc): Ditto. (hppa_elf_build_arg_reloc_stub): Check bfd_zalloc. (hppa_elf_build_long_branch_stub): Ditto. (elf32_hppa_backend_symbol_table_processing): Ditto. * ecoff.c (ecoff_set_symbol_info): Check bfd_alloc. Return a boolean. (ecoff_slurp_symbol_table): Check it. (ecoff_slurp_armap): Check bfd_alloc. (ecoff_write_armap): Check bfd_zalloc. (ecoff_link_hash_newfunc): Check bfd_hash_allocate and _bfd_link_hash_newfunc. (ecoff_link_add_externals): Check bfd_alloc. * ctor.c (bfd_constructor_entry): Check bfd_alloc. * coffgen.c (coff_real_object_p): Check bfd_alloc. (coff_renumber_symbols): Check bfd_alloc_by_size_t. Return a boolean. (coff_write_symbol): Check bfd_alloc. FIXME int (coff_write_linenumbers): Check bfd_alloc. Return a boolean. (coff_section_symbol): Check bfd_alloc_by_size_t. (coff_get_normalized_symtab): Check bfd_alloc. (coff_bfd_make_debug_symbol): Check bfd_zalloc. * libcoff-in.h: Change decls of coff_renumber_symbols, coff_write_linenumbers. * libcoff.h: Rebuilt. * coffcode.h (coff_write_object_contents): Check coff_renumber_symbols, coff_write_linenumbers. * coffcode.h: Indent. (coff_add_missing_symbols): Check bfd_alloc_by_size_t. Return a boolean. (coff_write_object_contents): Check it. * coff-alpha.c (alpha_relocate_section): Check bfd_alloc. * coff-mips.c (mips_relocate_section): Ditto. * archive.c (bfd_slurp_bsd_armap_f2): Check bfd_alloc value. (do_slurp_bsd_armap): Ditto. (compute_and_write_armap): Check bfd_realloc value. * aoutx.h (translate_from_native_sym_flags): Check bfd_alloc return value. Return boolean value. (NAME(aout,make_empty_symbol)): Check bfd_zalloc return value. (NAME(aout,slurp_symbol_table)): Check bf_alloc and bfd_zalloc return value. (add_to_stringtab): Ditto. FIXME void (aout_link_hash_newfunc): Check bfd_hash_allocate return value. (aout_link_add_symbols): Check bfd_alloc value. (translate_symbol_table): Check translate_from_native_sym_flags. * hp300hpux.c (MY(slurp_symbol_table)): Ditto. * aoutx.h (aout_link_hash_newfunc): Check _bfd_link_hash_newfunc. * opncls.c (bfd_zalloc bfd_realloc): Check result of bfd_alloc. * opncls.c (obstack_chunk_alloc): Define as malloc, not bfd_xmalloc_by_size_t. (_bfd_new_bfd): Check obstack_begin for 0 return. * ieee.c (obstack_chunk_alloc): Define as malloc, not bfd_xmalloc_by_size_t. (ieee_archive_p): Check obstack_begin for 0 return and obstack_finish for NULL return. * hash.c (obstack_chunk_alloc): Define as malloc, not bfd_xmalloc_by_size_t. (bfd_hash_table_init_n): Check obstack_begin for 0 return and obstack_finish for NULL return. (bfd_hash_lookup): Check obstack_alloc for NULL return. * ecofflink.c (obstack_chunk_alloc): Define as malloc, not bfd_xmalloc_by_size_t. bfd_ecoff_debug_accumulate bfd_ecoff_debug_accumulate_other): Check obstack_alloc. (add_file_shuffle add_memory_shuffle): Check obstack_alloc for NULL return. Return boolean, not void. (bfd_ecoff_debug_init): Check obstack_begin for 0 return. (bfd_ecoff_debug_accumulate): Check add_file_shuffle and add_memory_shuffle return. (string_hash_newfunc): Check bfd_hash_allocate and bfd_hash_newfunc. (bfd_ecoff_debug_accumulate): Check bfd_alloc. (ecoff_add_string): Check add_memory_shuffle return. * libbfd-in.h (xmalloc, bfd_xmalloc, bfd_xmalloc_by_size_t): Remove decls. * libbfd.h: Rebuilt.
1044 lines
24 KiB
C
1044 lines
24 KiB
C
/* BFD back-end for s-record objects.
|
|
Copyright 1990, 1991, 1992, 1993, 1994 Free Software Foundation, Inc.
|
|
Written by Steve Chamberlain of Cygnus Support <sac@cygnus.com>.
|
|
|
|
This file is part of BFD, the Binary File Descriptor library.
|
|
|
|
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. */
|
|
|
|
/*
|
|
SUBSECTION
|
|
S-Record handling
|
|
|
|
DESCRIPTION
|
|
|
|
Ordinary S-Records cannot hold anything but addresses and
|
|
data, so that's all that we implement.
|
|
|
|
The only interesting thing is that S-Records may come out of
|
|
order and there is no header, so an initial scan is required
|
|
to discover the minimum and maximum addresses used to create
|
|
the vma and size of the only section we create. We
|
|
arbitrarily call this section ".text".
|
|
|
|
When bfd_get_section_contents is called the file is read
|
|
again, and this time the data is placed into a bfd_alloc'd
|
|
area.
|
|
|
|
Any number of sections may be created for output, we save them
|
|
up and output them when it's time to close the bfd.
|
|
|
|
An s record looks like:
|
|
|
|
EXAMPLE
|
|
S<type><length><address><data><checksum>
|
|
|
|
DESCRIPTION
|
|
Where
|
|
o length
|
|
is the number of bytes following upto the checksum. Note that
|
|
this is not the number of chars following, since it takes two
|
|
chars to represent a byte.
|
|
o type
|
|
is one of:
|
|
0) header record
|
|
1) two byte address data record
|
|
2) three byte address data record
|
|
3) four byte address data record
|
|
7) four byte address termination record
|
|
8) three byte address termination record
|
|
9) two byte address termination record
|
|
|
|
o address
|
|
is the start address of the data following, or in the case of
|
|
a termination record, the start address of the image
|
|
o data
|
|
is the data.
|
|
o checksum
|
|
is the sum of all the raw byte data in the record, from the length
|
|
upwards, modulo 256 and subtracted from 255.
|
|
|
|
|
|
SUBSECTION
|
|
Symbol S-Record handling
|
|
|
|
DESCRIPTION
|
|
Some ICE equipment understands an addition to the standard
|
|
S-Record format; symbols and their addresses can be sent
|
|
before the data.
|
|
|
|
The format of this is:
|
|
($$ <modulename>
|
|
(<space> <symbol> <address>)*)
|
|
$$
|
|
|
|
so a short symbol table could look like:
|
|
|
|
EXAMPLE
|
|
$$ flash.x
|
|
$$ flash.c
|
|
_port6 $0
|
|
_delay $4
|
|
_start $14
|
|
_etext $8036
|
|
_edata $8036
|
|
_end $8036
|
|
$$
|
|
|
|
DESCRIPTION
|
|
We allow symbols to be anywhere in the data stream - the module names
|
|
are always ignored.
|
|
|
|
*/
|
|
|
|
#include "bfd.h"
|
|
#include "sysdep.h"
|
|
#include "libbfd.h"
|
|
|
|
/* Macros for converting between hex and binary */
|
|
|
|
static CONST char digs[] = "0123456789ABCDEF";
|
|
|
|
static char hex_value[1 + (unsigned char) ~0];
|
|
|
|
#define NOT_HEX 20
|
|
#define NIBBLE(x) hex_value[(unsigned char)(x)]
|
|
#define HEX(buffer) ((NIBBLE((buffer)[0])<<4) + NIBBLE((buffer)[1]))
|
|
#define TOHEX(d, x, ch) \
|
|
d[1] = digs[(x) & 0xf]; \
|
|
d[0] = digs[((x)>>4)&0xf]; \
|
|
ch += ((x) & 0xff);
|
|
#define ISHEX(x) (hex_value[(unsigned char)(x)] != NOT_HEX)
|
|
|
|
|
|
|
|
static void
|
|
DEFUN_VOID (srec_init)
|
|
{
|
|
unsigned int i;
|
|
static boolean inited = false;
|
|
|
|
if (inited == false)
|
|
{
|
|
|
|
inited = true;
|
|
|
|
for (i = 0; i < sizeof (hex_value); i++)
|
|
{
|
|
hex_value[i] = NOT_HEX;
|
|
}
|
|
|
|
for (i = 0; i < 10; i++)
|
|
{
|
|
hex_value[i + '0'] = i;
|
|
|
|
}
|
|
for (i = 0; i < 6; i++)
|
|
{
|
|
hex_value[i + 'a'] = i + 10;
|
|
hex_value[i + 'A'] = i + 10;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* The maximum number of bytes on a line is FF */
|
|
#define MAXCHUNK 0xff
|
|
/* The number of bytes we fit onto a line on output */
|
|
#define CHUNK 21
|
|
|
|
/* We cannot output our srecords as we see them, we have to glue them
|
|
together, this is done in this structure : */
|
|
|
|
struct srec_data_list_struct
|
|
{
|
|
unsigned char *data;
|
|
bfd_vma where;
|
|
bfd_size_type size;
|
|
struct srec_data_list_struct *next;
|
|
|
|
|
|
};
|
|
typedef struct srec_data_list_struct srec_data_list_type;
|
|
|
|
|
|
typedef struct srec_data_struct
|
|
{
|
|
srec_data_list_type *head;
|
|
unsigned int type;
|
|
|
|
int done_symbol_read;
|
|
int count;
|
|
asymbol *symbols;
|
|
char *strings;
|
|
int symbol_idx;
|
|
int string_size;
|
|
int string_idx;
|
|
}
|
|
tdata_type;
|
|
|
|
|
|
/*
|
|
called once per input S-Record, used to work out vma and size of data.
|
|
*/
|
|
|
|
static bfd_vma low, high;
|
|
|
|
/*ARGSUSED*/
|
|
static void
|
|
size_symbols (abfd, buf, len, val)
|
|
bfd *abfd;
|
|
char *buf;
|
|
int len;
|
|
int val;
|
|
{
|
|
abfd->symcount++;
|
|
abfd->tdata.srec_data->string_size += len + 1;
|
|
}
|
|
|
|
static void
|
|
fillup_symbols (abfd, buf, len, val)
|
|
bfd *abfd;
|
|
char *buf;
|
|
int len;
|
|
int val;
|
|
{
|
|
if (!abfd->tdata.srec_data->done_symbol_read)
|
|
{
|
|
asymbol *p;
|
|
if (abfd->tdata.srec_data->symbols == 0)
|
|
{
|
|
abfd->tdata.srec_data->symbols = (asymbol *) bfd_alloc (abfd, abfd->symcount * sizeof (asymbol));
|
|
abfd->tdata.srec_data->strings = (char *) bfd_alloc (abfd, abfd->tdata.srec_data->string_size);
|
|
if (!abfd->tdata.srec_data->symbols || !abfd->tdata.srec_data->strings)
|
|
{
|
|
bfd_error = no_memory;
|
|
abort (); /* FIXME */
|
|
}
|
|
abfd->tdata.srec_data->symbol_idx = 0;
|
|
abfd->tdata.srec_data->string_idx = 0;
|
|
}
|
|
|
|
p = abfd->tdata.srec_data->symbols + abfd->tdata.srec_data->symbol_idx++;
|
|
p->the_bfd = abfd;
|
|
p->name = abfd->tdata.srec_data->strings + abfd->tdata.srec_data->string_idx;
|
|
memcpy ((char *) (p->name), buf, len + 1);
|
|
abfd->tdata.srec_data->string_idx += len + 1;
|
|
p->value = val;
|
|
p->flags = BSF_EXPORT | BSF_GLOBAL;
|
|
p->section = &bfd_abs_section;
|
|
p->udata = 0;
|
|
}
|
|
}
|
|
/*ARGSUSED*/
|
|
static void
|
|
DEFUN (size_srec, (abfd, section, address, raw, length),
|
|
bfd * abfd AND
|
|
asection * section AND
|
|
bfd_vma address AND
|
|
bfd_byte * raw AND
|
|
unsigned int length)
|
|
{
|
|
if (address < low)
|
|
low = address;
|
|
if (address + length > high)
|
|
high = address + length - 1;
|
|
}
|
|
|
|
|
|
/*
|
|
called once per input S-Record, copies data from input into bfd_alloc'd area
|
|
*/
|
|
|
|
/*ARGSUSED*/
|
|
static void
|
|
DEFUN (fillup, (abfd, section, address, raw, length),
|
|
bfd * abfd AND
|
|
asection * section AND
|
|
bfd_vma address AND
|
|
bfd_byte * raw AND
|
|
unsigned int length)
|
|
{
|
|
unsigned int i;
|
|
bfd_byte *dst =
|
|
(bfd_byte *) (section->used_by_bfd) + address - section->vma;
|
|
/* length -1 because we don't read in the checksum */
|
|
for (i = 0; i < length - 1; i++)
|
|
{
|
|
*dst = HEX (raw);
|
|
dst++;
|
|
raw += 2;
|
|
}
|
|
}
|
|
|
|
/* Pass over an S-Record file, calling one of the above functions on each
|
|
record. */
|
|
|
|
static int
|
|
white (x)
|
|
char x;
|
|
{
|
|
return (x == ' ' || x == '\t' || x == '\n' || x == '\r');
|
|
}
|
|
static int
|
|
skipwhite (src, abfd)
|
|
char *src;
|
|
bfd *abfd;
|
|
{
|
|
int eof = 0;
|
|
while (white (*src) && !eof)
|
|
{
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
}
|
|
return eof;
|
|
}
|
|
|
|
static boolean
|
|
DEFUN (srec_mkobject, (abfd),
|
|
bfd * abfd)
|
|
{
|
|
if (abfd->tdata.srec_data == 0)
|
|
{
|
|
tdata_type *tdata = (tdata_type *) bfd_alloc (abfd, sizeof (tdata_type));
|
|
if (!tdata)
|
|
{
|
|
bfd_error = no_memory;
|
|
return false;
|
|
}
|
|
abfd->tdata.srec_data = tdata;
|
|
tdata->type = 1;
|
|
tdata->head = (srec_data_list_type *) NULL;
|
|
}
|
|
return true;
|
|
|
|
}
|
|
|
|
static void
|
|
pass_over (abfd, func, symbolfunc, section)
|
|
bfd *abfd;
|
|
void (*func) ();
|
|
void (*symbolfunc) ();
|
|
asection *section;
|
|
{
|
|
unsigned int bytes_on_line;
|
|
boolean eof = false;
|
|
|
|
srec_mkobject (abfd);
|
|
/* To the front of the file */
|
|
bfd_seek (abfd, (file_ptr) 0, SEEK_SET);
|
|
while (eof == false)
|
|
{
|
|
char buffer[MAXCHUNK];
|
|
char *src = buffer;
|
|
char type;
|
|
bfd_vma address = 0;
|
|
|
|
/* Find first 'S' or $ */
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
switch (*src)
|
|
{
|
|
default:
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
if (eof)
|
|
return;
|
|
break;
|
|
|
|
case '$':
|
|
/* Inside a symbol definition - just ignore the module name */
|
|
while (*src != '\n' && !eof)
|
|
{
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
}
|
|
break;
|
|
|
|
case ' ':
|
|
/* spaces - maybe just before a symbol */
|
|
while (*src != '\n' && *src != '\r' && white (*src))
|
|
{
|
|
eof = skipwhite (src, abfd);
|
|
|
|
{
|
|
int val = 0;
|
|
int slen = 0;
|
|
char symbol[MAXCHUNK];
|
|
|
|
/* get the symbol part */
|
|
while (!eof && !white (*src) && slen < MAXCHUNK)
|
|
{
|
|
symbol[slen++] = *src;
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
}
|
|
symbol[slen] = 0;
|
|
eof = skipwhite (src, abfd);
|
|
/* skip the $ for the hex value */
|
|
if (*src == '$')
|
|
{
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
}
|
|
|
|
/* Scan off the hex number */
|
|
while (isxdigit (*src))
|
|
{
|
|
val *= 16;
|
|
if (isdigit (*src))
|
|
val += *src - '0';
|
|
else if (isupper (*src))
|
|
{
|
|
val += *src - 'A' + 10;
|
|
}
|
|
else
|
|
{
|
|
val += *src - 'a' + 10;
|
|
}
|
|
eof = (boolean) (bfd_read (src, 1, 1, abfd) != 1);
|
|
}
|
|
symbolfunc (abfd, symbol, slen, val);
|
|
}
|
|
}
|
|
break;
|
|
case 'S':
|
|
src++;
|
|
|
|
/* Fetch the type and the length */
|
|
bfd_read (src, 1, 3, abfd);
|
|
|
|
type = *src++;
|
|
|
|
if (!ISHEX (src[0]) || !ISHEX (src[1]))
|
|
break;
|
|
|
|
bytes_on_line = HEX (src);
|
|
|
|
if (bytes_on_line > MAXCHUNK / 2)
|
|
break;
|
|
src += 2;
|
|
|
|
bfd_read (src, 1, bytes_on_line * 2, abfd);
|
|
|
|
switch (type)
|
|
{
|
|
case '0':
|
|
case '5':
|
|
/* Prologue - ignore */
|
|
break;
|
|
case '3':
|
|
address = HEX (src);
|
|
src += 2;
|
|
bytes_on_line--;
|
|
|
|
case '2':
|
|
address = HEX (src) | (address << 8);
|
|
src += 2;
|
|
bytes_on_line--;
|
|
case '1':
|
|
address = HEX (src) | (address << 8);
|
|
src += 2;
|
|
address = HEX (src) | (address << 8);
|
|
src += 2;
|
|
bytes_on_line -= 2;
|
|
func (abfd, section, address, src, bytes_on_line);
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
static bfd_target *
|
|
object_p (abfd)
|
|
bfd *abfd;
|
|
{
|
|
asection *section;
|
|
/* We create one section called .text for all the contents,
|
|
and allocate enough room for the entire file. */
|
|
|
|
section = bfd_make_section (abfd, ".text");
|
|
section->_raw_size = 0;
|
|
section->vma = 0xffffffff;
|
|
low = 0xffffffff;
|
|
high = 0;
|
|
pass_over (abfd, size_srec, size_symbols, section);
|
|
section->_raw_size = high - low;
|
|
section->vma = low;
|
|
section->flags = SEC_HAS_CONTENTS | SEC_LOAD | SEC_ALLOC;
|
|
|
|
if (abfd->symcount)
|
|
abfd->flags |= HAS_SYMS;
|
|
return abfd->xvec;
|
|
}
|
|
|
|
static bfd_target *
|
|
DEFUN (srec_object_p, (abfd),
|
|
bfd * abfd)
|
|
{
|
|
char b[4];
|
|
|
|
srec_init ();
|
|
|
|
bfd_seek (abfd, (file_ptr) 0, SEEK_SET);
|
|
bfd_read (b, 1, 4, abfd);
|
|
|
|
if (b[0] != 'S' || !ISHEX (b[1]) || !ISHEX (b[2]) || !ISHEX (b[3]))
|
|
return (bfd_target *) NULL;
|
|
|
|
/* We create one section called .text for all the contents,
|
|
and allocate enough room for the entire file. */
|
|
|
|
return object_p (abfd);
|
|
}
|
|
|
|
|
|
static bfd_target *
|
|
DEFUN (symbolsrec_object_p, (abfd),
|
|
bfd * abfd)
|
|
{
|
|
char b[4];
|
|
|
|
srec_init ();
|
|
|
|
bfd_seek (abfd, (file_ptr) 0, SEEK_SET);
|
|
bfd_read (b, 1, 4, abfd);
|
|
|
|
if (b[0] != '$' || b[1] != '$')
|
|
return (bfd_target *) NULL;
|
|
|
|
return object_p (abfd);
|
|
}
|
|
|
|
|
|
static boolean
|
|
DEFUN (srec_get_section_contents, (abfd, section, location, offset, count),
|
|
bfd * abfd AND
|
|
asection * section AND
|
|
PTR location AND
|
|
file_ptr offset AND
|
|
bfd_size_type count)
|
|
{
|
|
if (section->used_by_bfd == (PTR) NULL)
|
|
{
|
|
section->used_by_bfd = (PTR) bfd_alloc (abfd, section->_raw_size);
|
|
if (!section->used_by_bfd)
|
|
{
|
|
bfd_error = no_memory;
|
|
return false;
|
|
}
|
|
|
|
pass_over (abfd, fillup, fillup_symbols, section);
|
|
}
|
|
memcpy ((PTR) location,
|
|
(PTR) ((char *) (section->used_by_bfd) + offset),
|
|
count);
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
boolean
|
|
DEFUN (srec_set_arch_mach, (abfd, arch, machine),
|
|
bfd * abfd AND
|
|
enum bfd_architecture arch AND
|
|
unsigned long machine)
|
|
{
|
|
return bfd_default_set_arch_mach (abfd, arch, machine);
|
|
}
|
|
|
|
|
|
/* we have to save up all the Srecords for a splurge before output,
|
|
also remember */
|
|
|
|
static boolean
|
|
DEFUN (srec_set_section_contents, (abfd, section, location, offset, bytes_to_do),
|
|
bfd * abfd AND
|
|
sec_ptr section AND
|
|
PTR location AND
|
|
file_ptr offset AND
|
|
bfd_size_type bytes_to_do)
|
|
{
|
|
tdata_type *tdata = abfd->tdata.srec_data;
|
|
srec_data_list_type *entry = (srec_data_list_type *)
|
|
bfd_alloc (abfd, sizeof (srec_data_list_type));
|
|
|
|
if (!entry)
|
|
{
|
|
bfd_error = no_memory;
|
|
return false;
|
|
}
|
|
|
|
if ((section->flags & SEC_ALLOC)
|
|
&& (section->flags & SEC_LOAD))
|
|
{
|
|
unsigned char *data = (unsigned char *) bfd_alloc (abfd, bytes_to_do);
|
|
if (!data)
|
|
{
|
|
bfd_error = no_memory;
|
|
return false;
|
|
}
|
|
memcpy (data, location, bytes_to_do);
|
|
|
|
if ((section->lma + offset + bytes_to_do) <= 0xffff)
|
|
{
|
|
|
|
}
|
|
else if ((section->lma + offset + bytes_to_do) <= 0xffffff
|
|
&& tdata->type < 2)
|
|
{
|
|
tdata->type = 2;
|
|
}
|
|
else
|
|
{
|
|
tdata->type = 3;
|
|
}
|
|
|
|
entry->data = data;
|
|
entry->where = section->lma + offset;
|
|
entry->size = bytes_to_do;
|
|
entry->next = tdata->head;
|
|
tdata->head = entry;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/* Write a record of type, of the supplied number of bytes. The
|
|
supplied bytes and length don't have a checksum. That's worked out
|
|
here
|
|
*/
|
|
static
|
|
void
|
|
DEFUN (srec_write_record, (abfd, type, address, data, end),
|
|
bfd * abfd AND
|
|
char type AND
|
|
bfd_vma address AND
|
|
CONST unsigned char *data AND
|
|
CONST unsigned char *end)
|
|
|
|
{
|
|
char buffer[MAXCHUNK];
|
|
|
|
unsigned int check_sum = 0;
|
|
CONST unsigned char *src = data;
|
|
char *dst = buffer;
|
|
char *length;
|
|
|
|
|
|
*dst++ = 'S';
|
|
*dst++ = '0' + type;
|
|
|
|
length = dst;
|
|
dst += 2; /* leave room for dst*/
|
|
|
|
switch (type)
|
|
{
|
|
case 3:
|
|
case 7:
|
|
TOHEX (dst, (address >> 24), check_sum);
|
|
dst += 2;
|
|
case 8:
|
|
case 2:
|
|
TOHEX (dst, (address >> 16), check_sum);
|
|
dst += 2;
|
|
case 9:
|
|
case 1:
|
|
case 0:
|
|
TOHEX (dst, (address >> 8), check_sum);
|
|
dst += 2;
|
|
TOHEX (dst, (address), check_sum);
|
|
dst += 2;
|
|
break;
|
|
|
|
}
|
|
for (src = data; src < end; src++)
|
|
{
|
|
TOHEX (dst, *src, check_sum);
|
|
dst += 2;
|
|
}
|
|
|
|
/* Fill in the length */
|
|
TOHEX (length, (dst - length) / 2, check_sum);
|
|
check_sum &= 0xff;
|
|
check_sum = 255 - check_sum;
|
|
TOHEX (dst, check_sum, check_sum);
|
|
dst += 2;
|
|
|
|
*dst++ = '\r';
|
|
*dst++ = '\n';
|
|
bfd_write ((PTR) buffer, 1, dst - buffer, abfd);
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
DEFUN (srec_write_header, (abfd),
|
|
bfd * abfd)
|
|
{
|
|
unsigned char buffer[MAXCHUNK];
|
|
unsigned char *dst = buffer;
|
|
unsigned int i;
|
|
|
|
/* I'll put an arbitary 40 char limit on header size */
|
|
for (i = 0; i < 40 && abfd->filename[i]; i++)
|
|
{
|
|
*dst++ = abfd->filename[i];
|
|
}
|
|
srec_write_record (abfd, 0, 0, buffer, dst);
|
|
}
|
|
|
|
static void
|
|
DEFUN (srec_write_section, (abfd, tdata, list),
|
|
bfd * abfd AND
|
|
tdata_type * tdata AND
|
|
srec_data_list_type * list)
|
|
{
|
|
unsigned int bytes_written = 0;
|
|
unsigned char *location = list->data;
|
|
|
|
while (bytes_written < list->size)
|
|
{
|
|
bfd_vma address;
|
|
|
|
unsigned int bytes_this_chunk = list->size - bytes_written;
|
|
|
|
if (bytes_this_chunk > CHUNK)
|
|
{
|
|
bytes_this_chunk = CHUNK;
|
|
}
|
|
|
|
address = list->where + bytes_written;
|
|
|
|
srec_write_record (abfd,
|
|
tdata->type,
|
|
address,
|
|
location,
|
|
location + bytes_this_chunk);
|
|
|
|
bytes_written += bytes_this_chunk;
|
|
location += bytes_this_chunk;
|
|
}
|
|
|
|
}
|
|
|
|
static void
|
|
DEFUN (srec_write_terminator, (abfd, tdata),
|
|
bfd * abfd AND
|
|
tdata_type * tdata)
|
|
{
|
|
unsigned char buffer[2];
|
|
|
|
srec_write_record (abfd, 10 - tdata->type,
|
|
abfd->start_address, buffer, buffer);
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
srec_write_symbols (abfd)
|
|
bfd *abfd;
|
|
{
|
|
char buffer[MAXCHUNK];
|
|
/* Dump out the symbols of a bfd */
|
|
int i;
|
|
int len = bfd_get_symcount (abfd);
|
|
|
|
if (len)
|
|
{
|
|
asymbol **table = bfd_get_outsymbols (abfd);
|
|
sprintf (buffer, "$$ %s\r\n", abfd->filename);
|
|
|
|
bfd_write (buffer, strlen (buffer), 1, abfd);
|
|
|
|
for (i = 0; i < len; i++)
|
|
{
|
|
asymbol *s = table[i];
|
|
#if 0
|
|
int len = strlen (s->name);
|
|
|
|
/* If this symbol has a .[ocs] in it, it's probably a file name
|
|
and we'll output that as the module name */
|
|
|
|
if (len > 3 && s->name[len - 2] == '.')
|
|
{
|
|
int l;
|
|
sprintf (buffer, "$$ %s\r\n", s->name);
|
|
l = strlen (buffer);
|
|
bfd_write (buffer, l, 1, abfd);
|
|
}
|
|
else
|
|
#endif
|
|
if (s->flags & (BSF_GLOBAL | BSF_LOCAL)
|
|
&& (s->flags & BSF_DEBUGGING) == 0
|
|
&& s->name[0] != '.'
|
|
&& s->name[0] != 't')
|
|
{
|
|
/* Just dump out non debug symbols */
|
|
|
|
int l;
|
|
char buf2[40], *p;
|
|
|
|
sprintf_vma (buf2,
|
|
s->value + s->section->output_section->lma
|
|
+ s->section->output_offset);
|
|
p = buf2;
|
|
while (p[0] == '0' && p[1] != 0)
|
|
p++;
|
|
sprintf (buffer, " %s $%s\r\n", s->name, p);
|
|
l = strlen (buffer);
|
|
bfd_write (buffer, l, 1, abfd);
|
|
}
|
|
}
|
|
sprintf (buffer, "$$ \r\n");
|
|
bfd_write (buffer, strlen (buffer), 1, abfd);
|
|
}
|
|
}
|
|
|
|
static boolean
|
|
internal_srec_write_object_contents (abfd, symbols)
|
|
bfd *abfd;
|
|
int symbols;
|
|
{
|
|
tdata_type *tdata = abfd->tdata.srec_data;
|
|
srec_data_list_type *list;
|
|
|
|
if (symbols)
|
|
srec_write_symbols (abfd);
|
|
|
|
srec_write_header (abfd);
|
|
|
|
/* Now wander though all the sections provided and output them */
|
|
list = tdata->head;
|
|
|
|
while (list != (srec_data_list_type *) NULL)
|
|
{
|
|
srec_write_section (abfd, tdata, list);
|
|
list = list->next;
|
|
}
|
|
srec_write_terminator (abfd, tdata);
|
|
return true;
|
|
}
|
|
|
|
static boolean
|
|
srec_write_object_contents (abfd)
|
|
bfd *abfd;
|
|
{
|
|
return internal_srec_write_object_contents (abfd, 0);
|
|
}
|
|
|
|
static boolean
|
|
symbolsrec_write_object_contents (abfd)
|
|
bfd *abfd;
|
|
{
|
|
return internal_srec_write_object_contents (abfd, 1);
|
|
}
|
|
|
|
/*ARGSUSED*/
|
|
static int
|
|
DEFUN (srec_sizeof_headers, (abfd, exec),
|
|
bfd * abfd AND
|
|
boolean exec)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
static asymbol *
|
|
DEFUN (srec_make_empty_symbol, (abfd),
|
|
bfd * abfd)
|
|
{
|
|
asymbol *new = (asymbol *) bfd_zalloc (abfd, sizeof (asymbol));
|
|
if (new)
|
|
new->the_bfd = abfd;
|
|
return new;
|
|
}
|
|
|
|
static unsigned int
|
|
srec_get_symtab_upper_bound (abfd)
|
|
bfd *abfd;
|
|
{
|
|
/* Read in all the info */
|
|
srec_get_section_contents (abfd, abfd->sections, 0, 0, 0);
|
|
return (bfd_get_symcount (abfd) + 1) * (sizeof (asymbol *));
|
|
}
|
|
|
|
static unsigned int
|
|
DEFUN (srec_get_symtab, (abfd, alocation),
|
|
bfd * abfd AND
|
|
asymbol ** alocation)
|
|
{
|
|
int lim = abfd->symcount;
|
|
int i;
|
|
for (i = 0; i < lim; i++)
|
|
{
|
|
alocation[i] = abfd->tdata.srec_data->symbols + i;
|
|
}
|
|
alocation[i] = 0;
|
|
return lim;
|
|
}
|
|
|
|
/*ARGSUSED*/
|
|
void
|
|
DEFUN (srec_get_symbol_info, (ignore_abfd, symbol, ret),
|
|
bfd * ignore_abfd AND
|
|
asymbol * symbol AND
|
|
symbol_info * ret)
|
|
{
|
|
bfd_symbol_info (symbol, ret);
|
|
}
|
|
|
|
/*ARGSUSED*/
|
|
void
|
|
DEFUN (srec_print_symbol, (ignore_abfd, afile, symbol, how),
|
|
bfd * ignore_abfd AND
|
|
PTR afile AND
|
|
asymbol * symbol AND
|
|
bfd_print_symbol_type how)
|
|
{
|
|
FILE *file = (FILE *) afile;
|
|
switch (how)
|
|
{
|
|
case bfd_print_symbol_name:
|
|
fprintf (file, "%s", symbol->name);
|
|
break;
|
|
default:
|
|
bfd_print_symbol_vandf ((PTR) file, symbol);
|
|
fprintf (file, " %-5s %s",
|
|
symbol->section->name,
|
|
symbol->name);
|
|
|
|
}
|
|
}
|
|
|
|
#define FOO PROTO
|
|
#define srec_new_section_hook (FOO(boolean, (*), (bfd *, asection *)))bfd_true
|
|
|
|
#define srec_get_reloc_upper_bound (FOO(unsigned int, (*),(bfd*, asection *)))bfd_false
|
|
#define srec_canonicalize_reloc (FOO(unsigned int, (*),(bfd*,asection *, arelent **, asymbol **))) bfd_0
|
|
|
|
|
|
|
|
#define srec_openr_next_archived_file (FOO(bfd *, (*), (bfd*,bfd*))) bfd_nullvoidptr
|
|
#define srec_find_nearest_line (FOO(boolean, (*),(bfd*,asection*,asymbol**,bfd_vma, CONST char**, CONST char**, unsigned int *))) bfd_false
|
|
#define srec_generic_stat_arch_elt (FOO(int, (*), (bfd *,struct stat *))) bfd_0
|
|
|
|
|
|
#define srec_core_file_failing_command (char *(*)())(bfd_nullvoidptr)
|
|
#define srec_core_file_failing_signal (int (*)())bfd_0
|
|
#define srec_core_file_matches_executable_p (FOO(boolean, (*),(bfd*, bfd*)))bfd_false
|
|
#define srec_slurp_armap bfd_true
|
|
#define srec_slurp_extended_name_table bfd_true
|
|
#define srec_truncate_arname (void (*)())bfd_nullvoidptr
|
|
#define srec_write_armap (FOO( boolean, (*),(bfd *, unsigned int, struct orl *, unsigned int, int))) bfd_nullvoidptr
|
|
#define srec_get_lineno (struct lineno_cache_entry *(*)())bfd_nullvoidptr
|
|
#define srec_close_and_cleanup bfd_generic_close_and_cleanup
|
|
#define srec_bfd_debug_info_start bfd_void
|
|
#define srec_bfd_debug_info_end bfd_void
|
|
#define srec_bfd_debug_info_accumulate (FOO(void, (*), (bfd *, asection *))) bfd_void
|
|
#define srec_bfd_get_relocated_section_contents bfd_generic_get_relocated_section_contents
|
|
#define srec_bfd_relax_section bfd_generic_relax_section
|
|
#define srec_bfd_reloc_type_lookup \
|
|
((CONST struct reloc_howto_struct *(*) PARAMS ((bfd *, bfd_reloc_code_real_type))) bfd_nullvoidptr)
|
|
#define srec_bfd_make_debug_symbol \
|
|
((asymbol *(*) PARAMS ((bfd *, void *, unsigned long))) bfd_nullvoidptr)
|
|
#define srec_bfd_link_hash_table_create _bfd_generic_link_hash_table_create
|
|
#define srec_bfd_link_add_symbols _bfd_generic_link_add_symbols
|
|
#define srec_bfd_final_link _bfd_generic_final_link
|
|
|
|
bfd_target srec_vec =
|
|
{
|
|
"srec", /* name */
|
|
bfd_target_srec_flavour,
|
|
true, /* target byte order */
|
|
true, /* target headers byte order */
|
|
(HAS_RELOC | EXEC_P | /* object flags */
|
|
HAS_LINENO | HAS_DEBUG |
|
|
HAS_SYMS | HAS_LOCALS | WP_TEXT | D_PAGED),
|
|
(SEC_CODE | SEC_DATA | SEC_ROM | SEC_HAS_CONTENTS
|
|
| SEC_ALLOC | SEC_LOAD | SEC_RELOC), /* section flags */
|
|
0, /* leading underscore */
|
|
' ', /* ar_pad_char */
|
|
16, /* ar_max_namelen */
|
|
1, /* minimum alignment */
|
|
bfd_getb64, bfd_getb_signed_64, bfd_putb64,
|
|
bfd_getb32, bfd_getb_signed_32, bfd_putb32,
|
|
bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* data */
|
|
bfd_getb64, bfd_getb_signed_64, bfd_putb64,
|
|
bfd_getb32, bfd_getb_signed_32, bfd_putb32,
|
|
bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* hdrs */
|
|
|
|
{
|
|
_bfd_dummy_target,
|
|
srec_object_p, /* bfd_check_format */
|
|
(struct bfd_target * (*)()) bfd_nullvoidptr,
|
|
(struct bfd_target * (*)()) bfd_nullvoidptr,
|
|
},
|
|
{
|
|
bfd_false,
|
|
srec_mkobject,
|
|
_bfd_generic_mkarchive,
|
|
bfd_false,
|
|
},
|
|
{ /* bfd_write_contents */
|
|
bfd_false,
|
|
srec_write_object_contents,
|
|
_bfd_write_archive_contents,
|
|
bfd_false,
|
|
},
|
|
JUMP_TABLE (srec)
|
|
};
|
|
|
|
|
|
|
|
bfd_target symbolsrec_vec =
|
|
{
|
|
"symbolsrec", /* name */
|
|
bfd_target_srec_flavour,
|
|
true, /* target byte order */
|
|
true, /* target headers byte order */
|
|
(HAS_RELOC | EXEC_P | /* object flags */
|
|
HAS_LINENO | HAS_DEBUG |
|
|
HAS_SYMS | HAS_LOCALS | WP_TEXT | D_PAGED),
|
|
(SEC_CODE | SEC_DATA | SEC_ROM | SEC_HAS_CONTENTS
|
|
| SEC_ALLOC | SEC_LOAD | SEC_RELOC), /* section flags */
|
|
0, /* leading underscore */
|
|
' ', /* ar_pad_char */
|
|
16, /* ar_max_namelen */
|
|
1, /* minimum alignment */
|
|
bfd_getb64, bfd_getb_signed_64, bfd_putb64,
|
|
bfd_getb32, bfd_getb_signed_32, bfd_putb32,
|
|
bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* data */
|
|
bfd_getb64, bfd_getb_signed_64, bfd_putb64,
|
|
bfd_getb32, bfd_getb_signed_32, bfd_putb32,
|
|
bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* hdrs */
|
|
|
|
{
|
|
_bfd_dummy_target,
|
|
symbolsrec_object_p, /* bfd_check_format */
|
|
(struct bfd_target * (*)()) bfd_nullvoidptr,
|
|
(struct bfd_target * (*)()) bfd_nullvoidptr,
|
|
},
|
|
{
|
|
bfd_false,
|
|
srec_mkobject,
|
|
_bfd_generic_mkarchive,
|
|
bfd_false,
|
|
},
|
|
{ /* bfd_write_contents */
|
|
bfd_false,
|
|
symbolsrec_write_object_contents,
|
|
_bfd_write_archive_contents,
|
|
bfd_false,
|
|
},
|
|
JUMP_TABLE (srec),
|
|
(PTR) 0
|
|
};
|