* coff-mips.c (ecoff_set_symbol_info): stBlock symbols are always
[platform/upstream/binutils.git] / bfd / coff-mips.c
index 775feef..c5b3946 100644 (file)
@@ -1,7 +1,7 @@
 /* BFD back-end for MIPS Extended-Coff files.
-   Copyright 1990, 1991, 1992 Free Software Foundation, Inc.
+   Copyright 1990, 1991, 1992, 1993 Free Software Foundation, Inc.
    Original version by Per Bothner.
-   Full support by Ian Lance Taylor, ian@cygnus.com.
+   Full support added by Ian Lance Taylor, ian@cygnus.com.
 
 This file is part of BFD, the Binary File Descriptor library.
 
@@ -23,58 +23,15 @@ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
 #include "sysdep.h"
 #include "libbfd.h"
 #include "seclet.h"
+#include "aout/ar.h"
+#include "aout/ranlib.h"
 #include "coff/mips.h"
 #include "coff/internal.h"
 #include "coff/sym.h"
 #include "coff/symconst.h"
 #include "coff/ecoff-ext.h"
 #include "libcoff.h"
-
-/* `Tdata' information kept for ECOFF files.  */
-
-#define ecoff_data(abfd) ((abfd)->tdata.ecoff_obj_data)
-
-typedef struct ecoff_tdata
-{
-  /* The reloc file position, set by
-     ecoff_compute_section_file_positions.  */
-  file_ptr reloc_filepos;
-
-  /* The symbol table file position, set by ecoff_mkobject_hook.  */
-  file_ptr sym_filepos;
-
-  /* The cached gp value.  This is used when relocating.  */
-  bfd_vma gp;
-
-  /* The size of the unswapped ECOFF symbolic information.  */
-  bfd_size_type raw_size;
-
-  /* The unswapped ECOFF symbolic information.  */
-  PTR raw_syments;
-
-  /* The swapped ECOFF symbolic header.  */
-  HDRR symbolic_header;
-
-  /* Pointers to the unswapped symbolic information.  */
-  unsigned char *line;
-  struct dnr_ext *external_dnr;
-  struct pdr_ext *external_pdr;
-  struct sym_ext *external_sym;
-  struct opt_ext *external_opt;
-  union aux_ext *external_aux;
-  char *ss;
-  char *ssext;
-  struct fdr_ext *external_fdr;
-  struct rfd_ext *external_rfd;
-  struct ext_ext *external_ext;
-
-  /* The swapped fdr information.  */
-  FDR *fdr;
-
-  /* The canonical BFD symbols.  */
-  struct ecoff_symbol_struct *canonical_symbols;
-
-} ecoff_data_type;
+#include "libecoff.h"
 
 /* Each canonical asymbol really looks like this.  */
 
@@ -102,14 +59,128 @@ typedef struct ecoff_symbol_struct
    macro is only ever applied to an asymbol.  */
 #define ecoffsymbol(asymbol) ((ecoff_symbol_type *) (&((asymbol)->the_bfd)))
 
+/* This is a hack borrowed from coffcode.h; we need to save the index
+   of a symbol when we write it out so that can set the symbol index
+   correctly when we write out the relocs.  */
+#define ecoff_get_sym_index(symbol) ((unsigned long) (symbol)->udata)
+#define ecoff_set_sym_index(symbol, idx) ((symbol)->udata = (PTR) (idx))
+
+/* The page boundary used to align sections in the executable file.  */
+#define PAGE_SIZE 0x2000
+
+/* The linker needs a section to hold small common variables while
+   linking.  There is no convenient way to create it when the linker
+   needs it, so we always create one for each BFD.  We then avoid
+   writing it out.  */
+#define SCOMMON ".scommon"
+
 /* MIPS ECOFF has COFF sections, but the debugging information is
    stored in a completely different format.  This files uses the some
    of the swapping routines from coffswap.h, and some of the generic
    COFF routines in coffgen.c, but, unlike the real COFF targets, does
    not use coffcode.h itself.  */
-
+\f
+/* Prototypes for static functions.  */
+
+static boolean ecoff_bad_format_hook PARAMS ((bfd *abfd, PTR filehdr));
+static asection *ecoff_make_section_hook PARAMS ((bfd *abfd, char *name));
+static boolean ecoff_new_section_hook PARAMS ((bfd *abfd, asection *section));
+static boolean ecoff_mkobject PARAMS ((bfd *abfd));
+static PTR ecoff_mkobject_hook PARAMS ((bfd *abfd, PTR filehdr, PTR aouthdr));
+static boolean ecoff_set_arch_mach_hook PARAMS ((bfd *abfd, PTR filehdr));
+static long ecoff_sec_to_styp_flags PARAMS ((CONST char *name,
+                                            flagword flags));
+static flagword ecoff_styp_to_sec_flags PARAMS ((bfd *abfd, PTR hdr));
+static asymbol *ecoff_make_empty_symbol PARAMS ((bfd *abfd));
+static void ecoff_set_symbol_info PARAMS ((bfd *abfd, SYMR *ecoff_sym,
+                                          asymbol *asym, int ext));
+static boolean ecoff_slurp_symbol_table PARAMS ((bfd *abfd));
+static unsigned int ecoff_get_symtab_upper_bound PARAMS ((bfd *abfd));
+static unsigned int ecoff_get_symtab PARAMS ((bfd *abfd,
+                                             asymbol **alocation));
+static void ecoff_emit_aggregate PARAMS ((bfd *abfd, char *string,
+                                         RNDXR *rndx, long isym,
+                                         CONST char *which));
+static char *ecoff_type_to_string PARAMS ((bfd *abfd, union aux_ext *aux_ptr,
+                                          int indx, int bigendian));
+static void ecoff_print_symbol PARAMS ((bfd *abfd, PTR filep,
+                                       asymbol *symbol,
+                                       bfd_print_symbol_type how));
+static void ecoff_swap_reloc_in PARAMS ((bfd *abfd, RELOC *ext,
+                                        struct internal_reloc *intern));
+static unsigned int ecoff_swap_reloc_out PARAMS ((bfd *abfd, PTR src,
+                                                 PTR dst));
+static bfd_reloc_status_type ecoff_generic_reloc PARAMS ((bfd *abfd,
+                                                         arelent *reloc,
+                                                         asymbol *symbol,
+                                                         PTR data,
+                                                         asection *section,
+                                                         bfd *output_bfd));
+static bfd_reloc_status_type ecoff_refhi_reloc PARAMS ((bfd *abfd,
+                                                       arelent *reloc,
+                                                       asymbol *symbol,
+                                                       PTR data,
+                                                       asection *section,
+                                                       bfd *output_bfd));
+static bfd_reloc_status_type ecoff_gprel_reloc PARAMS ((bfd *abfd,
+                                                       arelent *reloc,
+                                                       asymbol *symbol,
+                                                       PTR data,
+                                                       asection *section,
+                                                       bfd *output_bfd));
+static boolean ecoff_slurp_reloc_table PARAMS ((bfd *abfd, asection *section,
+                                               asymbol **symbols));
+static unsigned int ecoff_canonicalize_reloc PARAMS ((bfd *abfd,
+                                                     asection *section,
+                                                     arelent **relptr,
+                                                     asymbol **symbols));
+static boolean ecoff_find_nearest_line PARAMS ((bfd *abfd,
+                                               asection *section,
+                                               asymbol **symbols,
+                                               bfd_vma offset,
+                                               CONST char **filename_ptr,
+                                               CONST char **fnname_ptr,
+                                               unsigned int *retline_ptr));
+static void ecoff_clear_output_flags PARAMS ((bfd *abfd));
+static boolean ecoff_rel PARAMS ((bfd *output_bfd, bfd_seclet_type *seclet,
+                                 asection *output_section, PTR data,
+                                 boolean relocateable));
+static boolean ecoff_dump_seclet PARAMS ((bfd *abfd, bfd_seclet_type *seclet,
+                                         asection *section, PTR data,
+                                         boolean relocateable));
+static long ecoff_add_string PARAMS ((bfd *output_bfd, FDR *fdr,
+                                     CONST char *string, boolean external));
+static boolean ecoff_get_debug PARAMS ((bfd *output_bfd,
+                                       bfd_seclet_type *seclet,
+                                       asection *section,
+                                       boolean relocateable));
+static boolean ecoff_bfd_seclet_link PARAMS ((bfd *abfd, PTR data,
+                                             boolean relocateable));
+static boolean ecoff_set_arch_mach PARAMS ((bfd *abfd,
+                                           enum bfd_architecture arch,
+                                           unsigned long machine));
+static int ecoff_sizeof_headers PARAMS ((bfd *abfd, boolean reloc));
+static void ecoff_compute_section_file_positions PARAMS ((bfd *abfd));
+static boolean ecoff_set_section_contents PARAMS ((bfd *abfd,
+                                                  asection *section,
+                                                  PTR location,
+                                                  file_ptr offset,
+                                                  bfd_size_type count));
+static boolean ecoff_write_object_contents PARAMS ((bfd *abfd));
+static unsigned int ecoff_armap_hash PARAMS ((CONST char *s,
+                                             unsigned int *rehash,
+                                             unsigned int size,
+                                             unsigned int hlog));
+static boolean ecoff_slurp_armap PARAMS ((bfd *abfd));
+static boolean ecoff_write_armap PARAMS ((bfd *abfd, unsigned int elength,
+                                         struct orl *map,
+                                         unsigned int orl_count,
+                                         int stridx));
+static bfd_target *ecoff_archive_p PARAMS ((bfd *abfd));
+\f
 /* Get the generic COFF swapping routines, except for the reloc,
    symbol, and lineno ones.  Give them ecoff names.  */
+#define MIPSECOFF
 #define NO_COFF_RELOCS
 #define NO_COFF_SYMBOLS
 #define NO_COFF_LINENOS
@@ -128,9 +199,9 @@ static asection bfd_debug_section = { "*DEBUG*" };
 /* See whether the magic number matches.  */
 
 static boolean
