* elf64-ppc.c (CROR_151515, CROR_313131): Define.
[deliverable/binutils-gdb.git] / bfd / elf-m10300.c
index 8f94bbfb73d1d861cc4e42e2fee288579524e58b..ce68a166f814fad4691a60675931bfc05077f359 100644 (file)
@@ -1,5 +1,5 @@
 /* Matsushita 10300 specific support for 32-bit ELF
-   Copyright 1996, 1997, 1998, 1999, 2000, 2001
+   Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2002
    Free Software Foundation, Inc.
 
 This file is part of BFD, the Binary File Descriptor library.
@@ -24,6 +24,24 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
 #include "elf-bfd.h"
 #include "elf/mn10300.h"
 
+static bfd_reloc_status_type mn10300_elf_final_link_relocate
+  PARAMS ((reloc_howto_type *, bfd *, bfd *, asection *, bfd_byte *,
+          bfd_vma, bfd_vma, bfd_vma, struct bfd_link_info *,
+          asection *, int));
+static boolean mn10300_elf_relocate_section
+  PARAMS ((bfd *, struct bfd_link_info *, bfd *, asection *, bfd_byte *,
+          Elf_Internal_Rela *, Elf_Internal_Sym *, asection **));
+static boolean mn10300_elf_relax_section
+  PARAMS ((bfd *, asection *, struct bfd_link_info *, boolean *));
+static bfd_byte * mn10300_elf_get_relocated_section_contents
+  PARAMS ((bfd *, struct bfd_link_info *, struct bfd_link_order *,
+          bfd_byte *, boolean, asymbol **));
+static unsigned long elf_mn10300_mach PARAMS ((flagword));
+
+void    _bfd_mn10300_elf_final_write_processing PARAMS ((bfd *, boolean));
+boolean _bfd_mn10300_elf_object_p PARAMS ((bfd *));
+boolean _bfd_mn10300_elf_merge_private_bfd_data PARAMS ((bfd *,bfd *));
+
 struct elf32_mn10300_link_hash_entry {
   /* The basic elf link hash table entry.  */
   struct elf_link_hash_entry root;
@@ -93,6 +111,8 @@ static struct bfd_hash_entry *elf32_mn10300_link_hash_newfunc
   PARAMS ((struct bfd_hash_entry *, struct bfd_hash_table *, const char *));
 static struct bfd_link_hash_table *elf32_mn10300_link_hash_table_create
   PARAMS ((bfd *));
+static void elf32_mn10300_link_hash_table_free
+  PARAMS ((struct bfd_link_hash_table *));
 
 static reloc_howto_type *bfd_elf32_bfd_reloc_type_lookup
   PARAMS ((bfd *abfd, bfd_reloc_code_real_type code));
@@ -107,7 +127,7 @@ static asection *mn10300_elf_gc_mark_hook
 static boolean mn10300_elf_relax_delete_bytes
   PARAMS ((bfd *, asection *, bfd_vma, int));
 static boolean mn10300_elf_symbol_address_p
-  PARAMS ((bfd *, asection *, Elf32_External_Sym *, bfd_vma));
+  PARAMS ((bfd *, asection *, bfd_vma));
 static boolean elf32_mn10300_finish_hash_table_entry
   PARAMS ((struct bfd_hash_entry *, PTR));
 static void compute_function_info
@@ -409,13 +429,7 @@ mn10300_elf_gc_mark_hook (abfd, info, rel, h, sym)
     }
   else
     {
-      if (!(elf_bad_symtab (abfd)
-           && ELF_ST_BIND (sym->st_info) != STB_LOCAL)
-         && ! ((sym->st_shndx <= 0 || sym->st_shndx >= SHN_LORESERVE)
-               && sym->st_shndx != SHN_COMMON))
-       {
-         return bfd_section_from_elf_index (abfd, sym->st_shndx);
-       }
+      return bfd_section_from_elf_index (abfd, sym->st_shndx);
     }
 
   return NULL;