-DEFUN(ecoff_bad_format_hook, (abfd, filehdr),
-      bfd *abfd AND
-      PTR filehdr)
+ecoff_bad_format_hook (abfd, filehdr)
+     bfd *abfd;
+     PTR filehdr;
 {
   struct internal_filehdr *internal_f = (struct internal_filehdr *) filehdr;
 
@@ -143,9 +214,9 @@ DEFUN(ecoff_bad_format_hook, (abfd, filehdr),
 /* This is a hook needed by SCO COFF, but we have nothing to do.  */
 
 static asection *
-DEFUN (ecoff_make_section_hook, (abfd, name),
-       bfd *abfd AND
-       char *name)
+ecoff_make_section_hook (abfd, name)
+     bfd *abfd;
+     char *name;
 {
   return (asection *) NULL;
 }
@@ -153,9 +224,9 @@ DEFUN (ecoff_make_section_hook, (abfd, name),
 /* Initialize a new section.  */
 
 static boolean
-DEFUN (ecoff_new_section_hook, (abfd, section),
-       bfd *abfd AND
-       asection *section)
+ecoff_new_section_hook (abfd, section)
+     bfd *abfd;
+     asection *section;
 {
   section->alignment_power = abfd->xvec->align_power_min;
 
@@ -179,12 +250,16 @@ DEFUN (ecoff_new_section_hook, (abfd, section),
   return true;
 }
 
+/* Set the alignment of a section; we have nothing to do.  */
+
 #define ecoff_set_alignment_hook \
   ((void (*) PARAMS ((bfd *, asection *, PTR))) bfd_void)
 
+/* Create an ECOFF object.  */
+
 static boolean
-DEFUN (ecoff_mkobject, (abfd),
-       bfd *abfd)
+ecoff_mkobject (abfd)
+     bfd *abfd;
 {
   abfd->tdata.ecoff_obj_data = ((struct ecoff_tdata *)
                                bfd_zalloc (abfd, sizeof(ecoff_data_type)));
@@ -194,17 +269,23 @@ DEFUN (ecoff_mkobject, (abfd),
       return false;
     }
 
+  /* Always create a .scommon section for every BFD.  This is a hack so
+     that the linker has something to attach scSCommon symbols to.  */
+  bfd_make_section (abfd, SCOMMON);
+
   return true;
 }
 
-/* Create the COFF backend specific information.  */
+/* Create the ECOFF backend specific information.  */
 
-static          PTR
-DEFUN(ecoff_mkobject_hook,(abfd, filehdr),
-      bfd            *abfd AND
-      PTR           filehdr)
+static PTR
+ecoff_mkobject_hook (abfd, filehdr, aouthdr)
+     bfd *abfd;
+     PTR filehdr;
+     PTR aouthdr;
 {
   struct internal_filehdr *internal_f = (struct internal_filehdr *) filehdr;
+  struct internal_aouthdr *internal_a = (struct internal_aouthdr *) aouthdr;
   ecoff_data_type *ecoff;
 
   if (ecoff_mkobject (abfd) == false)
@@ -212,43 +293,56 @@ DEFUN(ecoff_mkobject_hook,(abfd, filehdr),
 
   ecoff = ecoff_data (abfd);
   ecoff->sym_filepos = internal_f->f_symptr;
+
+  if (internal_a != (struct internal_aouthdr *) NULL)
+    {
+      int i;
+
+      ecoff->text_start = internal_a->text_start;
+      ecoff->text_end = internal_a->text_start + internal_a->tsize;
+      ecoff->gp = internal_a->gp_value;
+      ecoff->gprmask = internal_a->gprmask;
+      for (i = 0; i < 4; i++)
+       ecoff->cprmask[i] = internal_a->cprmask[i];
+    }
+
   return (PTR) ecoff;
 }
 
 /* Determine the machine architecture and type.  */
+
 static boolean
-DEFUN (ecoff_set_arch_mach_hook, (abfd, filehdr),
-       bfd *abfd AND
-       PTR filehdr)
+ecoff_set_arch_mach_hook (abfd, filehdr)
+     bfd *abfd;
+     PTR filehdr;
 {
-  long machine;
-  enum bfd_architecture arch;
   struct internal_filehdr *internal_f = (struct internal_filehdr *) filehdr;
+  enum bfd_architecture arch;
 
-  machine = 0;
-  switch (internal_f->f_magic) {
-  case  MIPS_MAGIC_1:
-  case  MIPS_MAGIC_2:
-  case  MIPS_MAGIC_3:
-    arch = bfd_arch_mips;
-    machine = 0;
-    break;
-
-  default:                     /* Unreadable input file type */
-    arch = bfd_arch_obscure;
-    break;
-  }
+  switch (internal_f->f_magic)
+    {
+    case MIPS_MAGIC_1:
+    case MIPS_MAGIC_LITTLE:
+    case MIPS_MAGIC_BIG:
+      arch = bfd_arch_mips;
+      break;
+
+    default:
+      arch = bfd_arch_obscure;
+      break;
+    }
+
+  bfd_default_set_arch_mach (abfd, arch, (unsigned long) 0);
 
-  bfd_default_set_arch_mach(abfd, arch, machine);
   return true;
 }
 
 /* Get the section s_flags to use for a section.  */
 
 static long
-DEFUN (sec_to_styp_flags, (name, flags),
-       CONST char *name AND
-       flagword flags)
+ecoff_sec_to_styp_flags (name, flags)
+     CONST char *name;
+     flagword flags;
 {
   long styp;
 
@@ -257,9 +351,9 @@ DEFUN (sec_to_styp_flags, (name, flags),
   if (strcmp (name, _TEXT) == 0)
     styp = STYP_TEXT;
   else if (strcmp (name, _DATA) == 0)
-    styp == STYP_DATA;
+    styp = STYP_DATA;
   else if (strcmp (name, _SDATA) == 0)
-    styp == STYP_SDATA;
+    styp = STYP_SDATA;
   else if (strcmp (name, _RDATA) == 0)
     styp = STYP_RDATA;
   else if (strcmp (name, _LIT8) == 0)
@@ -290,9 +384,9 @@ DEFUN (sec_to_styp_flags, (name, flags),
 /* Get the BFD flags to use for a section.  */
 
 static flagword
-DEFUN (styp_to_sec_flags, (abfd, hdr),
-       bfd *abfd AND
-       PTR hdr)
+ecoff_styp_to_sec_flags (abfd, hdr)
+     bfd *abfd;
+     PTR hdr;
 {
   struct internal_scnhdr *internal_s = (struct internal_scnhdr *) hdr;
   long styp_flags = internal_s->s_flags;
@@ -344,11 +438,13 @@ DEFUN (styp_to_sec_flags, (abfd, hdr),
 }
 \f
 /* Read in and swap the important symbolic information for an ECOFF
-   object file.  */
+   object file.  FIXME: This is called by gdb.  If there is ever
+   another ECOFF target, it should be moved into some sort of target
+   specific structure.  */
 
-static boolean
-DEFUN (ecoff_slurp_symbolic_info, (abfd),
-       bfd *abfd)
+boolean
+ecoff_slurp_symbolic_info (abfd)
+     bfd *abfd;
 {
   struct hdr_ext external_symhdr;
   HDRR *internal_symhdr;
@@ -400,11 +496,38 @@ DEFUN (ecoff_slurp_symbolic_info, (abfd),
   bfd_get_symcount (abfd) = (internal_symhdr->isymMax
                             + internal_symhdr->iextMax);
 
-  /* Read all the symbolic information at once.  This expression
-     assumes that the external symbols are always the last item.  */
-  raw_base = ecoff_data (abfd)->sym_filepos + sizeof (external_symhdr);
-  raw_size = (internal_symhdr->cbExtOffset - raw_base
-             + internal_symhdr->iextMax * sizeof (struct ext_ext));
+  /* Read all the symbolic information at once.  */
+  raw_base = ecoff_data (abfd)->sym_filepos + sizeof (struct hdr_ext);
+
+  if (internal_symhdr->cbExtOffset != 0)
+    raw_size = (internal_symhdr->cbExtOffset
+               - raw_base
+               + internal_symhdr->iextMax * sizeof (struct ext_ext));
+  else
+    {
+      long cbline, issmax, issextmax;
+
+      cbline = (internal_symhdr->cbLine + 3) &~ 3;
+      issmax = (internal_symhdr->issMax + 3) &~ 3;
+      issextmax = (internal_symhdr->issExtMax + 3) &~ 3;
+      raw_size = (cbline * sizeof (unsigned char)
+                 + internal_symhdr->idnMax * sizeof (struct dnr_ext)
+                 + internal_symhdr->ipdMax * sizeof (struct pdr_ext)
+                 + internal_symhdr->isymMax * sizeof (struct sym_ext)
+                 + internal_symhdr->ioptMax * sizeof (struct opt_ext)
+                 + internal_symhdr->iauxMax * sizeof (union aux_ext)
+                 + issmax * sizeof (char)
+                 + issextmax * sizeof (char)
+                 + internal_symhdr->ifdMax * sizeof (struct fdr_ext)
+                 + internal_symhdr->crfd * sizeof (struct rfd_ext)
+                 + internal_symhdr->iextMax * sizeof (struct ext_ext));
+    }
+
+  if (raw_size == 0)
+    {
+      ecoff_data (abfd)->sym_filepos = 0;
+      return true;
+    }
   raw = (PTR) bfd_alloc (abfd, raw_size);
   if (raw == NULL)
     {
@@ -470,11 +593,20 @@ DEFUN (ecoff_slurp_symbolic_info, (abfd),
 /* ECOFF symbol table routines.  The ECOFF symbol table is described
    in gcc/mips-tfile.c.  */
 
+/* ECOFF uses two common sections.  One is the usual one, and the
+   other is for small objects.  All the small objects are kept
+   together, and then referenced via the gp pointer, which yields
+   faster assembler code.  This is what we use for the small common
+   section.  */
+static asection ecoff_scom_section;
+static asymbol ecoff_scom_symbol;
+static asymbol *ecoff_scom_symbol_ptr;
+
 /* Create an empty symbol.  */
 
 static asymbol *
-DEFUN (ecoff_make_empty_symbol, (abfd),
-       bfd *abfd)
+ecoff_make_empty_symbol (abfd)
+     bfd *abfd;
 {
   ecoff_symbol_type *new;
 
@@ -495,16 +627,32 @@ DEFUN (ecoff_make_empty_symbol, (abfd),
 /* Set the BFD flags and section for an ECOFF symbol.  */
 
 static void
-DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
-       bfd *abfd AND
-       SYMR *ecoff_sym AND
-       asymbol *asym AND
-       int ext)
+ecoff_set_symbol_info (abfd, ecoff_sym, asym, ext)
+     bfd *abfd;
+     SYMR *ecoff_sym;
+     asymbol *asym;
+     int ext;
 {
   asym->the_bfd = abfd;
   asym->value = ecoff_sym->value;
   asym->section = &bfd_debug_section;
   asym->udata = NULL;
+
+  /* Most symbol types are just for debugging.  */
+  switch (ecoff_sym->st)
+    {
+    case stGlobal:
+    case stStatic:
+    case stLabel:
+    case stProc:
+    case stStaticProc:
+    case stNil:
+      break;
+    default:
+      asym->flags = BSF_DEBUGGING;
+      return;
+    }
+
   if (ext)
     asym->flags = BSF_EXPORT | BSF_GLOBAL;
   else
@@ -512,7 +660,11 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
   switch (ecoff_sym->sc)
     {
     case scNil:
-      asym->flags = 0;
+      /* Used for compiler generated labels.  Leave them in the
+        debugging section, and mark them as local.  If BSF_DEBUGGING
+        is set, then nm does not display them for some reason.  If no
+        flags are set then the linker whines about them.  */
+      asym->flags = BSF_LOCAL;
       break;
     case scText:
       asym->section = bfd_make_section_old_way (abfd, ".text");
@@ -523,19 +675,27 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
       asym->value -= asym->section->vma;
       break;
     case scBss:
-      asym->section = &bfd_com_section;
-      asym->flags = 0;
+      if (ext)
+       {
+         asym->section = &bfd_com_section;
+         asym->flags = 0;
+       }
+      else
+       {
+         asym->section = bfd_make_section_old_way (abfd, ".bss");
+         asym->value -= asym->section->vma;
+       }
       break;
     case scRegister:
       asym->flags = BSF_DEBUGGING;
       break;
     case scAbs:
       asym->section = &bfd_abs_section;
-      asym->flags = 0;
       break;
     case scUndefined:
       asym->section = &bfd_und_section;
       asym->flags = 0;
+      asym->value = 0;
       break;
     case scCdbLocal:
     case scBits:
@@ -550,8 +710,9 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
       asym->value -= asym->section->vma;
       break;
     case scSBss:
-      asym->section = &bfd_com_section;
-      asym->flags = 0;
+      asym->section = bfd_make_section_old_way (abfd, ".sbss");
+      if (! ext)
+       asym->value -= asym->section->vma;
       break;
     case scRData:
       asym->section = bfd_make_section_old_way (abfd, ".rdata");
@@ -561,8 +722,32 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
       asym->flags = BSF_DEBUGGING;
       break;
     case scCommon:
+      /* FIXME: We should take a -G argument, which gives the maximum
+        size of objects to be put in the small common section.  Until
+        we do, we put objects of sizes up to 8 in the small common
+        section.  The assembler should do this for us, but the native
+        assembler seems to get confused.  */
+      if (asym->value > 8)
+       {
+         asym->section = &bfd_com_section;
+         break;
+       }
+      /* Fall through.  */
     case scSCommon:
-      asym->section = &bfd_com_section;
+      if (ecoff_scom_section.name == NULL)
+       {
+         /* Initialize the small common section.  */
+         ecoff_scom_section.name = SCOMMON;
+         ecoff_scom_section.flags = SEC_IS_COMMON;
+         ecoff_scom_section.output_section = &ecoff_scom_section;
+         ecoff_scom_section.symbol = &ecoff_scom_symbol;
+         ecoff_scom_section.symbol_ptr_ptr = &ecoff_scom_symbol_ptr;
+         ecoff_scom_symbol.name = SCOMMON;
+         ecoff_scom_symbol.flags = BSF_SECTION_SYM;
+         ecoff_scom_symbol.section = &ecoff_scom_section;
+         ecoff_scom_symbol_ptr = &ecoff_scom_symbol;
+       }
+      asym->section = &ecoff_scom_section;
       asym->flags = 0;
       break;
     case scVarRegister:
@@ -572,6 +757,7 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
     case scSUndefined:
       asym->section = &bfd_und_section;
       asym->flags = 0;
+      asym->value = 0;
       break;
     case scInit:
       asym->section = bfd_make_section_old_way (abfd, ".init");
@@ -587,7 +773,6 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
       asym->value -= asym->section->vma;
       break;
     default:
-      asym->flags = 0;
       break;
     }
 }
@@ -595,8 +780,8 @@ DEFUN (ecoff_set_symbol_info, (abfd, ecoff_sym, asym, ext),
 /* Read an ECOFF symbol table.  */
 
 static boolean
-DEFUN (ecoff_slurp_symbol_table, (abfd),
-       bfd *abfd)
+ecoff_slurp_symbol_table (abfd)
+     bfd *abfd;
 {
   bfd_size_type internal_size;
   ecoff_symbol_type *internal;
@@ -674,8 +859,8 @@ DEFUN (ecoff_slurp_symbol_table, (abfd),
 }
 
 static unsigned int
-DEFUN (ecoff_get_symtab_upper_bound, (abfd),
-       bfd *abfd)
+ecoff_get_symtab_upper_bound (abfd)
+     bfd *abfd;
 {
   if (ecoff_slurp_symbolic_info (abfd) == false
       || bfd_get_symcount (abfd) == 0)
@@ -685,9 +870,9 @@ DEFUN (ecoff_get_symtab_upper_bound, (abfd),
 }
 
 static unsigned int
-DEFUN (ecoff_get_symtab, (abfd, alocation),
-       bfd *abfd AND
-       asymbol **alocation)
+ecoff_get_symtab (abfd, alocation)
+     bfd *abfd;
+     asymbol **alocation;
 {
   unsigned int counter = 0;
   ecoff_symbol_type *symbase;
@@ -708,18 +893,18 @@ DEFUN (ecoff_get_symtab, (abfd, alocation),
 }
 
 /* Turn ECOFF type information into a printable string.
-   emit_aggregate and type_to_string are from gcc/mips-tdump.c, with
-   swapping added and used_ptr removed.  */
+   ecoff_emit_aggregate and ecoff_type_to_string are from
+   gcc/mips-tdump.c, with swapping added and used_ptr removed.  */
 
 /* Write aggregate information to a string.  */
 
 static void
-DEFUN (emit_aggregate, (abfd, string, rndx, isym, which),
-       bfd *abfd AND
-       char *string AND
-       RNDXR *rndx AND
-       long isym AND
-       CONST char *which)
+ecoff_emit_aggregate (abfd, string, rndx, isym, which)
+     bfd *abfd;
+     char *string;
+     RNDXR *rndx;
+     long isym;
+     CONST char *which;
 {
   int ifd = rndx->rfd;
   int indx = rndx->index;
@@ -754,11 +939,11 @@ DEFUN (emit_aggregate, (abfd, string, rndx, isym, which),
 /* Convert the type information to string format.  */
 
 static char *
-DEFUN (type_to_string, (abfd, aux_ptr, indx, bigendian),
-       bfd *abfd AND
-       union aux_ext *aux_ptr AND
-       int indx AND
-       int bigendian)
+ecoff_type_to_string (abfd, aux_ptr, indx, bigendian)
+     bfd *abfd;
+     union aux_ext *aux_ptr;
+     int indx;
+     int bigendian;
 {
   AUXU u;
   struct qual {
@@ -855,9 +1040,9 @@ DEFUN (type_to_string, (abfd, aux_ptr, indx, bigendian),
 
     case btStruct:             /* Structure (Record) */
       ecoff_swap_rndx_in (bigendian, &aux_ptr[indx].a_rndx, &rndx);
-      emit_aggregate (abfd, p1, &rndx,
-                     AUX_GET_ISYM (bigendian, &aux_ptr[indx+1]),
-                     "struct");
+      ecoff_emit_aggregate (abfd, p1, &rndx,
+                           AUX_GET_ISYM (bigendian, &aux_ptr[indx+1]),
+                           "struct");
       indx++;                  /* skip aux words */
       break;
 
@@ -867,9 +1052,9 @@ DEFUN (type_to_string, (abfd, aux_ptr, indx, bigendian),
 
     case btUnion:              /* Union */
       ecoff_swap_rndx_in (bigendian, &aux_ptr[indx].a_rndx, &rndx);
-      emit_aggregate (abfd, p1, &rndx,
-                     AUX_GET_ISYM (bigendian, &aux_ptr[indx+1]),
-                     "union");
+      ecoff_emit_aggregate (abfd, p1, &rndx,
+                           AUX_GET_ISYM (bigendian, &aux_ptr[indx+1]),
+                           "union");
       indx++;                  /* skip aux words */
       break;
 
@@ -879,9 +1064,9 @@ DEFUN (type_to_string, (abfd, aux_ptr, indx, bigendian),
 
     case btEnum:               /* Enumeration */
       ecoff_swap_rndx_in (bigendian, &aux_ptr[indx].a_rndx, &rndx);
-      emit_aggregate (abfd, p1, &rndx,
-                     AUX_GET_ISYM (bigendian, &aux_ptr[indx+1]),
-                     "enum");
+      ecoff_emit_aggregate (abfd, p1, &rndx,
+                           AUX_GET_ISYM (bigendian, &aux_ptr[indx+1]),
+                           "enum");
       indx++;                  /* skip aux words */
       break;
 
@@ -1060,11 +1245,11 @@ DEFUN (type_to_string, (abfd, aux_ptr, indx, bigendian),
 /* Print information about an ECOFF symbol.  */
 
 static void
-DEFUN (ecoff_print_symbol, (abfd, filep, symbol, how),
-       bfd *abfd AND
-       PTR filep AND
-       asymbol *symbol AND
-       bfd_print_symbol_type how)
+ecoff_print_symbol (abfd, filep, symbol, how)
+     bfd *abfd;
+     PTR filep;
+     asymbol *symbol;
+     bfd_print_symbol_type how;
 {
   FILE *file = (FILE *)filep;
 
@@ -1207,8 +1392,8 @@ DEFUN (ecoff_print_symbol, (abfd, filep, symbol, how),
                          (AUX_GET_ISYM (bigendian,
                                         &aux_base[ecoff_ext.asym.index])
                           + sym_base),
-                         type_to_string (abfd, aux_base, indx + 1,
-                                         bigendian));
+                         ecoff_type_to_string (abfd, aux_base, indx + 1,
+                                               bigendian));
                else
                  printf ("\n      Local symbol: %d",
                          (indx
@@ -1219,7 +1404,8 @@ DEFUN (ecoff_print_symbol, (abfd, filep, symbol, how),
              default:
                if (!MIPS_IS_STAB (&ecoff_ext.asym))
                  printf ("\n      Type: %s",
-                         type_to_string (abfd, aux_base, indx, bigendian));
+                         ecoff_type_to_string (abfd, aux_base, indx,
+                                               bigendian));
                break;
              }
          }
@@ -1235,19 +1421,19 @@ DEFUN (ecoff_print_symbol, (abfd, filep, symbol, how),
 /* Swap a reloc in.  */
 
 static void
-DEFUN (ecoff_swap_reloc_in, (abfd, ext, intern),
-       bfd *abfd AND
-       RELOC *ext AND
-       struct internal_reloc *intern)
+ecoff_swap_reloc_in (abfd, ext, intern)
+     bfd *abfd;
+     RELOC *ext;
+     struct internal_reloc *intern;
 {
   intern->r_vaddr = bfd_h_get_32 (abfd, (bfd_byte *) ext->r_vaddr);
   if (abfd->xvec->header_byteorder_big_p != false)
     {
-      intern->r_symndx = ((ext->r_bits[0]
+      intern->r_symndx = (((int) ext->r_bits[0]
                           << RELOC_BITS0_SYMNDX_SH_LEFT_BIG)
-                         | (ext->r_bits[1]
+                         | ((int) ext->r_bits[1]
                             << RELOC_BITS1_SYMNDX_SH_LEFT_BIG)
-                         | (ext->r_bits[2]
+                         | ((int) ext->r_bits[2]
                             << RELOC_BITS2_SYMNDX_SH_LEFT_BIG));
       intern->r_type = ((ext->r_bits[3] & RELOC_BITS3_TYPE_BIG)
                        >> RELOC_BITS3_TYPE_SH_BIG);
@@ -1255,11 +1441,11 @@ DEFUN (ecoff_swap_reloc_in, (abfd, ext, intern),
     }
   else
     {
-      intern->r_symndx = ((ext->r_bits[0]
+      intern->r_symndx = (((int) ext->r_bits[0]
                           << RELOC_BITS0_SYMNDX_SH_LEFT_LITTLE)
-                         | (ext->r_bits[1]
+                         | ((int) ext->r_bits[1]
                             << RELOC_BITS1_SYMNDX_SH_LEFT_LITTLE)
-                         | (ext->r_bits[2]
+                         | ((int) ext->r_bits[2]
                             << RELOC_BITS2_SYMNDX_SH_LEFT_LITTLE));
       intern->r_type = ((ext->r_bits[3] & RELOC_BITS3_TYPE_LITTLE)
                        >> RELOC_BITS3_TYPE_SH_LITTLE);
@@ -1270,10 +1456,10 @@ DEFUN (ecoff_swap_reloc_in, (abfd, ext, intern),
 /* Swap a reloc out.  */
 
 static unsigned int
-DEFUN (ecoff_swap_reloc_out, (abfd, src, dst),
-       bfd *abfd AND
-       PTR src AND
-       PTR dst)
+ecoff_swap_reloc_out (abfd, src, dst)
+     bfd *abfd;
+     PTR src;
+     PTR dst;
 {
   struct internal_reloc *intern = (struct internal_reloc *) src;
   RELOC *ext = (RELOC *) dst;
@@ -1301,30 +1487,73 @@ DEFUN (ecoff_swap_reloc_out, (abfd, src, dst),
   return RELSZ;
 }
 
+/* ECOFF relocs are either against external symbols, or against
+   sections.  If we are producing relocateable output, and the reloc
+   is against an external symbol, the resulting reloc will also be
+   against the same symbol.  In such a case, we don't want to change
+   anything about the way the reloc is handled, since it will all be
+   done at final link time.  Rather than put special case code into
+   bfd_perform_relocation, all the reloc types use this howto
+   function.  It just short circuits the reloc if producing
+   relocateable output against an external symbol.  */
+
+static bfd_reloc_status_type
+ecoff_generic_reloc (abfd,
+                    reloc_entry,
+                    symbol,
+                    data,
+                    input_section,
+                    output_bfd)
+     bfd *abfd;
+     arelent *reloc_entry;
+     asymbol *symbol;
+     PTR data;
+     asection *input_section;
+     bfd *output_bfd;
+{
+  if (output_bfd != (bfd *) NULL
+      && (symbol->flags & BSF_SECTION_SYM) == 0)
+    {
+      reloc_entry->address += input_section->output_offset;
+      return bfd_reloc_ok;
+    }
+
+  return bfd_reloc_continue;
+}
+
 /* Do a REFHI relocation.  The next reloc must be the corresponding
    REFLO.  This has to be done in a function so that carry is handled
    correctly.  */
 
 static bfd_reloc_status_type
-DEFUN (ecoff_refhi_reloc, (abfd,
-                          reloc_entry,
-                          symbol,
-                          data,
-                          input_section,
-                          output_bfd),
-       bfd *abfd AND
-       arelent *reloc_entry AND
-       asymbol *symbol AND
-       PTR data AND
-       asection *input_section AND
-       bfd *output_bfd)
+ecoff_refhi_reloc (abfd,
+                  reloc_entry,
+                  symbol,
+                  data,
+                  input_section,
+                  output_bfd)
+     bfd *abfd;
+     arelent *reloc_entry;
+     asymbol *symbol;
+     PTR data;
+     asection *input_section;
+     bfd *output_bfd;
 {
   bfd_reloc_status_type ret;
   arelent *rello;
   bfd_vma relocation;
-  asection *reloc_target_output_section;
-  unsigned long val;
   unsigned long insn;
+  unsigned long val;
+  unsigned long vallo;
+
+  /* If we're relocating, and this an external symbol, we don't want
+     to change anything.  */
+  if (output_bfd != (bfd *) NULL
+      && (symbol->flags & BSF_SECTION_SYM) == 0)
+    {
+      reloc_entry->address += input_section->output_offset;
+      return bfd_reloc_ok;
+    }
 
   ret = bfd_reloc_ok;
   if (symbol->section == &bfd_und_section
@@ -1335,26 +1564,39 @@ DEFUN (ecoff_refhi_reloc, (abfd,
   BFD_ASSERT (rello->howto->type == ECOFF_R_REFLO
              && *rello->sym_ptr_ptr == *reloc_entry->sym_ptr_ptr);
 
-  if (symbol->section == &bfd_com_section)
+  if (bfd_is_com_section (symbol->section))
     relocation = 0;
   else
     relocation = symbol->value;
 
   relocation += symbol->section->output_section->vma;
   relocation += symbol->section->output_offset;
+  relocation += reloc_entry->addend;
 
   if (reloc_entry->address > input_section->_cooked_size)
     return bfd_reloc_outofrange;
 
-  if (output_bfd != (bfd *) NULL) 
-    reloc_entry->address += input_section->output_offset;
-
   insn = bfd_get_32 (abfd, (bfd_byte *) data + reloc_entry->address);
-  val = (((insn & 0xffff) << 16)
-        + (bfd_get_32 (abfd, (bfd_byte *) data + rello->address) & 0xffff));
-  insn = (insn &~ 0xffff) | (((val + symbol->value) >> 16) & 0xffff);
+  vallo = bfd_get_32 (abfd, (bfd_byte *) data + rello->address) & 0xffff;
+  val = ((insn & 0xffff) << 16) + vallo;
+  val += relocation;
+
+  /* The low order 16 bits are always treated as a signed value.
+     Therefore, a negative value in the low order bits requires an
+     adjustment in the high order bits.  We need to make this
+     adjustment in two ways: once for the bits we took from the data,
+     and once for the bits we are putting back in to the data.  */
+  if ((vallo & 0x8000) != 0)
+    val -= 0x10000;
+  if ((val & 0x8000) != 0)
+    val += 0x10000;
+
+  insn = (insn &~ 0xffff) | ((val >> 16) & 0xffff);
   bfd_put_32 (abfd, insn, (bfd_byte *) data + reloc_entry->address);
 
+  if (output_bfd != (bfd *) NULL) 
+    reloc_entry->address += input_section->output_offset;
+
   return ret;
 }
 
@@ -1362,71 +1604,91 @@ DEFUN (ecoff_refhi_reloc, (abfd,
    the offset from the gp register.  */
 
 static bfd_reloc_status_type
-DEFUN (ecoff_gprel_reloc, (abfd,
-                          reloc_entry,
-                          symbol,
-                          data,
-                          input_section,
-                          output_bfd),
-       bfd *abfd AND
-       arelent *reloc_entry AND
-       asymbol *symbol AND
-       PTR data AND
-       asection *input_section AND
-       bfd *output_bfd)
+ecoff_gprel_reloc (abfd,
+                  reloc_entry,
+                  symbol,
+                  data,
+                  input_section,
+                  output_bfd)
+     bfd *abfd;
+     arelent *reloc_entry;
+     asymbol *symbol;
+     PTR data;
+     asection *input_section;
+     bfd *output_bfd;
 {
-  bfd_reloc_status_type ret;
+  boolean relocateable;
   bfd_vma relocation;
   unsigned long val;
   unsigned long insn;
 
-  /* If this is a partial link, we don't need to do anything unusual
-     here.  */
+  /* If we're relocating, and this an external symbol, we don't want
+     to change anything.  */
+  if (output_bfd != (bfd *) NULL
+      && (symbol->flags & BSF_SECTION_SYM) == 0)
+    {
+      reloc_entry->address += input_section->output_offset;
+      return bfd_reloc_ok;
+    }
+
   if (output_bfd != (bfd *) NULL)
-    return bfd_reloc_continue;
+    relocateable = true;
+  else
+    {
+      relocateable = false;
+      output_bfd = symbol->section->output_section->owner;
+    }
 
-  ret = bfd_reloc_ok;
-  if (symbol->section == &bfd_und_section)
-    ret = bfd_reloc_undefined;
+  if (symbol->section == &bfd_und_section
+      && relocateable == false)
+    return bfd_reloc_undefined;
 
-  /* Otherwise we have to figure out the gp value, so that we can
-     adjust the symbol value correctly.  We look up the symbol _gp in
-     the output BFD.  If we can't find it, we're stuck.  We cache it
-     in the ECOFF target data.  */
-  output_bfd = symbol->section->output_section->owner;
+  /* We have to figure out the gp value, so that we can adjust the
+     symbol value correctly.  We look up the symbol _gp in the output
+     BFD.  If we can't find it, we're stuck.  We cache it in the ECOFF
+     target data.  */
   if (ecoff_data (output_bfd)->gp == 0)
     {
-      unsigned int count;
-      asymbol **sym;
-      unsigned int i;
+      if (relocateable != false)
+       {
+         /* Make up a value.  */
+         ecoff_data (output_bfd)->gp =
+           symbol->section->output_section->vma + 0x4000;
+       }
+      else
+       {
+         unsigned int count;
+         asymbol **sym;
+         unsigned int i;
 
-      count = bfd_get_symcount (output_bfd);
-      sym = bfd_get_outsymbols (output_bfd);
+         count = bfd_get_symcount (output_bfd);
+         sym = bfd_get_outsymbols (output_bfd);
 
-      /* We should do something more friendly here, but we don't have
-        a good reloc status to return.  */
-      if (sym == (asymbol **) NULL)
-       abort ();
-
-      for (i = 0; i < count; i++, sym++)
-       {
-         register CONST char *name;
+         /* We should do something more friendly here, but we don't
+            have a good reloc status to return.  */
+         if (sym == (asymbol **) NULL)
+           abort ();
 
-         name = bfd_asymbol_name (*sym);
-         if (*name == '_' && strcmp (name, "_gp") == 0)
+         for (i = 0; i < count; i++, sym++)
            {
-             ecoff_data (output_bfd)->gp = bfd_asymbol_value (*sym);
-             break;
+             register CONST char *name;
+
+             name = bfd_asymbol_name (*sym);
+             if (*name == '_' && strcmp (name, "_gp") == 0)
+               {
+                 ecoff_data (output_bfd)->gp = bfd_asymbol_value (*sym);
+                 break;
+               }
            }
-       }
 
-      /* We should do something more friendly here, but we don't have
-        a good reloc status to return.  */
-      if (i >= count)
-       abort ();
+         /* We should do something more friendly here, but we don't have
+            a good reloc status to return.  */
+         if (i >= count)
+           abort ();
+       }
     }
 
-  if (symbol->section == &bfd_com_section)
+  if (bfd_is_com_section (symbol->section))
     relocation = 0;
   else
     relocation = symbol->value;
@@ -1438,14 +1700,28 @@ DEFUN (ecoff_gprel_reloc, (abfd,
     return bfd_reloc_outofrange;
 
   insn = bfd_get_32 (abfd, (bfd_byte *) data + reloc_entry->address);
-  val = insn & 0xffff;
-  if ((val & 0x8000) != 0)
+
+  /* Set val to the offset into the section (if we are relocating
+     against an external symbol, insn & 0xffff will be zero and so
+     will reloc_entry->addend).  */
+  val = ((insn & 0xffff) + reloc_entry->addend) & 0xffff;
+  if (val & 0x8000)
     val -= 0x10000;
-  val -= ecoff_data (output_bfd)->gp;
+
+  /* Adjust val for the final section location and GP value.  */
+  val += (relocation - ecoff_data (output_bfd)->gp);
+
   insn = (insn &~ 0xffff) | (val & 0xffff);
   bfd_put_32 (abfd, insn, (bfd_byte *) data + reloc_entry->address);
 
-  return ret;
+  if (relocateable != false)
+    reloc_entry->address += input_section->output_offset;
+
+  /* Make sure it fit in 16 bits.  */
+  if (val >= 0x8000 && val < 0xffff8000)
+    return bfd_reloc_outofrange;
+
+  return bfd_reloc_ok;
 }
 
 /* How to process the various relocs types.  */
@@ -1479,7 +1755,7 @@ static reloc_howto_type ecoff_howto_table[] =
         0,                     /* bitpos */
         false,                 /* absolute (obsolete) */
         true,                  /* complain_on_overflow */
-        0,                     /* special_function */
+        ecoff_generic_reloc,   /* special_function */
         "REFHALF",             /* name */
         true,                  /* partial_inplace */
         0xffff,                /* src_mask */
@@ -1495,7 +1771,7 @@ static reloc_howto_type ecoff_howto_table[] =
         0,                     /* bitpos */
         false,                 /* absolute (obsolete) */
         true,                  /* complain_on_overflow */
-        0,                     /* special_function */
+        ecoff_generic_reloc,   /* special_function */
         "REFWORD",             /* name */
         true,                  /* partial_inplace */
         0xffffffff,            /* src_mask */
@@ -1511,7 +1787,7 @@ static reloc_howto_type ecoff_howto_table[] =
         0,                     /* bitpos */
         false,                 /* absolute (obsolete) */
         true,                  /* complain_on_overflow */
-        0,                     /* special_function */
+        ecoff_generic_reloc,   /* special_function */
         "JMPADDR",             /* name */
         true,                  /* partial_inplace */
         0x3ffffff,             /* src_mask */
@@ -1544,7 +1820,7 @@ static reloc_howto_type ecoff_howto_table[] =
         0,                     /* bitpos */
         false,                 /* absolute (obsolete) */
         true,                  /* complain_on_overflow */
-        0,                     /* special_function */
+        ecoff_generic_reloc,   /* special_function */
         "REFLO",               /* name */
         true,                  /* partial_inplace */
         0xffff,                /* src_mask */
@@ -1589,10 +1865,10 @@ static reloc_howto_type ecoff_howto_table[] =
 /* Read in the relocs for a section.  */
 
 static boolean
-DEFUN (ecoff_slurp_reloc_table, (abfd, section, symbols),
-       bfd *abfd AND
-       asection *section AND
-       asymbol **symbols)
+ecoff_slurp_reloc_table (abfd, section, symbols)
+     bfd *abfd;
+     asection *section;
+     asymbol **symbols;
 {
   RELOC *external_relocs;
   arelent *internal_relocs;
@@ -1632,7 +1908,7 @@ DEFUN (ecoff_slurp_reloc_table, (abfd, section, symbols),
 
       ecoff_swap_reloc_in (abfd, external_relocs + i, &intern);
 
-      if (intern.r_type < 0 || intern.r_type > ECOFF_R_LITERAL)
+      if (intern.r_type > ECOFF_R_LITERAL)
        abort ();
 
       if (intern.r_extern)
@@ -1642,6 +1918,7 @@ DEFUN (ecoff_slurp_reloc_table, (abfd, section, symbols),
                      && (intern.r_symndx
                          < ecoff_data (abfd)->symbolic_header.iextMax));
          rptr->sym_ptr_ptr = symbols + intern.r_symndx;
+         rptr->addend = 0;
        }
       else
        {
@@ -1667,10 +1944,14 @@ DEFUN (ecoff_slurp_reloc_table, (abfd, section, symbols),
          if (sec == (asection *) NULL)
            abort ();
          rptr->sym_ptr_ptr = sec->symbol_ptr_ptr;
+
+         rptr->addend = - bfd_get_section_vma (abfd, sec);
+         if (intern.r_type == ECOFF_R_GPREL
+             || intern.r_type == ECOFF_R_LITERAL)
+           rptr->addend += ecoff_data (abfd)->gp;
        }
 
       rptr->address = intern.r_vaddr - bfd_get_section_vma (abfd, section);
-      rptr->addend = 0;
       rptr->howto = &ecoff_howto_table[intern.r_type];
 
       /* If the type is ECOFF_R_IGNORE, make sure this is a reference
@@ -1689,11 +1970,11 @@ DEFUN (ecoff_slurp_reloc_table, (abfd, section, symbols),
 /* Get a canonical list of relocs.  */
 
 static unsigned int
-DEFUN (ecoff_canonicalize_reloc, (abfd, section, relptr, symbols),
-       bfd *abfd AND
-       asection *section AND
-       arelent **relptr AND
-       asymbol **symbols)
+ecoff_canonicalize_reloc (abfd, section, relptr, symbols)
+     bfd *abfd;
+     asection *section;
+     arelent **relptr;
+     asymbol **symbols;
 {
   unsigned int count;
 
@@ -1709,7 +1990,7 @@ DEFUN (ecoff_canonicalize_reloc, (abfd, section, relptr, symbols),
           count++, chain = chain->next)
        *relptr++ = &chain->relent;
     }
-  else 
+  else
     { 
       arelent *tblptr;
 
@@ -1734,20 +2015,20 @@ DEFUN (ecoff_canonicalize_reloc, (abfd, section, relptr, symbols),
    wanted location.  */
 
 static boolean
-DEFUN (ecoff_find_nearest_line, (abfd,
-                                section,
-                                ignore_symbols,
-                                offset,
-                                filename_ptr,
-                                functionname_ptr,
-                                retline_ptr),
-       bfd *abfd AND
-       asection *section AND
-       asymbol **ignore_symbols AND
-       bfd_vma offset AND
-       CONST char **filename_ptr AND
-       CONST char **functionname_ptr AND
-       unsigned int *retline_ptr)
+ecoff_find_nearest_line (abfd,
+                        section,
+                        ignore_symbols,
+                        offset,
+                        filename_ptr,
+                        functionname_ptr,
+                        retline_ptr)
+     bfd *abfd;
+     asection *section;
+     asymbol **ignore_symbols;
+     bfd_vma offset;
+     CONST char **filename_ptr;
+     CONST char **functionname_ptr;
+     unsigned int *retline_ptr;
 {
   FDR *fdr_ptr;
   FDR *fdr_start;
@@ -1759,11 +2040,12 @@ DEFUN (ecoff_find_nearest_line, (abfd,
   unsigned char *line_ptr;
   unsigned char *line_end;
   int lineno;
-  SYMR proc_sym;
 
   /* If we're not in the .text section, we don't have any line
      numbers.  */
-  if (strcmp (section->name, _TEXT) != 0)
+  if (strcmp (section->name, _TEXT) != 0
+      || offset < ecoff_data (abfd)->text_start
+      || offset >= ecoff_data (abfd)->text_end)
     return false;
 
   /* Make sure we have the FDR's.  */
@@ -1781,10 +2063,11 @@ DEFUN (ecoff_find_nearest_line, (abfd,
   fdr_hold = (FDR *) NULL;
   for (fdr_ptr = fdr_start; fdr_ptr < fdr_end; fdr_ptr++)
     {
+      if (fdr_ptr->cpd == 0)
+       continue;
       if (offset < fdr_ptr->adr)
        break;
-      if (fdr_ptr->cpd > 0)
-       fdr_hold = fdr_ptr;
+      fdr_hold = fdr_ptr;
     }
   if (fdr_hold == (FDR *) NULL)
     return false;
@@ -1848,17 +2131,38 @@ DEFUN (ecoff_find_nearest_line, (abfd,
       offset -= count * 4;
     }
 
-  /* If offset is too large, this line is not interesting.  */
-  if (offset > 100)
-    return false;
+  /* If fdr_ptr->rss is -1, then this file does not have full symbols,
+     at least according to gdb/mipsread.c.  */
+  if (fdr_ptr->rss == -1)
+    {
+      *filename_ptr = NULL;
+      if (pdr.isym == -1)
+       *functionname_ptr = NULL;
+      else
+       {
+         EXTR proc_ext;
 
-  *filename_ptr = ecoff_data (abfd)->ss + fdr_ptr->issBase + fdr_ptr->rss;
-  ecoff_swap_sym_in (abfd,
-                    (ecoff_data (abfd)->external_sym
-                     + fdr_ptr->isymBase
-                     + pdr.isym),
-                    &proc_sym);
-  *functionname_ptr = ecoff_data (abfd)->ss + proc_sym.iss;
+         ecoff_swap_ext_in (abfd,
+                            (ecoff_data (abfd)->external_ext
+                             + pdr.isym),
+                            &proc_ext);
+         *functionname_ptr = ecoff_data (abfd)->ssext + proc_ext.asym.iss;
+       }
+    }
+  else
+    {
+      SYMR proc_sym;
+
+      *filename_ptr = ecoff_data (abfd)->ss + fdr_ptr->issBase + fdr_ptr->rss;
+      ecoff_swap_sym_in (abfd,
+                        (ecoff_data (abfd)->external_sym
+                         + fdr_ptr->isymBase
+                         + pdr.isym),
+                        &proc_sym);
+      *functionname_ptr = (ecoff_data (abfd)->ss
+                          + fdr_ptr->issBase
+                          + proc_sym.iss);
+    }
   *retline_ptr = lineno;
   return true;
 }
@@ -1884,8 +2188,8 @@ DEFUN (ecoff_find_nearest_line, (abfd,
    once.  */
 
 static void
-DEFUN (ecoff_clear_output_flags, (abfd),
-       bfd *abfd)
+ecoff_clear_output_flags (abfd)
+     bfd *abfd;
 {
   register asection *o;
   register bfd_seclet_type *p;
@@ -1903,12 +2207,12 @@ DEFUN (ecoff_clear_output_flags, (abfd),
    any.  */
 
 static boolean
-DEFUN (ecoff_rel, (output_bfd, seclet, output_section, data, relocateable),
-       bfd *output_bfd AND
-       bfd_seclet_type *seclet AND
-       asection *output_section AND
-       PTR data AND
-       boolean relocateable)
+ecoff_rel (output_bfd, seclet, output_section, data, relocateable)
+     bfd *output_bfd;
+     bfd_seclet_type *seclet;
+     asection *output_section;
+     PTR data;
+     boolean relocateable;
 {
   bfd *input_bfd;
   HDRR *output_symhdr;
@@ -1955,7 +2259,8 @@ DEFUN (ecoff_rel, (output_bfd, seclet, output_section, data, relocateable),
       asymbol **sym_ptr;
       asymbol **sym_end;
 
-      /* We just accumulate symbols from a non-ECOFF BFD.  */
+      /* We just accumulate local symbols from a non-ECOFF BFD.  The
+        external symbols are handled separately.  */
 
       symbols = (asymbol **) bfd_alloc (output_bfd,
                                        get_symtab_upper_bound (input_bfd));
@@ -1976,11 +2281,6 @@ DEFUN (ecoff_rel, (output_bfd, seclet, output_section, data, relocateable),
              ++output_symhdr->isymMax;
              output_symhdr->issMax += len + 1;
            }
-         else
-           {
-             ++output_symhdr->iextMax;
-             output_symhdr->issExtMax += len + 1;
-           }
        }
 
       bfd_release (output_bfd, (PTR) symbols);
@@ -1992,13 +2292,15 @@ DEFUN (ecoff_rel, (output_bfd, seclet, output_section, data, relocateable),
 
   /* We simply add in the information from another ECOFF BFD.  First
      we make sure we have the symbolic information.  */
-  if (ecoff_slurp_symbolic_info (input_bfd) == false)
+  if (ecoff_slurp_symbol_table (input_bfd) == false)
     return false;
   if (bfd_get_symcount (input_bfd) == 0)
     return true;
 
   input_symhdr = &ecoff_data (input_bfd)->symbolic_header;
 
+  /* Figure out how much information we are going to be putting in.
+     The external symbols are handled separately.  */
   output_symhdr->ilineMax += input_symhdr->ilineMax;
   output_symhdr->cbLine += input_symhdr->cbLine;
   output_symhdr->idnMax += input_symhdr->idnMax;
@@ -2007,9 +2309,7 @@ DEFUN (ecoff_rel, (output_bfd, seclet, output_section, data, relocateable),
   output_symhdr->ioptMax += input_symhdr->ioptMax;
   output_symhdr->iauxMax += input_symhdr->iauxMax;
   output_symhdr->issMax += input_symhdr->issMax;
-  output_symhdr->issExtMax += input_symhdr->issExtMax;
   output_symhdr->ifdMax += input_symhdr->ifdMax;
-  output_symhdr->iextMax += input_symhdr->iextMax;
 
   /* The RFD's are special, since we create them if needed.  */
   if (input_symhdr->crfd > 0)
@@ -2023,12 +2323,12 @@ DEFUN (ecoff_rel, (output_bfd, seclet, output_section, data, relocateable),
 /* Handle an arbitrary seclet on the first pass.  */
 
 static boolean
-DEFUN (ecoff_dump_seclet, (abfd, seclet, section, data, relocateable),
-       bfd *abfd AND
-       bfd_seclet_type *seclet AND
-       asection *section AND
-       PTR data AND
-       boolean relocateable)
+ecoff_dump_seclet (abfd, seclet, section, data, relocateable)
+     bfd *abfd;
+     bfd_seclet_type *seclet;
+     asection *section;
+     PTR data;
+     boolean relocateable;
 {
   switch (seclet->type) 
     {
@@ -2038,21 +2338,29 @@ DEFUN (ecoff_dump_seclet, (abfd, seclet, section, data, relocateable),
       return ecoff_rel (abfd, seclet, section, data, relocateable);
 
     case bfd_fill_seclet:
-      /* Fill in the section with us.  */
-      {
-       char *d = (char *) bfd_alloc (abfd, seclet->size);
-       unsigned int i;
-       boolean ret;
-
-       for (i = 0; i < seclet->size; i+=2)
-         d[i] = seclet->u.fill.value >> 8;
-       for (i = 1; i < seclet->size; i+=2)
-         d[i] = seclet->u.fill.value;
-       ret = bfd_set_section_contents (abfd, section, d, seclet->offset,
-                                       seclet->size);
-       bfd_release (abfd, (PTR) d);
-       return ret;
-      }
+      /* Fill in the section with fill.value.  This is used to pad out
+        sections, but we must avoid padding the .bss section.  */
+      if ((section->flags & SEC_HAS_CONTENTS) == 0)
+       {
+         if (seclet->u.fill.value != 0)
+           abort ();
+       }
+      else
+       {
+         char *d = (char *) bfd_alloc (abfd, seclet->size);
+         unsigned int i;
+         boolean ret;
+
+         for (i = 0; i < seclet->size; i+=2)
+           d[i] = seclet->u.fill.value >> 8;
+         for (i = 1; i < seclet->size; i+=2)
+           d[i] = seclet->u.fill.value;
+         ret = bfd_set_section_contents (abfd, section, d, seclet->offset,
+                                         seclet->size);
+         bfd_release (abfd, (PTR) d);
+         return ret;
+       }
+      break;
 
     default:
       abort();
@@ -2066,11 +2374,11 @@ DEFUN (ecoff_dump_seclet, (abfd, seclet, section, data, relocateable),
    external string base.  */
 
 static long
-DEFUN (ecoff_add_string, (output_bfd, fdr, string, external),
-       bfd *output_bfd AND
-       FDR *fdr AND
-       CONST char *string AND
-       boolean external)
+ecoff_add_string (output_bfd, fdr, string, external)
+     bfd *output_bfd;
+     FDR *fdr;
+     CONST char *string;
+     boolean external;
 {
   HDRR *symhdr;
   size_t len;
@@ -2097,30 +2405,25 @@ DEFUN (ecoff_add_string, (output_bfd, fdr, string, external),
 /* Accumulate the debugging information from an input section.  */
 
 static boolean
-DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
-       bfd *output_bfd AND
-       bfd_seclet_type *seclet AND
-       asection *section)
+ecoff_get_debug (output_bfd, seclet, section, relocateable)
+     bfd *output_bfd;
+     bfd_seclet_type *seclet;
+     asection *section;
+     boolean relocateable;
 {
   bfd *input_bfd;
   HDRR *output_symhdr;
   HDRR *input_symhdr;
   ecoff_data_type *output_ecoff;
   ecoff_data_type *input_ecoff;
+  unsigned int count;
+  struct sym_ext *sym_out;
+  ecoff_symbol_type *esym_ptr;
+  ecoff_symbol_type *esym_end;
+  unsigned long pdr_off;
   FDR *fdr_ptr;
   FDR *fdr_end;
   struct fdr_ext *fdr_out;
-  struct ext_ext *ext_ptr;
-  struct ext_ext *ext_end;
-  struct ext_ext *ext_out;
-  long iss;
-  long isym;
-  long iline;
-  long iopt;
-  long ipdr;
-  long iaux;
-  long irfd;
-  long cbline;
 
   input_bfd = seclet->u.indirect.section->owner;
 
@@ -2152,7 +2455,7 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
                                  false);
       fdr.isymBase = output_symhdr->isymMax;
 
-      /* Get the symbols from the input BFD.  */
+      /* Get the local symbols from the input BFD.  */
       symbols = (asymbol **) bfd_alloc (output_bfd,
                                        get_symtab_upper_bound (input_bfd));
       if (symbols == (asymbol **) NULL)
@@ -2162,7 +2465,8 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
        }
       sym_end = symbols + bfd_canonicalize_symtab (input_bfd, symbols);
 
-      /* Handle the local symbols.  */
+      /* Handle the local symbols.  Any external symbols are handled
+        separately.  */
       fdr.csym = 0;
       for (sym_ptr = symbols; sym_ptr != sym_end; sym_ptr++)
        {
@@ -2175,7 +2479,14 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
                                               &fdr,
                                               (*sym_ptr)->name,
                                               false);
-         internal_sym.value = (*sym_ptr)->value;
+
+         if (bfd_is_com_section ((*sym_ptr)->section)
+             || (*sym_ptr)->section == &bfd_und_section)
+           internal_sym.value = (*sym_ptr)->value;
+         else
+           internal_sym.value = ((*sym_ptr)->value
+                                 + (*sym_ptr)->section->output_offset
+                                 + (*sym_ptr)->section->output_section->vma);
          internal_sym.st = stNil;
          internal_sym.sc = scUndefined;
          internal_sym.index = indexNil;
@@ -2186,36 +2497,12 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
          ++output_symhdr->isymMax;
        }
 
-      /* Handle the external symbols.  */
-      for (sym_ptr = symbols; sym_ptr != sym_end; sym_ptr++)
-       {
-         EXTR internal_ext;
-
-         if (((*sym_ptr)->flags & BSF_EXPORT) == 0)
-           continue;
-         memset (&internal_ext, 0, sizeof internal_ext);
-         internal_ext.ifd = output_symhdr->ifdMax;
-         internal_ext.asym.iss = ecoff_add_string (output_bfd,
-                                                   &fdr,
-                                                   (*sym_ptr)->name,
-                                                   true);
-         internal_ext.asym.value = (*sym_ptr)->value;
-         internal_ext.asym.st = stNil;
-         internal_ext.asym.sc = scUndefined;
-         internal_ext.asym.reserved = 0;
-         internal_ext.asym.index = indexNil;
-         ecoff_swap_ext_out (output_bfd, &internal_ext,
-                             (output_ecoff->external_ext
-                              + output_symhdr->iextMax));
-         ++output_symhdr->iextMax;
-       }
-
       bfd_release (output_bfd, (PTR) symbols);
 
-      /* Leave everything else zeroed out.  This will cause the lang
-        field to be langC.  The fBigendian field will indicate little
-        endian format, but it doesn't matter because it only applies
-        to aux fields and there are none.  */
+      /* Leave everything else in the FDR zeroed out.  This will cause
+        the lang field to be langC.  The fBigendian field will
+        indicate little endian format, but it doesn't matter because
+        it only applies to aux fields and there are none.  */
 
       ecoff_swap_fdr_out (output_bfd, &fdr,
                          (output_ecoff->external_fdr
@@ -2226,15 +2513,59 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
 
   /* This is an ECOFF BFD.  We want to grab the information from
      input_bfd and attach it to output_bfd.  */
-  if (bfd_get_symcount (input_bfd) == 0)
+  count = bfd_get_symcount (input_bfd);
+  if (count == 0)
     return true;
   input_ecoff = ecoff_data (input_bfd);
   input_symhdr = &input_ecoff->symbolic_header;
 
-  /* Because ECOFF uses relative pointers for most of the debugging
-     information, much of it can simply be copied from input_bfd to
-     output_bfd.  */
-  memcpy (output_ecoff->line + output_symhdr->cbLineOffset,
+  /* I think that it is more efficient to simply copy the debugging
+     information from the input BFD to the output BFD.  Because ECOFF
+     uses relative pointers for most of the debugging information,
+     only a little of it has to be changed at all.  */
+
+  /* Swap in the local symbols, adjust their values, and swap them out
+     again.  The external symbols are handled separately.  */
+  sym_out = output_ecoff->external_sym + output_symhdr->isymMax;
+
+  esym_ptr = ecoff_data (input_bfd)->canonical_symbols;
+  esym_end = esym_ptr + count;
+  for (; esym_ptr < esym_end; esym_ptr++)
+    {
+      if (esym_ptr->local)
+       {
+         SYMR sym;
+
+         ecoff_swap_sym_in (input_bfd, esym_ptr->native.lnative, &sym);
+
+         /* If we're producing an executable, move common symbols
+            into bss.  */
+         if (relocateable == false)
+           {
+             if (sym.sc == scCommon)
+               sym.sc = scBss;
+             else if (sym.sc == scSCommon)
+               sym.sc = scSBss;
+           }
+
+         if (! bfd_is_com_section (esym_ptr->symbol.section)
+             && (esym_ptr->symbol.flags & BSF_DEBUGGING) == 0
+             && esym_ptr->symbol.section != &bfd_und_section)
+           sym.value = (esym_ptr->symbol.value
+                        + esym_ptr->symbol.section->output_offset
+                        + esym_ptr->symbol.section->output_section->vma);
+         ecoff_swap_sym_out (output_bfd, &sym, sym_out);
+         ++sym_out;
+       }
+    }
+
+  /* That should have accounted for all the local symbols in
+     input_bfd.  */
+  BFD_ASSERT ((sym_out - output_ecoff->external_sym) - output_symhdr->isymMax
+             == input_symhdr->isymMax);
+
+  /* Copy the information that does not need swapping.  */
+  memcpy (output_ecoff->line + output_symhdr->cbLine,
          input_ecoff->line,
          input_symhdr->cbLine * sizeof (unsigned char));
   memcpy (output_ecoff->external_aux + output_symhdr->iauxMax,
@@ -2243,12 +2574,10 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
   memcpy (output_ecoff->ss + output_symhdr->issMax,
          input_ecoff->ss,
          input_symhdr->issMax * sizeof (char));
-  memcpy (output_ecoff->ssext + output_symhdr->issExtMax,
-         input_ecoff->ssext,
-         input_symhdr->issExtMax * sizeof (char));
 
   /* Some of the information may need to be swapped.  */
-  if (output_bfd->xvec->byteorder_big_p == input_bfd->xvec->byteorder_big_p)
+  if (output_bfd->xvec->header_byteorder_big_p
+      == input_bfd->xvec->header_byteorder_big_p)
     {
       /* The two BFD's have the same endianness, so memcpy will
         suffice.  */
@@ -2258,9 +2587,15 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
       memcpy (output_ecoff->external_pdr + output_symhdr->ipdMax,
              input_ecoff->external_pdr,
              input_symhdr->ipdMax * sizeof (struct pdr_ext));
-      memcpy (output_ecoff->external_sym + output_symhdr->isymMax,
-             input_ecoff->external_sym,
-             input_symhdr->isymMax * sizeof (struct sym_ext));
+      if (input_symhdr->ipdMax == 0)
+       pdr_off = 0;
+      else
+       {
+         PDR pdr;
+
+         ecoff_swap_pdr_in (input_bfd, input_ecoff->external_pdr, &pdr);
+         pdr_off = pdr.adr;
+       }
       memcpy (output_ecoff->external_opt + output_symhdr->ioptMax,
              input_ecoff->external_opt,
              input_symhdr->ioptMax * sizeof (struct opt_ext));
@@ -2273,16 +2608,14 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
       struct pdr_ext *pdr_in;
       struct pdr_ext *pdr_end;
       struct pdr_ext *pdr_out;
-      struct sym_ext *sym_in;
-      struct sym_ext *sym_end;
-      struct sym_ext *sym_out;
+      int first_pdr;
       struct opt_ext *opt_in;
       struct opt_ext *opt_end;
       struct opt_ext *opt_out;
 
       /* The two BFD's have different endianness, so we must swap
         everything in and out.  This code would always work, but it
-        would be very slow in the normal case.  */
+        would be slow in the normal case.  */
       dnr_in = input_ecoff->external_dnr;
       dnr_end = dnr_in + input_symhdr->idnMax;
       dnr_out = output_ecoff->external_dnr + output_symhdr->idnMax;
@@ -2296,22 +2629,19 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
       pdr_in = input_ecoff->external_pdr;
       pdr_end = pdr_in + input_symhdr->ipdMax;
       pdr_out = output_ecoff->external_pdr + output_symhdr->ipdMax;
+      first_pdr = 1;
+      pdr_off = 0;
       for (; pdr_in < pdr_end; pdr_in++, pdr_out++)
        {
          PDR pdr;
 
          ecoff_swap_pdr_in (input_bfd, pdr_in, &pdr);
          ecoff_swap_pdr_out (output_bfd, &pdr, pdr_out);
-       }
-      sym_in = input_ecoff->external_sym;
-      sym_end = sym_in + input_symhdr->isymMax;
-      sym_out = output_ecoff->external_sym + output_symhdr->isymMax;
-      for (; sym_in < sym_end; sym_in++, sym_out++)
-       {
-         SYMR sym;
-
-         ecoff_swap_sym_in (input_bfd, sym_in, &sym);
-         ecoff_swap_sym_out (output_bfd, &sym, sym_out);
+         if (first_pdr)
+           {
+             pdr_off = pdr.adr;
+             first_pdr = 0;
+           }
        }
       opt_in = input_ecoff->external_opt;
       opt_end = opt_in + input_symhdr->ioptMax;
@@ -2325,16 +2655,9 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
        }
     }
 
-  /* Step through the FDR's of input_bfd, adjust the offsets, and
-     swap them out.  */
-  iss = output_symhdr->issMax;
-  isym = output_symhdr->isymMax;
-  iline = output_symhdr->ilineMax;
-  iopt = output_symhdr->ioptMax;
-  ipdr = output_symhdr->ipdMax;
-  iaux = output_symhdr->iauxMax;
-  irfd = output_symhdr->crfd;
-  cbline = output_symhdr->cbLineOffset;
+  /* Set ifdbase so that the external symbols know how to adjust their
+     ifd values.  */
+  input_ecoff->ifdbase = output_symhdr->ifdMax;
 
   fdr_ptr = input_ecoff->fdr;
   fdr_end = fdr_ptr + input_symhdr->ifdMax;
@@ -2346,27 +2669,30 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
       fdr = *fdr_ptr;
 
       /* The memory address for this fdr is the address for the seclet
-        plus the offset to this fdr within input_bfd.  */
+        plus the offset to this fdr within input_bfd.  For some
+        reason the offset of the first procedure pointer is also
+        added in.  */
       fdr.adr = (bfd_get_section_vma (output_bfd, section)
                 + seclet->offset
-                + (fdr_ptr->adr - input_ecoff->fdr->adr));
-
-      fdr.issBase = iss;
-      iss += fdr.cbSs * sizeof (char);
-      fdr.isymBase = isym;
-      isym += fdr.csym * sizeof (struct sym_ext);
-      fdr.ilineBase = iline;
-      iline += fdr.cline;
-      fdr.ioptBase = iopt;
-      iopt += fdr.copt * sizeof (struct opt_ext);
-      fdr.ipdFirst = ipdr;
-      ipdr += fdr.cpd * sizeof (struct pdr_ext);
-      fdr.iauxBase = iaux;
-      iaux += fdr.caux * sizeof (union aux_ext);
-      fdr.rfdBase = irfd;
-      irfd += fdr.crfd * sizeof (struct rfd_ext);
-      fdr.cbLineOffset = cbline;
-      cbline += fdr.cbLine * sizeof (unsigned char);
+                + (fdr_ptr->adr - input_ecoff->fdr->adr)
+                + pdr_off);
+
+      fdr.issBase += output_symhdr->issMax;
+      fdr.isymBase += output_symhdr->isymMax;
+      fdr.ilineBase += output_symhdr->ilineMax;
+      fdr.ioptBase += output_symhdr->ioptMax;
+      fdr.ipdFirst += output_symhdr->ipdMax;
+      fdr.iauxBase += output_symhdr->iauxMax;
+      fdr.rfdBase += output_symhdr->crfd;
+
+      /* If there are no RFD's, we are going to add some.  We don't
+        want to adjust irfd for this, so that all the FDR's can share
+        the RFD's.  */
+      if (input_symhdr->crfd == 0)
+       fdr.crfd = input_symhdr->ifdMax;
+
+      if (fdr.cbLine != 0)
+       fdr.cbLineOffset += output_symhdr->cbLine;
 
       ecoff_swap_fdr_out (output_bfd, &fdr, fdr_out);
     }
@@ -2414,19 +2740,14 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
       output_symhdr->crfd += input_symhdr->ifdMax;
     }
 
-  /* We have to swap the external symbols in and out because we have
-     to adjust the file descriptor indices.  */
-  ext_ptr = input_ecoff->external_ext;
-  ext_end = ext_ptr + input_symhdr->iextMax;
-  ext_out = output_ecoff->external_ext + output_symhdr->iextMax;
-  for (; ext_ptr < ext_end; ext_ptr++, ext_out++)
-    {
-      EXTR ext;
+  /* Combine the register masks.  */
+  {
+    int i;
 
-      ecoff_swap_ext_in (input_bfd, ext_ptr, &ext);
-      ext.ifd += output_symhdr->ifdMax;
-      ecoff_swap_ext_out (output_bfd, &ext, ext_out);
-    }
+    output_ecoff->gprmask |= input_ecoff->gprmask;
+    for (i = 0; i < 4; i++)
+      output_ecoff->cprmask[i] |= input_ecoff->cprmask[i];
+  }
 
   /* Update the counts.  */
   output_symhdr->ilineMax += input_symhdr->ilineMax;
@@ -2437,19 +2758,7 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
   output_symhdr->ioptMax += input_symhdr->ioptMax;
   output_symhdr->iauxMax += input_symhdr->iauxMax;
   output_symhdr->issMax += input_symhdr->issMax;
-  output_symhdr->issExtMax += input_symhdr->issExtMax;
   output_symhdr->ifdMax += input_symhdr->ifdMax;
-  output_symhdr->iextMax += input_symhdr->iextMax;
-
-  /* Double check that the counts we got by stepping through the FDR's
-     match the counts we got from input_symhdr.  */
-  BFD_ASSERT (output_symhdr->issMax == iss
-             && output_symhdr->isymMax == isym
-             && output_symhdr->ilineMax == iline
-             && output_symhdr->ioptMax == iopt
-             && output_symhdr->ipdMax == ipdr
-             && output_symhdr->iauxMax == iaux
-             && output_symhdr->cbLineOffset == cbline);
 
   return true;
 }
@@ -2458,21 +2767,25 @@ DEFUN (ecoff_get_debug, (output_bfd, seclet, section),
    seclets.  */
 
 static boolean
-DEFUN (ecoff_bfd_seclet_link, (abfd, data, relocateable),
-       bfd *abfd AND
-       PTR data AND
-       boolean relocateable)
+ecoff_bfd_seclet_link (abfd, data, relocateable)
+     bfd *abfd;
+     PTR data;
+     boolean relocateable;
 {
   HDRR *symhdr;
   int ipass;
   register asection *o;
   register bfd_seclet_type *p;
+  asymbol **sym_ptr_ptr;
   bfd_size_type size;
   char *raw;
 
   /* We accumulate the debugging information counts in the symbolic
      header.  */
   symhdr = &ecoff_data (abfd)->symbolic_header;
+  symhdr->magic = magicSym;
+  /* FIXME: What should the version stamp be?  */
+  symhdr->vstamp = 0;
   symhdr->ilineMax = 0;
   symhdr->cbLine = 0;
   symhdr->idnMax = 0;
@@ -2486,13 +2799,18 @@ DEFUN (ecoff_bfd_seclet_link, (abfd, data, relocateable),
   symhdr->crfd = 0;
   symhdr->iextMax = 0;
 
-  /* We are only going to look at each input BFD once.  It is
-     convenient to look at the code section first, so that the first
-     time we look at a BFD we can set the text address (otherwise,
-     when we get to the text section, we would have to be able to
-     track down the file information associated with that BFD).  So we
-     actually do each pass in two sub passes; first the code sections,
-     then the non-code sections.  */
+  /* We need to copy over the debugging symbols from each input BFD.
+     When we do this copying, we have to adjust the text address in
+     the FDR structures, so we have to know the text address used for
+     the input BFD.  Since we only want to copy the symbols once per
+     input BFD, but we are going to look at each input BFD multiple
+     times (once for each section it provides), we arrange to always
+     look at the text section first.  That means that when we copy the
+     debugging information, we always know the text address.  So we
+     actually do each pass in two sub passes; first the text sections,
+     then the non-text sections.  We use the output_has_begun flag to
+     determine whether we have copied over the debugging information
+     yet.  */
 
   /* Do the first pass: set the output section contents and count the
      debugging information.  */
@@ -2518,6 +2836,32 @@ DEFUN (ecoff_bfd_seclet_link, (abfd, data, relocateable),
        }
     }
 
+  /* We handle the external symbols differently.  We use the ones
+     attached to the output_bfd.  The linker will have already
+     determined which symbols are to be attached.  Here we just
+     determine how much space we will need for them.  */
+  sym_ptr_ptr = bfd_get_outsymbols (abfd);
+  if (sym_ptr_ptr != NULL)
+    {
+      asymbol **sym_end;
+
+      sym_end = sym_ptr_ptr + bfd_get_symcount (abfd);
+      for (; sym_ptr_ptr < sym_end; sym_ptr_ptr++)
+       {
+         if (((*sym_ptr_ptr)->flags & BSF_DEBUGGING) == 0
+             && ((*sym_ptr_ptr)->flags & BSF_LOCAL) == 0)
+           {
+             ++symhdr->iextMax;
+             symhdr->issExtMax += strlen ((*sym_ptr_ptr)->name) + 1;
+           }
+       }
+    }
+
+  /* Adjust the counts so that structures are longword aligned.  */
+  symhdr->cbLine = (symhdr->cbLine + 3) &~ 3;
+  symhdr->issMax = (symhdr->issMax + 3) &~ 3;
+  symhdr->issExtMax = (symhdr->issExtMax + 3) &~ 3;
+
   /* Now the counts in symhdr are the correct size for the debugging
      information.  We allocate the right amount of space, and reset
      the counts so that the second pass can use them as indices.  It
@@ -2593,13 +2937,100 @@ DEFUN (ecoff_bfd_seclet_link, (abfd, data, relocateable),
            {
              if (p->type == bfd_indirect_seclet)
                {
-                 if (ecoff_get_debug (abfd, p, o) == false)
+                 if (ecoff_get_debug (abfd, p, o, relocateable) == false)
                    return false;
                }
            }
        }
     }
 
+  /* Put in the external symbols.  */
+  sym_ptr_ptr = bfd_get_outsymbols (abfd);
+  if (sym_ptr_ptr != NULL)
+    {
+      char *ssext;
+      struct ext_ext *external_ext;
+
+      ssext = ecoff_data (abfd)->ssext;
+      external_ext = ecoff_data (abfd)->external_ext;
+      for (; *sym_ptr_ptr != NULL; sym_ptr_ptr++)
+       {
+         asymbol *sym_ptr;
+         EXTR esym;
+
+         sym_ptr = *sym_ptr_ptr;
+
+         if ((sym_ptr->flags & BSF_DEBUGGING) != 0
+             || (sym_ptr->flags & BSF_LOCAL) != 0)
+           continue;
+
+         /* The enative pointer can be NULL for a symbol created by
+            the linker via ecoff_make_empty_symbol.  */
+         if (bfd_asymbol_flavour (sym_ptr) != bfd_target_ecoff_flavour
+             || (((ecoff_symbol_type *) sym_ptr)->native.enative
+                 == (struct ext_ext *) NULL))
+           {
+             esym.jmptbl = 0;
+             esym.cobol_main = 0;
+             esym.weakext = 0;
+             esym.reserved = 0;
+             esym.ifd = ifdNil;
+             /* FIXME: we can do better than this for st and sc.  */
+             esym.asym.st = stGlobal;
+             esym.asym.sc = scAbs;
+             esym.asym.reserved = 0;
+             esym.asym.index = indexNil;
+           }
+         else
+           {
+             ecoff_symbol_type *ecoff_sym_ptr;
+
+             ecoff_sym_ptr = (ecoff_symbol_type *) sym_ptr;
+             if (ecoff_sym_ptr->local)
+               abort ();
+             ecoff_swap_ext_in (abfd, ecoff_sym_ptr->native.enative, &esym);
+
+             /* If we're producing an executable, move common symbols
+                into bss.  */
+             if (relocateable == false)
+               {
+                 if (esym.asym.sc == scCommon)
+                   esym.asym.sc = scBss;
+                 else if (esym.asym.sc == scSCommon)
+                   esym.asym.sc = scSBss;
+               }
+
+             /* Adjust the FDR index for the symbol by that used for
+                the input BFD.  */
+             esym.ifd += ecoff_data (bfd_asymbol_bfd (sym_ptr))->ifdbase;
+           }
+
+         esym.asym.iss = symhdr->issExtMax;
+
+         if (bfd_is_com_section (sym_ptr->section)
+             || sym_ptr->section == &bfd_und_section)
+           esym.asym.value = sym_ptr->value;
+         else
+           esym.asym.value = (sym_ptr->value
+                              + sym_ptr->section->output_offset
+                              + sym_ptr->section->output_section->vma);
+
+         ecoff_swap_ext_out (abfd, &esym, external_ext + symhdr->iextMax);
+
+         ecoff_set_sym_index (sym_ptr, symhdr->iextMax);
+
+         ++symhdr->iextMax;
+
+         strcpy (ssext + symhdr->issExtMax, sym_ptr->name);
+         symhdr->issExtMax += strlen (sym_ptr->name) + 1;
+       }
+    }
+
+  /* Adjust the counts so that structures are longword aligned.  */
+  symhdr->cbLine = (symhdr->cbLine + 3) &~ 3;
+  symhdr->issMax = (symhdr->issMax + 3) &~ 3;
+  symhdr->issExtMax = (symhdr->issExtMax + 3) &~ 3;
+
   return true;
 }
 \f
@@ -2608,56 +3039,69 @@ DEFUN (ecoff_bfd_seclet_link, (abfd, data, relocateable),
    the return value.  */
 
 static boolean
-DEFUN (ecoff_set_arch_mach, (abfd, arch, machine),
-       bfd *abfd AND
-       enum bfd_architecture arch AND
-       unsigned long machine)
+ecoff_set_arch_mach (abfd, arch, machine)
+     bfd *abfd;
+     enum bfd_architecture arch;
+     unsigned long machine;
 {
   bfd_default_set_arch_mach (abfd, arch, machine);
   return arch == bfd_arch_mips;
 }
 
+/* Get the size of the section headers.  We do not output the .scommon
+   section which we created in ecoff_mkobject.  */
+
+static int
+ecoff_sizeof_headers (abfd, reloc)
+     bfd *abfd;
+     boolean reloc;
+{
+  return FILHSZ + AOUTSZ + (abfd->section_count - 1) * SCNHSZ;
+}
+
 /* Calculate the file position for each section, and set
    reloc_filepos.  */
 
 static void
-DEFUN (ecoff_compute_section_file_positions, (abfd),
-       bfd *abfd)
+ecoff_compute_section_file_positions (abfd)
+     bfd *abfd;
 {
   asection *current;
-  asection *previous;
   file_ptr sofar;
   file_ptr old_sofar;
-
-  sofar = FILHSZ;
+  boolean first_data;
 
   if (bfd_get_start_address (abfd)) 
-    {
-      /* A start address may have been added to the original file. In
-        this case it will need an optional header to record it.  */
-      abfd->flags |= EXEC_P;
-    }
+    abfd->flags |= EXEC_P;
 
-  if (abfd->flags & EXEC_P)
-    sofar += AOUTSZ;
-
-  sofar += abfd->section_count * SCNHSZ;
+  sofar = ecoff_sizeof_headers (abfd, false);
 
-  previous = (asection *) NULL;
+  first_data = true;
   for (current = abfd->sections;
        current != (asection *) NULL;
        current = current->next)
     {
       /* Only deal with sections which have contents */
-      if (! (current->flags & SEC_HAS_CONTENTS))
+      if (! (current->flags & SEC_HAS_CONTENTS)
+         || strcmp (current->name, SCOMMON) == 0)
        continue;
 
+      /* On Ultrix, the data sections in an executable file must be
+        aligned to a page boundary within the file.  This does not
+        affect the section size, though.  FIXME: Does this work for
+        other platforms?  */
+      if ((abfd->flags & EXEC_P) != 0
+         && first_data != false
+         && (current->flags & SEC_CODE) == 0)
+       {
+         sofar = (sofar + PAGE_SIZE - 1) &~ (PAGE_SIZE - 1);
+         first_data = false;
+       }
+
       /* Align the sections in the file to the same boundary on
         which they are aligned in virtual memory.  */
       old_sofar = sofar;
       sofar = BFD_ALIGN (sofar, 1 << current->alignment_power);
-      if (previous != (asection *) NULL)
-       previous->_raw_size += sofar - old_sofar;
 
       current->filepos = sofar;
 
@@ -2667,7 +3111,6 @@ DEFUN (ecoff_compute_section_file_positions, (abfd),
       old_sofar = sofar;
       sofar = BFD_ALIGN (sofar, 1 << current->alignment_power);
       current->_raw_size += sofar - old_sofar;
-      previous = current;
     }
 
   ecoff_data (abfd)->reloc_filepos = sofar;
@@ -2676,12 +3119,12 @@ DEFUN (ecoff_compute_section_file_positions, (abfd),
 /* Set the contents of a section.  */
 
 static boolean
-DEFUN (ecoff_set_section_contents, (abfd, section, location, offset, count),
-       bfd *abfd AND
-       asection *section AND
-       PTR location AND
-       file_ptr offset AND
-       bfd_size_type count)
+ecoff_set_section_contents (abfd, section, location, offset, count)
+     bfd *abfd;
+     asection *section;
+     PTR location;
+     file_ptr offset;
+     bfd_size_type count;
 {
   if (abfd->output_has_begun == false)
     ecoff_compute_section_file_positions (abfd);
@@ -2697,8 +3140,8 @@ DEFUN (ecoff_set_section_contents, (abfd, section, location, offset, count),
 /* Write out an ECOFF file.  */
 
 static boolean
-DEFUN (ecoff_write_object_contents, (abfd),
-       bfd *abfd)
+ecoff_write_object_contents (abfd)
+     bfd *abfd;
 {
   asection *current;
   unsigned int count;
@@ -2713,6 +3156,7 @@ DEFUN (ecoff_write_object_contents, (abfd),
   unsigned long bss_size;
   struct internal_filehdr internal_f;
   struct internal_aouthdr internal_a;
+  int i;
 
   bfd_error = system_call_error;
 
@@ -2731,6 +3175,8 @@ DEFUN (ecoff_write_object_contents, (abfd),
        current != (asection *)NULL; 
        current = current->next) 
     {
+      if (strcmp (current->name, SCOMMON) == 0)
+       continue;
       current->target_index = count;
       ++count;
       if (current->reloc_count != 0)
@@ -2747,9 +3193,16 @@ DEFUN (ecoff_write_object_contents, (abfd),
     }
 
   sym_base = reloc_base + reloc_size;
+
+  /* At least on Ultrix, the symbol table of an executable file must
+     be aligned to a page boundary.  FIXME: Is this true on other
+     platforms?  */
+  if ((abfd->flags & EXEC_P) != 0)
+    sym_base = (sym_base + PAGE_SIZE - 1) &~ (PAGE_SIZE - 1);
+
   ecoff_data (abfd)->sym_filepos = sym_base;
 
-  text_size = 0;
+  text_size = ecoff_sizeof_headers (abfd, false);
   text_start = 0;
   data_size = 0;
   data_start = 0;
@@ -2758,10 +3211,7 @@ DEFUN (ecoff_write_object_contents, (abfd),
   /* Write section headers to the file.  */
 
   internal_f.f_nscns = 0;
-  if (bfd_seek (abfd,
-               (file_ptr) ((abfd->flags & EXEC_P) ?
-                           (FILHSZ + AOUTSZ) : FILHSZ),
-               SEEK_SET) != 0)
+  if (bfd_seek (abfd, (file_ptr) (FILHSZ + AOUTSZ), SEEK_SET) != 0)
     return false;
   for (current = abfd->sections;
        current != (asection *) NULL;
@@ -2770,6 +3220,13 @@ DEFUN (ecoff_write_object_contents, (abfd),
       struct internal_scnhdr section;
       bfd_vma vma;
 
+      if (strcmp (current->name, SCOMMON) == 0)
+       {
+         BFD_ASSERT (bfd_get_section_size_before_reloc (current) == 0
+                     && current->reloc_count == 0);
+         continue;
+       }
+
       ++internal_f.f_nscns;
 
       strncpy (section.s_name, current->name, sizeof section.s_name);
@@ -2805,7 +3262,8 @@ DEFUN (ecoff_write_object_contents, (abfd),
 
       section.s_nreloc = current->reloc_count;
       section.s_nlnno = 0;
-      section.s_flags = sec_to_styp_flags (current->name, current->flags);
+      section.s_flags = ecoff_sec_to_styp_flags (current->name,
+                                                current->flags);
 
       {
        SCNHDR buff;
@@ -2815,18 +3273,16 @@ DEFUN (ecoff_write_object_contents, (abfd),
          return false;
       }
 
-      /* FIXME: These numbers don't add up to what the MIPS tools
-        produce, although I don't think it matters.  */
-      if ((section.s_flags & STYP_TEXT) != 0
-         || (section.s_flags & STYP_RDATA) != 0
-         || (section.s_flags & STYP_LIT8) != 0
-         || (section.s_flags & STYP_LIT4) != 0)
+      if ((section.s_flags & STYP_TEXT) != 0)
        {
          text_size += bfd_get_section_size_before_reloc (current);
          if (text_start == 0 || text_start > vma)
            text_start = vma;
        }
-      else if ((section.s_flags & STYP_DATA) != 0
+      else if ((section.s_flags & STYP_RDATA) != 0
+              || (section.s_flags & STYP_DATA) != 0
+              || (section.s_flags & STYP_LIT8) != 0
+              || (section.s_flags & STYP_LIT4) != 0
               || (section.s_flags & STYP_SDATA) != 0)
        {
          data_size += bfd_get_section_size_before_reloc (current);
@@ -2840,7 +3296,10 @@ DEFUN (ecoff_write_object_contents, (abfd),
 
   /* Set up the file header.  */
 
-  internal_f.f_magic = MIPS_MAGIC_2;
+  if (abfd->xvec->header_byteorder_big_p != false)
+    internal_f.f_magic = MIPS_MAGIC_BIG;
+  else
+    internal_f.f_magic = MIPS_MAGIC_LITTLE;
 
   /*
     We will NOT put a fucking timestamp in the header here. Every time you
@@ -2850,18 +3309,22 @@ DEFUN (ecoff_write_object_contents, (abfd),
     */
   internal_f.f_timdat = 0;
 
-  internal_f.f_nsyms =  bfd_get_symcount(abfd);
-  if (internal_f.f_nsyms != 0)
-    internal_f.f_symptr = sym_base;
+  if (bfd_get_symcount (abfd) != 0)
+    {
+      /* The ECOFF f_nsyms field is not actually the number of
+        symbols, it's the size of symbolic information header.  */
+      internal_f.f_nsyms = sizeof (struct hdr_ext);
+      internal_f.f_symptr = sym_base;
+    }
   else
-    internal_f.f_symptr = 0;
+    {
+      internal_f.f_nsyms = 0;
+      internal_f.f_symptr = 0;
+    }
 
-  if (abfd->flags & EXEC_P)
-    internal_f.f_opthdr = AOUTSZ;
-  else
-    internal_f.f_opthdr = 0;
+  internal_f.f_opthdr = AOUTSZ;
 
-  internal_f.f_flags = 0;
+  internal_f.f_flags = F_LNNO;
   if (reloc_size == 0)
     internal_f.f_flags |= F_RELFLG;
   if (bfd_get_symcount (abfd) == 0)
@@ -2874,25 +3337,39 @@ DEFUN (ecoff_write_object_contents, (abfd),
   else
     internal_f.f_flags |= F_AR32W;
 
-  /* Set up the optional header.  */
-
-  if ((abfd->flags & EXEC_P) != 0)
-    {
-      internal_a.magic = ZMAGIC;
+  /* Set up the ``optional'' header.  */
+  internal_a.magic = ZMAGIC;
+
+  /* FIXME: This is what Ultrix puts in, and it makes the Ultrix
+     linker happy.  But, is it right?  */
+  internal_a.vstamp = 0x20a;
+
+  /* At least on Ultrix, these have to be rounded to page boundaries.
+     FIXME: Is this true on other platforms?  */
+  internal_a.tsize = (text_size + PAGE_SIZE - 1) &~ (PAGE_SIZE - 1);
+  internal_a.text_start = text_start &~ (PAGE_SIZE - 1);
+  internal_a.dsize = (data_size + PAGE_SIZE - 1) &~ (PAGE_SIZE - 1);
+  internal_a.data_start = data_start &~ (PAGE_SIZE - 1);
+
+  /* On Ultrix, the initial portions of the .sbss and .bss segments
+     are at the end of the data section.  The bsize field in the
+     optional header records how many bss bytes are required beyond
+     those in the data section.  The value is not rounded to a page
+     boundary.  */
+  if (bss_size < internal_a.dsize - data_size)
+    bss_size = 0;
+  else
+    bss_size -= internal_a.dsize - data_size;
+  internal_a.bsize = bss_size;
+  internal_a.bss_start = internal_a.data_start + internal_a.dsize;
 
-      /* FIXME: What should this be?  */
-      internal_a.vstamp = 0;
+  internal_a.entry = bfd_get_start_address (abfd);
 
-      internal_a.tsize = text_size;
-      internal_a.text_start = text_start;
-      internal_a.dsize = data_size;
-      internal_a.data_start = data_start;
-      internal_a.bsize = bss_size;
+  internal_a.gp_value = ecoff_data (abfd)->gp;
 
-      internal_a.entry = bfd_get_start_address (abfd);
-
-      /* FIXME: The MIPS optional header is larger than this....  */
-    }
+  internal_a.gprmask = ecoff_data (abfd)->gprmask;
+  for (i = 0; i < 4; i++)
+    internal_a.cprmask[i] = ecoff_data (abfd)->cprmask[i];
 
   /* Write out the file header and the optional header.  */
 
@@ -2906,22 +3383,129 @@ DEFUN (ecoff_write_object_contents, (abfd),
       return false;
   }
 
-  if ((abfd->flags & EXEC_P) != 0)
+  {
+    AOUTHDR buff;
+
+    ecoff_swap_aouthdr_out (abfd, (PTR) &internal_a, (PTR) &buff);
+    if (bfd_write ((PTR) &buff, 1, AOUTSZ, abfd) != AOUTSZ)
+      return false;
+  }
+
+  /* Write out the relocs.  */
+  for (current = abfd->sections;
+       current != (asection *) NULL;
+       current = current->next)
     {
-      AOUTHDR buff;
+      RELOC *buff;
+      arelent **reloc_ptr_ptr;
+      arelent **reloc_end;
+      RELOC *out_ptr;
+
+      if (current->reloc_count == 0)
+       continue;
+
+      buff = (RELOC *) bfd_alloc (abfd, current->reloc_count * RELSZ);
+      if (buff == (RELOC *) NULL)
+       {
+         bfd_error = no_memory;
+         return false;
+       }
+
+      reloc_ptr_ptr = current->orelocation;
+      reloc_end = reloc_ptr_ptr + current->reloc_count;
+      out_ptr = buff;
+      for (; reloc_ptr_ptr < reloc_end; reloc_ptr_ptr++, out_ptr++)
+       {
+         arelent *reloc;
+         asymbol *sym;
+         struct internal_reloc in;
+         
+         memset (&in, 0, sizeof in);
+
+         reloc = *reloc_ptr_ptr;
+         sym = *reloc->sym_ptr_ptr;
+
+         in.r_vaddr = reloc->address + bfd_get_section_vma (abfd, current);
+         in.r_type = reloc->howto->type;
+         if ((sym->flags & BSF_SECTION_SYM) == 0)
+           {
+             in.r_symndx = ecoff_get_sym_index (*reloc->sym_ptr_ptr);
+             in.r_extern = 1;
+           }
+         else
+           {
+             CONST char *name;
+
+             name = bfd_get_section_name (abfd, bfd_get_section (sym));
+             if (strcmp (name, ".text") == 0)
+               in.r_symndx = RELOC_SECTION_TEXT;
+             else if (strcmp (name, ".rdata") == 0)
+               in.r_symndx = RELOC_SECTION_RDATA;
+             else if (strcmp (name, ".data") == 0)
+               in.r_symndx = RELOC_SECTION_DATA;
+             else if (strcmp (name, ".sdata") == 0)
+               in.r_symndx = RELOC_SECTION_SDATA;
+             else if (strcmp (name, ".sbss") == 0)
+               in.r_symndx = RELOC_SECTION_SBSS;
+             else if (strcmp (name, ".bss") == 0)
+               in.r_symndx = RELOC_SECTION_BSS;
+             else if (strcmp (name, ".init") == 0)
+               in.r_symndx = RELOC_SECTION_INIT;
+             else if (strcmp (name, ".lit8") == 0)
+               in.r_symndx = RELOC_SECTION_LIT8;
+             else if (strcmp (name, ".lit4") == 0)
+               in.r_symndx = RELOC_SECTION_LIT4;
+             else
+               abort ();
+             in.r_extern = 0;
+           }
 
-      ecoff_swap_aouthdr_out (abfd, (PTR) &internal_a, (PTR) &buff);
-      if (bfd_write ((PTR) &buff, 1, AOUTSZ, abfd) != AOUTSZ)
+         ecoff_swap_reloc_out (abfd, (PTR) &in, (PTR) out_ptr);
+       }
+
+      if (bfd_seek (abfd, current->rel_filepos, SEEK_SET) != 0)
+       return false;
+      if (bfd_write ((PTR) buff, RELSZ, current->reloc_count, abfd)
+         != RELSZ * current->reloc_count)
        return false;
+      bfd_release (abfd, (PTR) buff);
     }
 
-  /* Write out the relocs.  */
-
   /* Write out the symbolic debugging information.  */
   if (bfd_get_symcount (abfd) > 0)
     {
+      HDRR *symhdr;
+      unsigned long sym_offset;
       struct hdr_ext buff;
 
+      /* Set up the offsets in the symbolic header.  */
+      symhdr = &ecoff_data (abfd)->symbolic_header;
+      sym_offset = ecoff_data (abfd)->sym_filepos + sizeof (struct hdr_ext);
+
+#define SET(offset, size, ptr) \
+  if (symhdr->size == 0) \
+    symhdr->offset = 0; \
+  else \
+    symhdr->offset = (((char *) ecoff_data (abfd)->ptr \
+                      - (char *) ecoff_data (abfd)->raw_syments) \
+                     + sym_offset);
+
+      SET (cbLineOffset, cbLine, line);
+      SET (cbDnOffset, idnMax, external_dnr);
+      SET (cbPdOffset, ipdMax, external_pdr);
+      SET (cbSymOffset, isymMax, external_sym);
+      SET (cbOptOffset, ioptMax, external_opt);
+      SET (cbAuxOffset, iauxMax, external_aux);
+      SET (cbSsOffset, issMax, ss);
+      SET (cbSsExtOffset, issExtMax, ssext);
+      SET (cbFdOffset, ifdMax, external_fdr);
+      SET (cbRfdOffset, crfd, external_rfd);
+      SET (cbExtOffset, iextMax, external_ext);
+#undef SET
+
+      if (bfd_seek (abfd, (file_ptr) ecoff_data (abfd)->sym_filepos,
+                   SEEK_SET) != 0)
+       return false;
       ecoff_swap_hdr_out (abfd, &ecoff_data (abfd)->symbolic_header, &buff);
       if (bfd_write ((PTR) &buff, 1, sizeof buff, abfd) != sizeof buff)
        return false;
@@ -2934,6 +3518,439 @@ DEFUN (ecoff_write_object_contents, (abfd),
   return true;
 }
 \f
+/* Archive handling.  ECOFF uses what appears to be a unique type of
+   archive header (which I call an armap).  The byte ordering of the
+   armap and the contents are encoded in the name of the armap itself.
+   At least for now, we only support archives with the same byte
+   ordering in the armap and the contents.
+
+   The first four bytes in the armap are the number of symbol
+   definitions.  This is always a power of two.
+
+   This is followed by the symbol definitions.  Each symbol definition
+   occupies 8 bytes.  The first four bytes are the offset from the
+   start of the armap strings to the null-terminated string naming
+   this symbol.  The second four bytes are the file offset to the
+   archive member which defines this symbol.  If the second four bytes
+   are 0, then this is not actually a symbol definition, and it should
+   be ignored.
+
+   The symbols are hashed into the armap with a closed hashing scheme.
+   See the functions below for the details of the algorithm.
+
+   We could use the hash table when looking up symbols in a library.
+   This would require a new BFD target entry point to replace the
+   bfd_get_next_mapent function used by the linker.
+
+   After the symbol definitions comes four bytes holding the size of
+   the string table, followed by the string table itself.  */
+
+/* The name of an archive headers looks like this:
+   __________E[BL]E[BL]_ (with a trailing space).
+   The trailing space is changed to an X if the archive is changed to
+   indicate that the armap is out of date.  */
+
+#define ARMAP_BIG_ENDIAN 'B'
+#define ARMAP_LITTLE_ENDIAN 'L'
+#define ARMAP_MARKER 'E'
+#define ARMAP_START "__________"
+#define ARMAP_HEADER_MARKER_INDEX 10
+#define ARMAP_HEADER_ENDIAN_INDEX 11
+#define ARMAP_OBJECT_MARKER_INDEX 12
+#define ARMAP_OBJECT_ENDIAN_INDEX 13
+#define ARMAP_END_INDEX 14
+#define ARMAP_END "_ "
+
+/* This is a magic number used in the hashing algorithm.  */
+#define ARMAP_HASH_MAGIC 0x9dd68ab5
+
+/* This returns the hash value to use for a string.  It also sets
+   *REHASH to the rehash adjustment if the first slot is taken.  SIZE
+   is the number of entries in the hash table, and HLOG is the log
+   base 2 of SIZE.  */
+
+static unsigned int
+ecoff_armap_hash (s, rehash, size, hlog)
+     CONST char *s;
+     unsigned int *rehash;
+     unsigned int size;
+     unsigned int hlog;
+{
+  unsigned int hash;
+
+  hash = *s++;
+  while (*s != '\0')
+    hash = ((hash >> 27) | (hash << 5)) + *s++;
+  hash *= ARMAP_HASH_MAGIC;
+  *rehash = (hash & (size - 1)) | 1;
+  return hash >> (32 - hlog);
+}
+
+/* Read in the armap.  */
+
+static boolean
+ecoff_slurp_armap (abfd)
+     bfd *abfd;
+{
+  char nextname[17];
+  unsigned int i;
+  struct areltdata *mapdata;
+  bfd_size_type parsed_size;
+  char *raw_armap;
+  struct artdata *ardata;
+  unsigned int count;
+  char *raw_ptr;
+  struct symdef *symdef_ptr;
+  char *stringbase;
+  
+  /* Get the name of the first element.  */
+  i = bfd_read ((PTR) nextname, 1, 16, abfd);
+  if (i == 0)
+      return true;
+  if (i != 16)
+      return false;
+
+  bfd_seek (abfd, (file_ptr) -16, SEEK_CUR);
+
+  /* See if the first element is an armap.  */
+  if (strncmp (nextname, ARMAP_START, sizeof ARMAP_START - 1) != 0
+      || nextname[ARMAP_HEADER_MARKER_INDEX] != ARMAP_MARKER
+      || (nextname[ARMAP_HEADER_ENDIAN_INDEX] != ARMAP_BIG_ENDIAN
+         && nextname[ARMAP_HEADER_ENDIAN_INDEX] != ARMAP_LITTLE_ENDIAN)
+      || nextname[ARMAP_OBJECT_MARKER_INDEX] != ARMAP_MARKER
+      || (nextname[ARMAP_OBJECT_ENDIAN_INDEX] != ARMAP_BIG_ENDIAN
+         && nextname[ARMAP_OBJECT_ENDIAN_INDEX] != ARMAP_LITTLE_ENDIAN)
+      || strncmp (nextname + ARMAP_END_INDEX,
+                 ARMAP_END, sizeof ARMAP_END - 1) != 0)
+    {
+      bfd_has_map (abfd) = false;
+      return true;
+    }
+
+  /* Make sure we have the right byte ordering.  */
+  if (((nextname[ARMAP_HEADER_ENDIAN_INDEX] == ARMAP_BIG_ENDIAN)
+       ^ (abfd->xvec->header_byteorder_big_p != false))
+      || ((nextname[ARMAP_OBJECT_ENDIAN_INDEX] == ARMAP_BIG_ENDIAN)
+         ^ (abfd->xvec->byteorder_big_p != false)))
+    {
+      bfd_error = wrong_format;
+      return false;
+    }
+
+  /* Read in the armap.  */
+  ardata = bfd_ardata (abfd);
+  mapdata = snarf_ar_hdr (abfd);
+  if (mapdata == (struct areltdata *) NULL)
+    return false;
+  parsed_size = mapdata->parsed_size;
+  bfd_release (abfd, (PTR) mapdata);
+    
+  raw_armap = (char *) bfd_alloc (abfd, parsed_size);
+  if (raw_armap == (char *) NULL)
+    {
+      bfd_error = no_memory;
+      return false;
+    }
+    
+  if (bfd_read ((PTR) raw_armap, 1, parsed_size, abfd) != parsed_size)
+    {
+      bfd_error = malformed_archive;
+      bfd_release (abfd, (PTR) raw_armap);
+      return false;
+    }
+    
+  count = bfd_h_get_32 (abfd, (PTR) raw_armap);
+
+  ardata->symdef_count = 0;
+  ardata->cache = (struct ar_cache *) NULL;
+
+  /* Hack: overlay the symdefs on top of the raw archive data.  This
+     is the way do_slurp_bsd_armap works.  */
+  raw_ptr = raw_armap + LONG_SIZE;
+  symdef_ptr = (struct symdef *) raw_ptr;
+  ardata->symdefs = (carsym *) symdef_ptr;
+  stringbase = raw_ptr + count * (2 * LONG_SIZE) + LONG_SIZE;
+
+#ifdef CHECK_ARMAP_HASH
+  {
+    unsigned int hlog;
+
+    /* Double check that I have the hashing algorithm right by making
+       sure that every symbol can be looked up successfully.  */
+    hlog = 0;
+    for (i = 1; i < count; i <<= 1)
+      hlog++;
+    BFD_ASSERT (i == count);
+
+    for (i = 0; i < count; i++, raw_ptr += 2 * LONG_SIZE)
+      {
+       unsigned int name_offset, file_offset;
+       unsigned int hash, rehash, srch;
+      
+       name_offset = bfd_h_get_32 (abfd, (PTR) raw_ptr);
+       file_offset = bfd_h_get_32 (abfd, (PTR) (raw_ptr + LONG_SIZE));
+       if (file_offset == 0)
+         continue;
+       hash = ecoff_armap_hash (stringbase + name_offset, &rehash, count,
+                                hlog);
+       if (hash == i)
+         continue;
+
+       /* See if we can rehash to this location.  */
+       for (srch = (hash + rehash) & (count - 1);
+            srch != hash && srch != i;
+            srch = (srch + rehash) & (count - 1))
+         BFD_ASSERT (bfd_h_get_32 (abfd,
+                                   (PTR) (raw_armap
+                                          + LONG_SIZE
+                                          + (srch * 2 * LONG_SIZE)
+                                          + LONG_SIZE))
+                     != 0);
+       BFD_ASSERT (srch == i);
+      }
+  }
+
+  raw_ptr = raw_armap + LONG_SIZE;
+#endif /* CHECK_ARMAP_HASH */
+
+  for (i = 0; i < count; i++, raw_ptr += 2 * LONG_SIZE)
+    {
+      unsigned int name_offset, file_offset;
+
+      name_offset = bfd_h_get_32 (abfd, (PTR) raw_ptr);
+      file_offset = bfd_h_get_32 (abfd, (PTR) (raw_ptr + LONG_SIZE));
+      if (file_offset == 0)
+       continue;
+      symdef_ptr->s.name = stringbase + name_offset;
+      symdef_ptr->file_offset = file_offset;
+      ++symdef_ptr;
+      ++ardata->symdef_count;
+    }
+
+  ardata->first_file_filepos = bfd_tell (abfd);
+  /* Pad to an even boundary.  */
+  ardata->first_file_filepos += ardata->first_file_filepos % 2;
+
+  bfd_has_map (abfd) = true;
+
+  return true;
+}
+
+/* Write out an armap.  */
+
+static boolean
+ecoff_write_armap (abfd, elength, map, orl_count, stridx)
+     bfd *abfd;
+     unsigned int elength;
+     struct orl *map;
+     unsigned int orl_count;
+     int stridx;
+{
+  unsigned int hashsize, hashlog;
+  unsigned int symdefsize;
+  int padit;
+  unsigned int stringsize;
+  unsigned int mapsize;
+  file_ptr firstreal;
+  struct ar_hdr hdr;
+  struct stat statbuf;
+  unsigned int i;
+  bfd_byte temp[LONG_SIZE];
+  bfd_byte *hashtable;
+  bfd *current;
+  bfd *last_elt;
+
+  /* Ultrix appears to use as a hash table size the least power of two
+     greater than twice the number of entries.  */
+  for (hashlog = 0; (1 << hashlog) <= 2 * orl_count; hashlog++)
+    ;
+  hashsize = 1 << hashlog;
+
+  symdefsize = hashsize * 2 * LONG_SIZE;
+  padit = stridx % 2;
+  stringsize = stridx + padit;
+
+  /* Include 8 bytes to store symdefsize and stringsize in output. */
+  mapsize = LONG_SIZE + symdefsize + stringsize + LONG_SIZE;
+
+  firstreal = SARMAG + sizeof (struct ar_hdr) + mapsize + elength;
+
+  memset ((PTR) &hdr, 0, sizeof hdr);
+
+  /* Work out the ECOFF armap name.  */
+  strcpy (hdr.ar_name, ARMAP_START);
+  hdr.ar_name[ARMAP_HEADER_MARKER_INDEX] = ARMAP_MARKER;
+  hdr.ar_name[ARMAP_HEADER_ENDIAN_INDEX] =
+    (abfd->xvec->header_byteorder_big_p
+     ? ARMAP_BIG_ENDIAN
+     : ARMAP_LITTLE_ENDIAN);
+  hdr.ar_name[ARMAP_OBJECT_MARKER_INDEX] = ARMAP_MARKER;
+  hdr.ar_name[ARMAP_OBJECT_ENDIAN_INDEX] =
+    abfd->xvec->byteorder_big_p ? ARMAP_BIG_ENDIAN : ARMAP_LITTLE_ENDIAN;
+  memcpy (hdr.ar_name + ARMAP_END_INDEX, ARMAP_END, sizeof ARMAP_END - 1);
+
+  /* Write the timestamp of the archive header to be just a little bit
+     later than the timestamp of the file, otherwise the linker will
+     complain that the index is out of date.  Actually, the Ultrix
+     linker just checks the archive name; the GNU linker may check the
+     date.  */
+  stat (abfd->filename, &statbuf);
+  sprintf (hdr.ar_date, "%ld", (long) (statbuf.st_mtime + 60));
+
+  /* The DECstation uses zeroes for the uid, gid and mode of the
+     armap.  */
+  hdr.ar_uid[0] = '0';
+  hdr.ar_gid[0] = '0';
+  hdr.ar_mode[0] = '0';
+
+  sprintf (hdr.ar_size, "%-10d", (int) mapsize);
+
+  hdr.ar_fmag[0] = '`';
+  hdr.ar_fmag[1] = '\n';
+
+  /* Turn all null bytes in the header into spaces.  */
+  for (i = 0; i < sizeof (struct ar_hdr); i++)
+   if (((char *)(&hdr))[i] == '\0')
+     (((char *)(&hdr))[i]) = ' ';
+
+  if (bfd_write ((PTR) &hdr, 1, sizeof (struct ar_hdr), abfd)
+      != sizeof (struct ar_hdr))
+    return false;
+
+  bfd_h_put_32 (abfd, hashsize, temp);
+  if (bfd_write (temp, 1, LONG_SIZE, abfd) != LONG_SIZE)
+    return false;
+  
+  hashtable = (bfd_byte *) bfd_zalloc (abfd, symdefsize);
+
+  current = abfd->archive_head;
+  last_elt = current;
+  for (i = 0; i < orl_count; i++)
+    {
+      unsigned int hash, rehash;
+
+      /* Advance firstreal to the file position of this archive
+        element.  */
+      if (((bfd *) map[i].pos) != last_elt)
+       {
+         do
+           {
+             firstreal += arelt_size (current) + sizeof (struct ar_hdr);
+             firstreal += firstreal % 2;
+             current = current->next;
+           }
+         while (current != (bfd *) map[i].pos);
+       }
+
+      last_elt = current;
+
+      hash = ecoff_armap_hash (*map[i].name, &rehash, hashsize, hashlog);
+      if (bfd_h_get_32 (abfd, (PTR) (hashtable
+                                    + (hash * 2 * LONG_SIZE)
+                                    + LONG_SIZE))
+         != 0)
+       {
+         unsigned int srch;
+
+         /* The desired slot is already taken.  */
+         for (srch = (hash + rehash) & (hashsize - 1);
+              srch != hash;
+              srch = (srch + rehash) & (hashsize - 1))
+           if (bfd_h_get_32 (abfd, (PTR) (hashtable
+                                          + (srch * 2 * LONG_SIZE)
+                                          + LONG_SIZE))
+               == 0)
+             break;
+
+         BFD_ASSERT (srch != hash);
+
+         hash = srch;
+       }
+       
+      bfd_h_put_32 (abfd, map[i].namidx,
+                   (PTR) (hashtable + hash * 2 * LONG_SIZE));
+      bfd_h_put_32 (abfd, firstreal,
+                   (PTR) (hashtable + hash * 2 * LONG_SIZE + LONG_SIZE));
+    }
+
+  if (bfd_write (hashtable, 1, symdefsize, abfd) != symdefsize)
+    return false;
+
+  bfd_release (abfd, hashtable);
+
+  /* Now write the strings.  */
+  bfd_h_put_32 (abfd, stringsize, temp);
+  if (bfd_write (temp, 1, LONG_SIZE, abfd) != LONG_SIZE)
+    return false;
+  for (i = 0; i < orl_count; i++)
+    {
+      bfd_size_type len;
+
+      len = strlen (*map[i].name) + 1;
+      if (bfd_write ((PTR) (*map[i].name), 1, len, abfd) != len)
+       return false;
+    }
+
+  /* The spec sez this should be a newline.  But in order to be
+     bug-compatible for DECstation ar we use a null.  */
+  if (padit)
+    {
+      if (bfd_write ("\0", 1, 1, abfd) != 1)
+       return false;
+    }
+
+  return true;
+}
+
+/* We just use the generic extended name support.  This is a GNU
+   extension.  */
+#define ecoff_slurp_extended_name_table        _bfd_slurp_extended_name_table
+
+/* See whether this BFD is an archive.  If it is, read in the armap
+   and the extended name table.  */
+
+static bfd_target *
+ecoff_archive_p (abfd)
+     bfd *abfd;
+{
+  char armag[SARMAG + 1];
+
+  if (bfd_read ((PTR) armag, 1, SARMAG, abfd) != SARMAG
+      || strncmp (armag, ARMAG, SARMAG) != 0)
+    {
+      bfd_error = wrong_format;
+      return (bfd_target *) NULL;
+    }
+
+  /* We are setting bfd_ardata(abfd) here, but since bfd_ardata
+     involves a cast, we can't do it as the left operand of
+     assignment.  */
+  abfd->tdata.aout_ar_data =
+    (struct artdata *) bfd_zalloc (abfd, sizeof (struct artdata));
+
+  if (bfd_ardata (abfd) == (struct artdata *) NULL)
+    {
+      bfd_error = no_memory;
+      return (bfd_target *) NULL;
+    }
+
+  bfd_ardata (abfd)->first_file_filepos = SARMAG;
+  
+  if (ecoff_slurp_armap (abfd) == false
+      || ecoff_slurp_extended_name_table (abfd) == false)
+    {
+      bfd_release (abfd, bfd_ardata (abfd));
+      abfd->tdata.aout_ar_data = (struct artdata *) NULL;
+      return (bfd_target *) NULL;
+    }
+  
+  return abfd->xvec;
+}
+\f
+/* This is the COFF backend structure.  The backend_data field of the
+   bfd_target structure is set to this.  The section reading code in
+   coffgen.c uses this structure.  */
+
 static CONST bfd_coff_backend_data bfd_ecoff_std_swap_table = {
   (void (*) PARAMS ((bfd *,PTR,int,int,PTR))) bfd_void, /* aux_in */
   (void (*) PARAMS ((bfd *,PTR,PTR))) bfd_void, /* sym_in */
@@ -2946,7 +3963,7 @@ static CONST bfd_coff_backend_data bfd_ecoff_std_swap_table = {
   FILHSZ, AOUTSZ, SCNHSZ, 0, 0, 0, true,
   ecoff_swap_filehdr_in, ecoff_swap_aouthdr_in, ecoff_swap_scnhdr_in,
   ecoff_bad_format_hook, ecoff_set_arch_mach_hook, ecoff_mkobject_hook,
-  styp_to_sec_flags, ecoff_make_section_hook, ecoff_set_alignment_hook,
+  ecoff_styp_to_sec_flags, ecoff_make_section_hook, ecoff_set_alignment_hook,
   ecoff_slurp_symbol_table
 };
 
@@ -2956,24 +3973,24 @@ static CONST bfd_coff_backend_data bfd_ecoff_std_swap_table = {
 #define ecoff_get_lineno \
   ((alent *(*) PARAMS ((bfd *, asymbol *))) bfd_nullvoidptr)
 
+/* These bfd_target functions are defined in other files.  */
+
 #define ecoff_core_file_failing_command        _bfd_dummy_core_file_failing_command
 #define ecoff_core_file_failing_signal _bfd_dummy_core_file_failing_signal
-#define ecoff_core_file_matches_executable_p   _bfd_dummy_core_file_matches_executable_p
-#define ecoff_slurp_armap              bfd_slurp_coff_armap
-#define ecoff_slurp_extended_name_table        _bfd_slurp_extended_name_table
-#define ecoff_write_armap              coff_write_armap
+#define ecoff_core_file_matches_executable_p \
+  _bfd_dummy_core_file_matches_executable_p
 #define ecoff_truncate_arname          bfd_dont_truncate_arname
 #define ecoff_openr_next_archived_file bfd_generic_openr_next_archived_file
 #define ecoff_generic_stat_arch_elt    bfd_generic_stat_arch_elt
 #define        ecoff_get_section_contents      bfd_generic_get_section_contents
 #define ecoff_get_reloc_upper_bound    coff_get_reloc_upper_bound
 #define        ecoff_close_and_cleanup         bfd_generic_close_and_cleanup
-#define ecoff_sizeof_headers           coff_sizeof_headers
 #define ecoff_bfd_debug_info_start     bfd_void
 #define ecoff_bfd_debug_info_end       bfd_void
 #define ecoff_bfd_debug_info_accumulate        \
   ((void (*) PARAMS ((bfd *, struct sec *))) bfd_void)
-#define ecoff_bfd_get_relocated_section_contents  bfd_generic_get_relocated_section_contents
+#define ecoff_bfd_get_relocated_section_contents \
+  bfd_generic_get_relocated_section_contents
 #define ecoff_bfd_relax_section                bfd_generic_relax_section
 
 bfd_target ecoff_little_vec =
@@ -2997,10 +4014,11 @@ bfd_target ecoff_little_vec =
   _do_getl64, _do_putl64,      _do_getl32, _do_putl32, _do_getl16, _do_putl16, /* hdrs */
 
   {_bfd_dummy_target, coff_object_p, /* bfd_check_format */
-   bfd_generic_archive_p, _bfd_dummy_target},
-  {bfd_false, ecoff_mkobject, bfd_false, /* bfd_set_format */
-   bfd_false},
-  {bfd_false, ecoff_write_object_contents, bfd_false, bfd_false},
+     ecoff_archive_p, _bfd_dummy_target},
+  {bfd_false, ecoff_mkobject, _bfd_generic_mkarchive, /* bfd_set_format */
+     bfd_false},
+  {bfd_false, ecoff_write_object_contents, /* bfd_write_contents */
+     _bfd_write_archive_contents, bfd_false},
   JUMP_TABLE (ecoff),
   0, 0,
   (PTR) &bfd_ecoff_std_swap_table
@@ -3025,11 +4043,11 @@ bfd_target ecoff_big_vec =
   _do_getb64, _do_putb64,      _do_getb32, _do_putb32, _do_getb16, _do_putb16,
   _do_getb64, _do_putb64,      _do_getb32, _do_putb32, _do_getb16, _do_putb16,
  {_bfd_dummy_target, coff_object_p, /* bfd_check_format */
-   bfd_generic_archive_p, _bfd_dummy_target},
- {bfd_false, ecoff_mkobject, bfd_false, /* bfd_set_format */
-   bfd_false},
+    ecoff_archive_p, _bfd_dummy_target},
+ {bfd_false, ecoff_mkobject, _bfd_generic_mkarchive, /* bfd_set_format */
+    bfd_false},
  {bfd_false, ecoff_write_object_contents, /* bfd_write_contents */
-   bfd_false, bfd_false},
+    _bfd_write_archive_contents, bfd_false},
   JUMP_TABLE(ecoff),
   0, 0,
   (PTR) &bfd_ecoff_std_swap_table