@@ -592,9 +606,7 @@ mn10300_elf_relocate_section (output_bfd, info, input_bfd, input_section,
        {
          sym = local_syms + r_symndx;
          sec = local_sections[r_symndx];
-         relocation = (sec->output_section->vma
-                       + sec->output_offset
-                       + sym->st_value);
+         relocation = _bfd_elf_rela_local_sym (output_bfd, sym, sec, rel);
        }
       else
        {
@@ -699,6 +711,9 @@ elf32_mn10300_finish_hash_table_entry (gen_entry, in_args)
 
   entry = (struct elf32_mn10300_link_hash_entry *) gen_entry;
 
+  if (entry->root.root.type == bfd_link_hash_warning)
+    entry = (struct elf32_mn10300_link_hash_entry *) entry->root.root.u.i.link;
+
   /* If we already know we want to convert "call" to "calls" for calls
      to this symbol, then return now.  */
   if (entry->flags == MN10300_CONVERT_CALL_TO_CALLS)
@@ -787,6 +802,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
      boolean *again;
 {
   Elf_Internal_Shdr *symtab_hdr;
+  Elf_Internal_Shdr *shndx_hdr;
   Elf_Internal_Rela *internal_relocs = NULL;
   Elf_Internal_Rela *free_relocs = NULL;
   Elf_Internal_Rela *irel, *irelend;
@@ -794,6 +810,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
   bfd_byte *free_contents = NULL;
   Elf32_External_Sym *extsyms = NULL;
   Elf32_External_Sym *free_extsyms = NULL;
+  Elf_External_Sym_Shndx *shndx_buf = NULL;
   struct elf32_mn10300_link_hash_table *hash_table;
 
   /* Assume nothing changes.  */
@@ -816,6 +833,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
 
          /* We're going to need all the symbols for each bfd.  */
          symtab_hdr = &elf_tdata (input_bfd)->symtab_hdr;
+         shndx_hdr = &elf_tdata (input_bfd)->symtab_shndx_hdr;
 
          /* Get cached copy if it exists.  */
          if (symtab_hdr->contents != NULL)
@@ -823,14 +841,30 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
          else
            {
              /* Go get them off disk.  */
-             extsyms = ((Elf32_External_Sym *)
-                        bfd_malloc (symtab_hdr->sh_size));
+             bfd_size_type amt;
+
+             amt = symtab_hdr->sh_info;
+             amt *= sizeof (Elf32_External_Sym);
+             extsyms = (Elf32_External_Sym *) bfd_malloc (amt);
              if (extsyms == NULL)
                goto error_return;
              free_extsyms = extsyms;
              if (bfd_seek (input_bfd, symtab_hdr->sh_offset, SEEK_SET) != 0
-                 || (bfd_read (extsyms, 1, symtab_hdr->sh_size, input_bfd)
-                     != symtab_hdr->sh_size))
+                 || bfd_bread ((PTR) extsyms, amt, input_bfd) != amt)
+               goto error_return;
+           }
+
+         if (shndx_hdr->sh_size != 0)
+           {
+             bfd_size_type amt;
+
+             amt = symtab_hdr->sh_info;
+             amt *= sizeof (Elf_External_Sym_Shndx);
+             shndx_buf = (Elf_External_Sym_Shndx *) bfd_malloc (amt);
+             if (shndx_buf == NULL)
+               goto error_return;
+             if (bfd_seek (input_bfd, shndx_hdr->sh_offset, SEEK_SET) != 0
+                 || bfd_bread ((PTR) shndx_buf, amt, input_bfd) != amt)
                goto error_return;
            }
 
@@ -845,6 +879,12 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
              const char *sym_name;
              char *new_name;
 
+             /* If there's nothing to do in this section, skip it.  */
+             if (! (((section->flags & SEC_RELOC) != 0
+                     && section->reloc_count != 0)
+                    || (section->flags & SEC_CODE) != 0))
+               continue;
+
              /* Get cached copy of section contents if it exists.  */
              if (elf_section_data (section)->this_hdr.contents != NULL)
                contents = elf_section_data (section)->this_hdr.contents;
@@ -906,26 +946,33 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                      if (r_index < symtab_hdr->sh_info)
                        {
                          /* A local symbol.  */
+                         Elf32_External_Sym *esym;
+                         Elf_External_Sym_Shndx *shndx;
                          Elf_Internal_Sym isym;
+                         struct elf_link_hash_table *elftab;
+                         bfd_size_type amt;
 
-                         bfd_elf32_swap_symbol_in (input_bfd,
-                                                   extsyms + r_index, &isym);
+                         esym = extsyms + r_index;
+                         shndx = shndx_buf + (shndx_buf ? r_index : 0);
+                         bfd_elf32_swap_symbol_in (input_bfd, esym, shndx,
+                                                   &isym);
 
                          if (isym.st_shndx == SHN_UNDEF)
                            sym_sec = bfd_und_section_ptr;
-                         else if (isym.st_shndx > 0
-                                  && isym.st_shndx < SHN_LORESERVE)
-                           sym_sec
-                             = bfd_section_from_elf_index (input_bfd,
-                                                           isym.st_shndx);
                          else if (isym.st_shndx == SHN_ABS)
                            sym_sec = bfd_abs_section_ptr;
                          else if (isym.st_shndx == SHN_COMMON)
                            sym_sec = bfd_com_section_ptr;
+                         else
+                           sym_sec
+                             = bfd_section_from_elf_index (input_bfd,
+                                                           isym.st_shndx);
 
-                         sym_name = bfd_elf_string_from_elf_section (input_bfd,
-                                                          symtab_hdr->sh_link,
-                                                          isym.st_name);
+                         sym_name
+                           = bfd_elf_string_from_elf_section (input_bfd,
+                                                              (symtab_hdr
+                                                               ->sh_link),
+                                                              isym.st_name);
 
                          /* If it isn't a function, then we don't care
                             about it.  */
@@ -935,7 +982,8 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
 
                          /* Tack on an ID so we can uniquely identify this
                             local symbol in the global hash table.  */
-                         new_name = bfd_malloc (strlen (sym_name) + 10);
+                         amt = strlen (sym_name) + 10;
+                         new_name = bfd_malloc (amt);
                          if (new_name == 0)
                            goto error_return;
 
@@ -943,10 +991,10 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                                   sym_name, (int) sym_sec);
                          sym_name = new_name;
 
-                         hash = (struct elf32_mn10300_link_hash_entry *)
-                                  elf_link_hash_lookup (&hash_table->static_hash_table->root,
-                                                        sym_name, true,
-                                                        true, false);
+                         elftab = &hash_table->static_hash_table->root;
+                         hash = ((struct elf32_mn10300_link_hash_entry *)
+                                 elf_link_hash_lookup (elftab, sym_name,
+                                                       true, true, false));
                          free (new_name);
                        }
                      else
@@ -981,42 +1029,48 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                {
 
                  Elf32_External_Sym *esym, *esymend;
-                 int idx, shndx;
+                 Elf_External_Sym_Shndx *shndx;
+                 int idx;
+                 unsigned int sec_shndx;
 
-                 shndx = _bfd_elf_section_from_bfd_section (input_bfd,
-                                                            section);
+                 sec_shndx = _bfd_elf_section_from_bfd_section (input_bfd,
+                                                                section);
 
                  /* Look at each function defined in this section and
                     update info for that function.  */
-                 esym = extsyms;
-                 esymend = esym + symtab_hdr->sh_info;
-                 for (; esym < esymend; esym++)
+                 for (esym = extsyms, esymend = esym + symtab_hdr->sh_info,
+                        shndx = shndx_buf;
+                      esym < esymend;
+                      esym++, shndx = (shndx ? shndx + 1 : NULL))
                    {
                      Elf_Internal_Sym isym;
 
-                     bfd_elf32_swap_symbol_in (input_bfd, esym, &isym);
-                     if (isym.st_shndx == shndx
+                     bfd_elf32_swap_symbol_in (input_bfd, esym, shndx, &isym);
+                     if (isym.st_shndx == sec_shndx
                          && ELF_ST_TYPE (isym.st_info) == STT_FUNC)
                        {
+                         struct elf_link_hash_table *elftab;
+                         bfd_size_type amt;
+
                          if (isym.st_shndx == SHN_UNDEF)
                            sym_sec = bfd_und_section_ptr;
-                         else if (isym.st_shndx > 0
-                                  && isym.st_shndx < SHN_LORESERVE)
-                           sym_sec
-                             = bfd_section_from_elf_index (input_bfd,
-                                                           isym.st_shndx);
                          else if (isym.st_shndx == SHN_ABS)
                            sym_sec = bfd_abs_section_ptr;
                          else if (isym.st_shndx == SHN_COMMON)
                            sym_sec = bfd_com_section_ptr;
+                         else
+                           sym_sec
+                             = bfd_section_from_elf_index (input_bfd,
+                                                           isym.st_shndx);
 
-                         sym_name = bfd_elf_string_from_elf_section (input_bfd,
-                                                       symtab_hdr->sh_link,
-                                                       isym.st_name);
+                         sym_name = (bfd_elf_string_from_elf_section
+                                     (input_bfd, symtab_hdr->sh_link,
+                                      isym.st_name));
 
                          /* Tack on an ID so we can uniquely identify this
                             local symbol in the global hash table.  */
-                         new_name = bfd_malloc (strlen (sym_name) + 10);
+                         amt = strlen (sym_name) + 10;
+                         new_name = bfd_malloc (amt);
                          if (new_name == 0)
                            goto error_return;
 
@@ -1024,10 +1078,10 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                                   sym_name, (int) sym_sec);
                          sym_name = new_name;
 
-                         hash = (struct elf32_mn10300_link_hash_entry *)
-                                   elf_link_hash_lookup (&hash_table->static_hash_table->root,
-                                                         sym_name, true,
-                                                         true, false);
+                         elftab = &hash_table->static_hash_table->root;
+                         hash = ((struct elf32_mn10300_link_hash_entry *)
+                                 elf_link_hash_lookup (elftab, sym_name,
+                                                       true, true, false));
                          free (new_name);
                          compute_function_info (input_bfd, hash,
                                                 isym.st_value, contents);
@@ -1041,14 +1095,12 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    {
                      Elf_Internal_Sym isym;
 
-                     bfd_elf32_swap_symbol_in (input_bfd, esym, &isym);
                      hash = (struct elf32_mn10300_link_hash_entry *)
                               elf_sym_hashes (input_bfd)[idx];
-                     if (isym.st_shndx == shndx
-                         && ELF_ST_TYPE (isym.st_info) == STT_FUNC
-                         && (hash)->root.root.u.def.section == section
-                         && ((hash)->root.root.type == bfd_link_hash_defined
-                             || (hash)->root.root.type == bfd_link_hash_defweak))
+                     if ((hash->root.root.type == bfd_link_hash_defined
+                          || hash->root.root.type == bfd_link_hash_defweak)
+                         && hash->root.root.u.def.section == section
+                         && ELF_ST_TYPE (isym.st_info) == STT_FUNC)
                        compute_function_info (input_bfd, hash,
                                               (hash)->root.root.u.def.value,
                                               contents);
@@ -1076,6 +1128,12 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                }
            }
 
+         if (shndx_buf != NULL)
+           {
+             free (shndx_buf);
+             shndx_buf = NULL;
+           }
+
          /* Cache or free any memory we allocated for the symbols.  */
          if (free_extsyms != NULL)
            {
@@ -1084,7 +1142,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
              else
                {
                  /* Cache the symbols for elf_link_input_bfd.  */
-                 symtab_hdr->contents = extsyms;
+                 symtab_hdr->contents = (unsigned char *) extsyms;
                }
              free_extsyms = NULL;
            }
@@ -1112,8 +1170,9 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
        {
          asection *section;
 
-         /* We're going to need all the symbols for each bfd.  */
+         /* We're going to need all the local symbols for each bfd.  */
          symtab_hdr = &elf_tdata (input_bfd)->symtab_hdr;
+         shndx_hdr = &elf_tdata (input_bfd)->symtab_shndx_hdr;
 
          /* Get cached copy if it exists.  */
          if (symtab_hdr->contents != NULL)
@@ -1121,15 +1180,33 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
          else
            {
              /* Go get them off disk.  */
-             extsyms = ((Elf32_External_Sym *)
-                        bfd_malloc (symtab_hdr->sh_size));
+             bfd_size_type amt;
+
+             amt = symtab_hdr->sh_info;
+             amt *= sizeof (Elf32_External_Sym);
+             extsyms = (Elf32_External_Sym *) bfd_malloc (amt);
              if (extsyms == NULL)
                goto error_return;
              free_extsyms = extsyms;
              if (bfd_seek (input_bfd, symtab_hdr->sh_offset, SEEK_SET) != 0
-                 || (bfd_read (extsyms, 1, symtab_hdr->sh_size, input_bfd)
-                     != symtab_hdr->sh_size))
+                 || bfd_bread ((PTR) extsyms, amt, input_bfd) != amt)
+               goto error_return;
+             symtab_hdr->contents = (bfd_byte *) extsyms;
+           }
+
+         if (shndx_hdr->sh_size != 0)
+           {
+             bfd_size_type amt;
+
+             amt = symtab_hdr->sh_info;
+             amt *= sizeof (Elf_External_Sym_Shndx);
+             shndx_buf = (Elf_External_Sym_Shndx *) bfd_malloc (amt);
+             if (shndx_buf == NULL)
                goto error_return;
+             if (bfd_seek (input_bfd, shndx_hdr->sh_offset, SEEK_SET) != 0
+                 || bfd_bread ((PTR) shndx_buf, amt, input_bfd) != amt)
+               goto error_return;
+             shndx_hdr->contents = (bfd_byte *) shndx_buf;
            }
 
          /* Walk over each section in this bfd.  */
@@ -1137,9 +1214,10 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
               section != NULL;
               section = section->next)
            {
-             int shndx;
+             unsigned int sec_shndx;
              Elf32_External_Sym *esym, *esymend;
-             int idx;
+             Elf_External_Sym_Shndx *shndx;
+             unsigned int idx;
 
              /* Skip non-code sections and empty sections.  */
              if ((section->flags & SEC_CODE) == 0 || section->_raw_size == 0)
@@ -1175,60 +1253,64 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    goto error_return;
                }
 
-             shndx = _bfd_elf_section_from_bfd_section (input_bfd, section);
+             sec_shndx = _bfd_elf_section_from_bfd_section (input_bfd,
+                                                            section);
 
              /* Now look for any function in this section which needs
                 insns deleted from its prologue.  */
-             esym = extsyms;
-             esymend = esym + symtab_hdr->sh_info;
-             for (; esym < esymend; esym++)
+             for (esym = extsyms, esymend = esym + symtab_hdr->sh_info,
+                    shndx = shndx_buf;
+                  esym < esymend;
+                  esym++, shndx = (shndx ? shndx + 1 : NULL))
                {
                  Elf_Internal_Sym isym;
                  struct elf32_mn10300_link_hash_entry *sym_hash;
                  asection *sym_sec = NULL;
                  const char *sym_name;
                  char *new_name;
+                 struct elf_link_hash_table *elftab;
+                 bfd_size_type amt;
 
-                 bfd_elf32_swap_symbol_in (input_bfd, esym, &isym);
+                 bfd_elf32_swap_symbol_in (input_bfd, esym, shndx, &isym);
 
-                 if (isym.st_shndx != shndx)
+                 if (isym.st_shndx != sec_shndx)
                    continue;
 
                  if (isym.st_shndx == SHN_UNDEF)
                    sym_sec = bfd_und_section_ptr;
-                 else if (isym.st_shndx > 0 && isym.st_shndx < SHN_LORESERVE)
-                   sym_sec
-                     = bfd_section_from_elf_index (input_bfd, isym.st_shndx);
                  else if (isym.st_shndx == SHN_ABS)
                    sym_sec = bfd_abs_section_ptr;
                  else if (isym.st_shndx == SHN_COMMON)
                    sym_sec = bfd_com_section_ptr;
                  else
-                   abort ();
+                   sym_sec
+                     = bfd_section_from_elf_index (input_bfd, isym.st_shndx);
 
-                 sym_name = bfd_elf_string_from_elf_section (input_bfd,
-                                                       symtab_hdr->sh_link,
-                                                       isym.st_name);
+                 sym_name
+                   = bfd_elf_string_from_elf_section (input_bfd,
+                                                      symtab_hdr->sh_link,
+                                                      isym.st_name);
 
                  /* Tack on an ID so we can uniquely identify this
                     local symbol in the global hash table.  */
-                 new_name = bfd_malloc (strlen (sym_name) + 10);
+                 amt = strlen (sym_name) + 10;
+                 new_name = bfd_malloc (amt);
                  if (new_name == 0)
                    goto error_return;
                  sprintf (new_name, "%s_%08x", sym_name, (int) sym_sec);
                  sym_name = new_name;
 
-                 sym_hash = (struct elf32_mn10300_link_hash_entry *)
-                           elf_link_hash_lookup (&hash_table->static_hash_table->root,
-                                                 sym_name, false,
-                                                 false, false);
+                 elftab = &hash_table->static_hash_table->root;
+                 sym_hash = ((struct elf32_mn10300_link_hash_entry *)
+                             elf_link_hash_lookup (elftab, sym_name,
+                                                   false, false, false));
 
                  free (new_name);
                  if (sym_hash == NULL)
                    continue;
 
-                 if (! ((sym_hash)->flags & MN10300_CONVERT_CALL_TO_CALLS)
-                     && ! ((sym_hash)->flags & MN10300_DELETED_PROLOGUE_BYTES))
+                 if (! (sym_hash->flags & MN10300_CONVERT_CALL_TO_CALLS)
+                     && ! (sym_hash->flags & MN10300_DELETED_PROLOGUE_BYTES))
                    {
                      int bytes = 0;
 
@@ -1239,7 +1321,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                      elf_section_data (section)->this_hdr.contents = contents;
                      free_contents = NULL;
 
-                     symtab_hdr->contents = (bfd_byte *) extsyms;
                      free_extsyms = NULL;
 
                      /* Count how many bytes we're going to delete.  */
@@ -1271,23 +1352,23 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
 
              /* Look for any global functions in this section which
                 need insns deleted from their prologues.  */
-             esym = extsyms + symtab_hdr->sh_info;
-             esymend = extsyms + (symtab_hdr->sh_size
-                                  / sizeof (Elf32_External_Sym));
-             for (idx = 0; esym < esymend; esym++, idx++)
+             for (idx = 0;
+                  idx < (symtab_hdr->sh_size / sizeof (Elf32_External_Sym)
+                         - symtab_hdr->sh_info);
+                  idx++)
                {
-                 Elf_Internal_Sym isym;
                  struct elf32_mn10300_link_hash_entry *sym_hash;
 
-                 bfd_elf32_swap_symbol_in (input_bfd, esym, &isym);
                  sym_hash = (struct elf32_mn10300_link_hash_entry *)
                               (elf_sym_hashes (input_bfd)[idx]);
-                 if (isym.st_shndx == shndx
-                     && (sym_hash)->root.root.u.def.section == section
-                     && ! ((sym_hash)->flags & MN10300_CONVERT_CALL_TO_CALLS)
-                     && ! ((sym_hash)->flags & MN10300_DELETED_PROLOGUE_BYTES))
+                 if ((sym_hash->root.root.type == bfd_link_hash_defined
+                      || sym_hash->root.root.type == bfd_link_hash_defweak)
+                     && sym_hash->root.root.u.def.section == section
+                     && ! (sym_hash->flags & MN10300_CONVERT_CALL_TO_CALLS)
+                     && ! (sym_hash->flags & MN10300_DELETED_PROLOGUE_BYTES))
                    {
                      int bytes = 0;
+                     bfd_vma symval;
 
                      /* Note that we've changed things.  */
                      elf_section_data (section)->relocs = internal_relocs;
@@ -1296,7 +1377,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                      elf_section_data (section)->this_hdr.contents = contents;
                      free_contents = NULL;
 
-                     symtab_hdr->contents = (bfd_byte *) extsyms;
                      free_extsyms = NULL;
 
                      /* Count how many bytes we're going to delete.  */
@@ -1314,9 +1394,10 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                      sym_hash->flags |= MN10300_DELETED_PROLOGUE_BYTES;
 
                      /* Actually delete the bytes.  */
+                     symval = sym_hash->root.root.u.def.value;
                      if (!mn10300_elf_relax_delete_bytes (input_bfd,
                                                           section,
-                                                          (sym_hash)->root.root.u.def.value,
+                                                          symval,
                                                           bytes))
                        goto error_return;
 
@@ -1347,15 +1428,20 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                }
            }
 
+         if (shndx_buf != NULL)
+           {
+             shndx_hdr->contents = NULL;
+             free (shndx_buf);
+             shndx_buf = NULL;
+           }
+
          /* Cache or free any memory we allocated for the symbols.  */
          if (free_extsyms != NULL)
            {
              if (! link_info->keep_memory)
-               free (free_extsyms);
-             else
                {
-                 /* Cache the symbols for elf_link_input_bfd.  */
-                 symtab_hdr->contents = extsyms;
+                 symtab_hdr->contents = NULL;
+                 free (free_extsyms);
                }
              free_extsyms = NULL;
            }
@@ -1385,6 +1471,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
     sec->_cooked_size = sec->_raw_size;
 
   symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
+  shndx_hdr = &elf_tdata (abfd)->symtab_shndx_hdr;
 
   /* Get a copy of the native relocations.  */
   internal_relocs = (_bfd_elf32_link_read_relocs
@@ -1438,41 +1525,59 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
          else
            {
              /* Go get them off disk.  */
-             extsyms = ((Elf32_External_Sym *)
-                        bfd_malloc (symtab_hdr->sh_size));
+             bfd_size_type amt;
+
+             amt = symtab_hdr->sh_info;
+             amt *= sizeof (Elf32_External_Sym);
+             extsyms = (Elf32_External_Sym *) bfd_malloc (amt);
              if (extsyms == NULL)
                goto error_return;
              free_extsyms = extsyms;
              if (bfd_seek (abfd, symtab_hdr->sh_offset, SEEK_SET) != 0
-                 || (bfd_read (extsyms, 1, symtab_hdr->sh_size, abfd)
-                     != symtab_hdr->sh_size))
+                 || bfd_bread ((PTR) extsyms, amt, abfd) != amt)
                goto error_return;
+             symtab_hdr->contents = (bfd_byte *) extsyms;
+           }
+
+         if (shndx_hdr->sh_size != 0)
+           {
+             bfd_size_type amt;
+
+             amt = symtab_hdr->sh_info;
+             amt *= sizeof (Elf_External_Sym_Shndx);
+             shndx_buf = (Elf_External_Sym_Shndx *) bfd_malloc (amt);
+             if (shndx_buf == NULL)
+               goto error_return;
+             if (bfd_seek (abfd, shndx_hdr->sh_offset, SEEK_SET) != 0
+                 || bfd_bread ((PTR) shndx_buf, amt, abfd) != amt)
+               goto error_return;
+             shndx_hdr->contents = (bfd_byte *) shndx_buf;
            }
        }
 
       /* Get the value of the symbol referred to by the reloc.  */
       if (ELF32_R_SYM (irel->r_info) < symtab_hdr->sh_info)
        {
+         Elf32_External_Sym *esym;
+         Elf_External_Sym_Shndx *shndx;
          Elf_Internal_Sym isym;
          asection *sym_sec = NULL;
          const char *sym_name;
          char *new_name;
 
          /* A local symbol.  */
-         bfd_elf32_swap_symbol_in (abfd,
-                                   extsyms + ELF32_R_SYM (irel->r_info),
-                                   &isym);
+         esym = extsyms + ELF32_R_SYM (irel->r_info);
+         shndx = shndx_buf + (shndx_buf ? ELF32_R_SYM (irel->r_info) : 0);
+         bfd_elf32_swap_symbol_in (abfd, esym, shndx, &isym);
 
          if (isym.st_shndx == SHN_UNDEF)
            sym_sec = bfd_und_section_ptr;
-         else if (isym.st_shndx > 0 && isym.st_shndx < SHN_LORESERVE)
-           sym_sec = bfd_section_from_elf_index (abfd, isym.st_shndx);
          else if (isym.st_shndx == SHN_ABS)
            sym_sec = bfd_abs_section_ptr;
          else if (isym.st_shndx == SHN_COMMON)
            sym_sec = bfd_com_section_ptr;
          else
-           abort ();
+           sym_sec = bfd_section_from_elf_index (abfd, isym.st_shndx);
 
          symval = (isym.st_value
                    + sym_sec->output_section->vma
@@ -1483,7 +1588,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
 
          /* Tack on an ID so we can uniquely identify this
             local symbol in the global hash table.  */
-         new_name = bfd_malloc (strlen (sym_name) + 10);
+         new_name = bfd_malloc ((bfd_size_type) strlen (sym_name) + 10);
          if (new_name == 0)
            goto error_return;
          sprintf (new_name, "%s_%08x", sym_name, (int) sym_sec);
@@ -1552,7 +1657,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                  elf_section_data (sec)->this_hdr.contents = contents;
                  free_contents = NULL;
 
-                 symtab_hdr->contents = (bfd_byte *) extsyms;
                  free_extsyms = NULL;
 
                  /* Fix the opcode.  */
@@ -1617,7 +1721,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
              elf_section_data (sec)->this_hdr.contents = contents;
              free_contents = NULL;
 
-             symtab_hdr->contents = (bfd_byte *) extsyms;
              free_extsyms = NULL;
 
              /* Fix the opcode.  */
@@ -1669,7 +1772,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                  elf_section_data (sec)->this_hdr.contents = contents;
                  free_contents = NULL;
 
-                 symtab_hdr->contents = (bfd_byte *) extsyms;
                  free_extsyms = NULL;
 
                  /* Fix the opcode.  */
@@ -1732,7 +1834,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
              elf_section_data (sec)->this_hdr.contents = contents;
              free_contents = NULL;
 
-             symtab_hdr->contents = (bfd_byte *) extsyms;
              free_extsyms = NULL;
 
              /* Fix the opcode.  */
@@ -1817,8 +1918,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
 
          /* We also have to be sure there is no symbol/label
             at the unconditional branch.  */
-         if (mn10300_elf_symbol_address_p (abfd, sec, extsyms,
-                                           irel->r_offset + 1))
+         if (mn10300_elf_symbol_address_p (abfd, sec, irel->r_offset + 1))
            continue;
 
          /* Note that we've changed the relocs, section contents, etc.  */
@@ -1828,7 +1928,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
          elf_section_data (sec)->this_hdr.contents = contents;
          free_contents = NULL;
 
-         symtab_hdr->contents = (bfd_byte *) extsyms;
          free_extsyms = NULL;
 
          /* Reverse the condition of the first branch.  */
@@ -1942,7 +2041,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                          elf_section_data (sec)->this_hdr.contents = contents;
                          free_contents = NULL;
 
-                         symtab_hdr->contents = (bfd_byte *) extsyms;
                          free_extsyms = NULL;
 
                          /* Fix the opcode.  */
@@ -2018,7 +2116,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                          elf_section_data (sec)->this_hdr.contents = contents;
                          free_contents = NULL;
 
-                         symtab_hdr->contents = (bfd_byte *) extsyms;
                          free_extsyms = NULL;
 
                          /* Fix the opcode.  */
@@ -2102,7 +2199,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    elf_section_data (sec)->this_hdr.contents = contents;
                    free_contents = NULL;
 
-                   symtab_hdr->contents = (bfd_byte *) extsyms;
                    free_extsyms = NULL;
 
                    /* Fix the opcode.  */
@@ -2140,7 +2236,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    elf_section_data (sec)->this_hdr.contents = contents;
                    free_contents = NULL;
 
-                   symtab_hdr->contents = (bfd_byte *) extsyms;
                    free_extsyms = NULL;
 
                    if ((code & 0xf3) == 0x81)
@@ -2195,7 +2290,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    elf_section_data (sec)->this_hdr.contents = contents;
                    free_contents = NULL;
 
-                   symtab_hdr->contents = (bfd_byte *) extsyms;
                    free_extsyms = NULL;
 
                    /* Fix the opcode.  */
@@ -2248,7 +2342,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    elf_section_data (sec)->this_hdr.contents = contents;
                    free_contents = NULL;
 
-                   symtab_hdr->contents = (bfd_byte *) extsyms;
                    free_extsyms = NULL;
 
                    if ((code & 0xfc) == 0xcc)
@@ -2330,7 +2423,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                    elf_section_data (sec)->this_hdr.contents = contents;
                    free_contents = NULL;
 
-                   symtab_hdr->contents = (bfd_byte *) extsyms;
                    free_extsyms = NULL;
 
                    /* Fix the opcode.  */
@@ -2362,7 +2454,6 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
                  elf_section_data (sec)->this_hdr.contents = contents;
                  free_contents = NULL;
 
-                 symtab_hdr->contents = (bfd_byte *) extsyms;
                  free_extsyms = NULL;
 
                  /* Fix the opcode.  */
@@ -2388,10 +2479,7 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
     }
 
   if (free_relocs != NULL)
-    {
-      free (free_relocs);
-      free_relocs = NULL;
-    }
+    free (free_relocs);
 
   if (free_contents != NULL)
     {
@@ -2402,19 +2490,21 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
          /* Cache the section contents for elf_link_input_bfd.  */
          elf_section_data (sec)->this_hdr.contents = contents;
        }
-      free_contents = NULL;
+    }
+
+  if (shndx_buf != NULL)
+    {
+      shndx_hdr->contents = NULL;
+      free (shndx_buf);
     }
 
   if (free_extsyms != NULL)
     {
       if (! link_info->keep_memory)
-       free (free_extsyms);
-      else
        {
-         /* Cache the symbols for elf_link_input_bfd.  */
-         symtab_hdr->contents = extsyms;
+         symtab_hdr->contents = NULL;
+         free (free_extsyms);
        }
-      free_extsyms = NULL;
     }
 
   return true;
@@ -2424,8 +2514,17 @@ mn10300_elf_relax_section (abfd, sec, link_info, again)
     free (free_relocs);
   if (free_contents != NULL)
     free (free_contents);
+  if (shndx_buf != NULL)
+    {
+      shndx_hdr->contents = NULL;
+      free (shndx_buf);
+    }
   if (free_extsyms != NULL)
-    free (free_extsyms);
+    {
+      symtab_hdr->contents = NULL;
+      free (free_extsyms);
+    }
+
   return false;
 }
 
@@ -2539,19 +2638,23 @@ mn10300_elf_relax_delete_bytes (abfd, sec, addr, count)
      int count;
 {
   Elf_Internal_Shdr *symtab_hdr;
+  Elf_Internal_Shdr *shndx_hdr;
   Elf32_External_Sym *extsyms;
-  int shndx, index;
+  unsigned int sec_shndx;
   bfd_byte *contents;
   Elf_Internal_Rela *irel, *irelend;
   Elf_Internal_Rela *irelalign;
   bfd_vma toaddr;
   Elf32_External_Sym *esym, *esymend;
-  struct elf32_mn10300_link_hash_entry *sym_hash;
+  Elf_External_Sym_Shndx *shndx;
+  struct elf_link_hash_entry **sym_hashes;
+  struct elf_link_hash_entry **end_hashes;
+  unsigned int symcount;
 
   symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
   extsyms = (Elf32_External_Sym *) symtab_hdr->contents;
 
-  shndx = _bfd_elf_section_from_bfd_section (abfd, sec);
+  sec_shndx = _bfd_elf_section_from_bfd_section (abfd, sec);
 
   contents = elf_section_data (sec)->this_hdr.contents;
 
@@ -2565,7 +2668,8 @@ mn10300_elf_relax_delete_bytes (abfd, sec, addr, count)
   irelend = irel + sec->reloc_count;
 
   /* Actually delete the bytes.  */
-  memmove (contents + addr, contents + addr + count, toaddr - addr - count);
+  memmove (contents + addr, contents + addr + count,
+          (size_t) (toaddr - addr - count));
   sec->_cooked_size -= count;
 
   /* Adjust all the relocs.  */
@@ -2578,41 +2682,41 @@ mn10300_elf_relax_delete_bytes (abfd, sec, addr, count)
     }
 
   /* Adjust the local symbols defined in this section.  */
+  shndx_hdr = &elf_tdata (abfd)->symtab_shndx_hdr;
+  shndx = (Elf_External_Sym_Shndx *) shndx_hdr->contents;
   esym = extsyms;
   esymend = esym + symtab_hdr->sh_info;
-  for (; esym < esymend; esym++)
+  for (; esym < esymend; esym++, shndx = (shndx ? shndx + 1 : NULL))
     {
       Elf_Internal_Sym isym;
+      Elf_External_Sym_Shndx dummy;
 
-      bfd_elf32_swap_symbol_in (abfd, esym, &isym);
+      bfd_elf32_swap_symbol_in (abfd, esym, shndx, &isym);
 
-      if (isym.st_shndx == shndx
+      if (isym.st_shndx == sec_shndx
          && isym.st_value > addr
          && isym.st_value < toaddr)
        {
          isym.st_value -= count;
-         bfd_elf32_swap_symbol_out (abfd, &isym, esym);
+         bfd_elf32_swap_symbol_out (abfd, &isym, (PTR) esym, (PTR) &dummy);
        }
     }
 
   /* Now adjust the global symbols defined in this section.  */
-  esym = extsyms + symtab_hdr->sh_info;
-  esymend = extsyms + (symtab_hdr->sh_size / sizeof (Elf32_External_Sym));
-  for (index = 0; esym < esymend; esym++, index++)
+  symcount = (symtab_hdr->sh_size / sizeof (Elf32_External_Sym)
+             - symtab_hdr->sh_info);
+  sym_hashes = elf_sym_hashes (abfd);
+  end_hashes = sym_hashes + symcount;
+  for (; sym_hashes < end_hashes; sym_hashes++)
     {
-      Elf_Internal_Sym isym;
-
-      bfd_elf32_swap_symbol_in (abfd, esym, &isym);
-      sym_hash = (struct elf32_mn10300_link_hash_entry *)
-                  (elf_sym_hashes (abfd)[index]);
-      if (isym.st_shndx == shndx
-         && ((sym_hash)->root.root.type == bfd_link_hash_defined
-             || (sym_hash)->root.root.type == bfd_link_hash_defweak)
-         && (sym_hash)->root.root.u.def.section == sec
-         && (sym_hash)->root.root.u.def.value > addr
-         && (sym_hash)->root.root.u.def.value < toaddr)
+      struct elf_link_hash_entry *sym_hash = *sym_hashes;
+      if ((sym_hash->root.type == bfd_link_hash_defined
+          || sym_hash->root.type == bfd_link_hash_defweak)
+         && sym_hash->root.u.def.section == sec
+         && sym_hash->root.u.def.value > addr
+         && sym_hash->root.u.def.value < toaddr)
        {
-         (sym_hash)->root.root.u.def.value -= count;
+         sym_hash->root.u.def.value -= count;
        }
     }
 
@@ -2622,46 +2726,53 @@ mn10300_elf_relax_delete_bytes (abfd, sec, addr, count)
 /* Return true if a symbol exists at the given address, else return
    false.  */
 static boolean
-mn10300_elf_symbol_address_p (abfd, sec, extsyms, addr)
+mn10300_elf_symbol_address_p (abfd, sec, addr)
      bfd *abfd;
      asection *sec;
-     Elf32_External_Sym *extsyms;
      bfd_vma addr;
 {
   Elf_Internal_Shdr *symtab_hdr;
-  int shndx;
+  Elf_Internal_Shdr *shndx_hdr;
+  unsigned int sec_shndx;
   Elf32_External_Sym *esym, *esymend;
-  struct elf32_mn10300_link_hash_entry **sym_hash, **sym_hash_end;
+  Elf_External_Sym_Shndx *shndx;
+  struct elf_link_hash_entry **sym_hashes;
+  struct elf_link_hash_entry **end_hashes;
+  unsigned int symcount;
 
-  symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
-  shndx = _bfd_elf_section_from_bfd_section (abfd, sec);
+  sec_shndx = _bfd_elf_section_from_bfd_section (abfd, sec);
 
   /* Examine all the symbols.  */
-  esym = extsyms;
+  symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
+  shndx_hdr = &elf_tdata (abfd)->symtab_shndx_hdr;
+  shndx = (Elf_External_Sym_Shndx *) shndx_hdr->contents;
+  esym = (Elf32_External_Sym *) symtab_hdr->contents;
   esymend = esym + symtab_hdr->sh_info;
-  for (; esym < esymend; esym++)
+  for (; esym < esymend; esym++, shndx = (shndx ? shndx + 1 : NULL))
     {
       Elf_Internal_Sym isym;
 
-      bfd_elf32_swap_symbol_in (abfd, esym, &isym);
+      bfd_elf32_swap_symbol_in (abfd, esym, shndx, &isym);
 
-      if (isym.st_shndx == shndx
+      if (isym.st_shndx == sec_shndx
          && isym.st_value == addr)
        return true;
     }
 
-  sym_hash = (struct elf32_mn10300_link_hash_entry **) (elf_sym_hashes (abfd));
-  sym_hash_end = (sym_hash
-                 + (symtab_hdr->sh_size / sizeof (Elf32_External_Sym)
-                    - symtab_hdr->sh_info));
-  for (; sym_hash < sym_hash_end; sym_hash++)
+  symcount = (symtab_hdr->sh_size / sizeof (Elf32_External_Sym)
+             - symtab_hdr->sh_info);
+  sym_hashes = elf_sym_hashes (abfd);
+  end_hashes = sym_hashes + symcount;
+  for (; sym_hashes < end_hashes; sym_hashes++)
     {
-      if (((*sym_hash)->root.root.type == bfd_link_hash_defined
-          || (*sym_hash)->root.root.type == bfd_link_hash_defweak)
-         && (*sym_hash)->root.root.u.def.section == sec
-         && (*sym_hash)->root.root.u.def.value == addr)
+      struct elf_link_hash_entry *sym_hash = *sym_hashes;
+      if ((sym_hash->root.type == bfd_link_hash_defined
+          || sym_hash->root.type == bfd_link_hash_defweak)
+         && sym_hash->root.u.def.section == sec
+         && sym_hash->root.u.def.value == addr)
        return true;
     }
+
   return false;
 }
 
@@ -2679,11 +2790,14 @@ mn10300_elf_get_relocated_section_contents (output_bfd, link_info, link_order,
      asymbol **symbols;
 {
   Elf_Internal_Shdr *symtab_hdr;
+  Elf_Internal_Shdr *shndx_hdr;
   asection *input_section = link_order->u.indirect.section;
   bfd *input_bfd = input_section->owner;
   asection **sections = NULL;
   Elf_Internal_Rela *internal_relocs = NULL;
   Elf32_External_Sym *external_syms = NULL;
+  Elf_External_Sym_Shndx *shndx_buf = NULL;
+  Elf_External_Sym_Shndx *shndx;
   Elf_Internal_Sym *internal_syms = NULL;
 
   /* We only need to handle the case of relaxing, or of having a
@@ -2696,9 +2810,10 @@ mn10300_elf_get_relocated_section_contents (output_bfd, link_info, link_order,
                                                       symbols);
 
   symtab_hdr = &elf_tdata (input_bfd)->symtab_hdr;
+  shndx_hdr = &elf_tdata (input_bfd)->symtab_shndx_hdr;
 
   memcpy (data, elf_section_data (input_section)->this_hdr.contents,
-         input_section->_raw_size);
+         (size_t) input_section->_raw_size);
 
   if ((input_section->flags & SEC_RELOC) != 0
       && input_section->reloc_count > 0)
@@ -2706,20 +2821,31 @@ mn10300_elf_get_relocated_section_contents (output_bfd, link_info, link_order,
       Elf_Internal_Sym *isymp;
       asection **secpp;
       Elf32_External_Sym *esym, *esymend;
+      bfd_size_type amt;
 
       if (symtab_hdr->contents != NULL)
        external_syms = (Elf32_External_Sym *) symtab_hdr->contents;
-      else
+      else if (symtab_hdr->sh_info != 0)
        {
-         external_syms = ((Elf32_External_Sym *)
-                          bfd_malloc (symtab_hdr->sh_info
-                                      * sizeof (Elf32_External_Sym)));
-         if (external_syms == NULL && symtab_hdr->sh_info > 0)
+         amt = symtab_hdr->sh_info;
+         amt *= sizeof (Elf32_External_Sym);
+         external_syms = (Elf32_External_Sym *) bfd_malloc (amt);
+         if (external_syms == NULL)
            goto error_return;
          if (bfd_seek (input_bfd, symtab_hdr->sh_offset, SEEK_SET) != 0
-             || (bfd_read (external_syms, sizeof (Elf32_External_Sym),
-                           symtab_hdr->sh_info, input_bfd)
-                 != (symtab_hdr->sh_info * sizeof (Elf32_External_Sym))))
+             || bfd_bread ((PTR) external_syms, amt, input_bfd) != amt)
+           goto error_return;
+       }
+
+      if (symtab_hdr->sh_info != 0 && shndx_hdr->sh_size != 0)
+       {
+         amt = symtab_hdr->sh_info;
+         amt *= sizeof (Elf_External_Sym_Shndx);
+         shndx_buf = (Elf_External_Sym_Shndx *) bfd_malloc (amt);
+         if (shndx_buf == NULL)
+           goto error_return;
+         if (bfd_seek (input_bfd, shndx_hdr->sh_offset, SEEK_SET) != 0
+             || bfd_bread ((PTR) shndx_buf, amt, input_bfd) != amt)
            goto error_return;
        }
 
@@ -2729,40 +2855,35 @@ mn10300_elf_get_relocated_section_contents (output_bfd, link_info, link_order,
       if (internal_relocs == NULL)
        goto error_return;
 
-      internal_syms = ((Elf_Internal_Sym *)
-                      bfd_malloc (symtab_hdr->sh_info
-                                  * sizeof (Elf_Internal_Sym)));
-      if (internal_syms == NULL && symtab_hdr->sh_info > 0)
+      amt = symtab_hdr->sh_info;
+      amt *= sizeof (Elf_Internal_Sym);
+      internal_syms = (Elf_Internal_Sym *) bfd_malloc (amt);
+      if (internal_syms == NULL && amt != 0)
        goto error_return;
 
-      sections = (asection **) bfd_malloc (symtab_hdr->sh_info
-                                          * sizeof (asection *));
-      if (sections == NULL && symtab_hdr->sh_info > 0)
+      amt = symtab_hdr->sh_info;
+      amt *= sizeof (asection *);
+      sections = (asection **) bfd_malloc (amt);
+      if (sections == NULL && amt != 0)
        goto error_return;
 
-      isymp = internal_syms;
-      secpp = sections;
-      esym = external_syms;
-      esymend = esym + symtab_hdr->sh_info;
-      for (; esym < esymend; ++esym, ++isymp, ++secpp)
+      for (isymp = internal_syms, secpp = sections, shndx = shndx_buf,
+            esym = external_syms, esymend = esym + symtab_hdr->sh_info;
+          esym < esymend;
+          ++esym, ++isymp, ++secpp, shndx = (shndx ? shndx + 1 : NULL))
        {
          asection *isec;
 
-         bfd_elf32_swap_symbol_in (input_bfd, esym, isymp);
+         bfd_elf32_swap_symbol_in (input_bfd, esym, shndx, isymp);
 
          if (isymp->st_shndx == SHN_UNDEF)
            isec = bfd_und_section_ptr;
-         else if (isymp->st_shndx > 0 && isymp->st_shndx < SHN_LORESERVE)
-           isec = bfd_section_from_elf_index (input_bfd, isymp->st_shndx);
          else if (isymp->st_shndx == SHN_ABS)
            isec = bfd_abs_section_ptr;
          else if (isymp->st_shndx == SHN_COMMON)
            isec = bfd_com_section_ptr;
          else
-           {
-             /* Who knows?  */
-             isec = NULL;
-           }
+           isec = bfd_section_from_elf_index (input_bfd, isymp->st_shndx);
 
          *secpp = isec;
        }
@@ -2774,16 +2895,14 @@ mn10300_elf_get_relocated_section_contents (output_bfd, link_info, link_order,
 
       if (sections != NULL)
        free (sections);
-      sections = NULL;
       if (internal_syms != NULL)
        free (internal_syms);
-      internal_syms = NULL;
+      if (shndx_buf != NULL)
+       free (shndx_buf);
       if (external_syms != NULL && symtab_hdr->contents == NULL)
        free (external_syms);
-      external_syms = NULL;
       if (internal_relocs != elf_section_data (input_section)->relocs)
        free (internal_relocs);
-      internal_relocs = NULL;
     }
 
   return data;
@@ -2792,6 +2911,8 @@ mn10300_elf_get_relocated_section_contents (output_bfd, link_info, link_order,
   if (internal_relocs != NULL
       && internal_relocs != elf_section_data (input_section)->relocs)
     free (internal_relocs);
+  if (shndx_buf != NULL)
+    free (shndx_buf);
   if (external_syms != NULL && symtab_hdr->contents == NULL)
     free (external_syms);
   if (internal_syms != NULL)
@@ -2848,40 +2969,55 @@ elf32_mn10300_link_hash_table_create (abfd)
      bfd *abfd;
 {
   struct elf32_mn10300_link_hash_table *ret;
+  bfd_size_type amt = sizeof (struct elf32_mn10300_link_hash_table);
 
-  ret = ((struct elf32_mn10300_link_hash_table *)
-        bfd_alloc (abfd, sizeof (struct elf32_mn10300_link_hash_table)));
+  ret = (struct elf32_mn10300_link_hash_table *) bfd_malloc (amt);
   if (ret == (struct elf32_mn10300_link_hash_table *) NULL)
     return NULL;
 
   if (! _bfd_elf_link_hash_table_init (&ret->root, abfd,
                                       elf32_mn10300_link_hash_newfunc))
     {
-      bfd_release (abfd, ret);
+      free (ret);
       return NULL;
     }
 
   ret->flags = 0;
+  amt = sizeof (struct elf_link_hash_table);
   ret->static_hash_table
-    = ((struct elf32_mn10300_link_hash_table *)
-       bfd_alloc (abfd, sizeof (struct elf_link_hash_table)));
+    = (struct elf32_mn10300_link_hash_table *) bfd_malloc (amt);
   if (ret->static_hash_table == NULL)
     {
-      bfd_release (abfd, ret);
+      free (ret);
       return NULL;
     }
 
   if (! _bfd_elf_link_hash_table_init (&ret->static_hash_table->root, abfd,
                                       elf32_mn10300_link_hash_newfunc))
     {
-      bfd_release (abfd, ret->static_hash_table);
-      bfd_release (abfd, ret);
+      free (ret->static_hash_table);
+      free (ret);
       return NULL;
     }
   return &ret->root.root;
 }
 
-static int
+/* Free an mn10300 ELF linker hash table.  */
+
+static void
+elf32_mn10300_link_hash_table_free (hash)
+     struct bfd_link_hash_table *hash;
+{
+  struct elf32_mn10300_link_hash_table *ret
+    = (struct elf32_mn10300_link_hash_table *) hash;
+
+  _bfd_generic_link_hash_table_free
+    ((struct bfd_link_hash_table *) ret->static_hash_table);
+  _bfd_generic_link_hash_table_free
+    ((struct bfd_link_hash_table *) ret);
+}
+
+static unsigned long
 elf_mn10300_mach (flags)
      flagword flags;
 {
@@ -2958,7 +3094,8 @@ _bfd_mn10300_elf_merge_private_bfd_data (ibfd, obfd)
 #define TARGET_LITTLE_SYM      bfd_elf32_mn10300_vec
 #define TARGET_LITTLE_NAME     "elf32-mn10300"
 #define ELF_ARCH               bfd_arch_mn10300
-#define ELF_MACHINE_CODE       EM_CYGNUS_MN10300
+#define ELF_MACHINE_CODE       EM_MN10300
+#define ELF_MACHINE_ALT1       EM_CYGNUS_MN10300
 #define ELF_MAXPAGESIZE                0x1000
 
 #define elf_info_to_howto              mn10300_info_to_howto
@@ -2972,6 +3109,8 @@ _bfd_mn10300_elf_merge_private_bfd_data (ibfd, obfd)
                                mn10300_elf_get_relocated_section_contents
 #define bfd_elf32_bfd_link_hash_table_create \
                                elf32_mn10300_link_hash_table_create
+#define bfd_elf32_bfd_link_hash_table_free \
+                               elf32_mn10300_link_hash_table_free
 
 #define elf_symbol_leading_char '_'
 
This page took 0.039681 seconds and 4 git commands to generate.