Update year range in copyright notice of binutils files
[deliverable/binutils-gdb.git] / bfd / coff-m88k.c
index a261c8f4fd1989079fd9326421d314b4093b4074..93a2d48729a25296bb4542c07d3267584428880b 100644 (file)
-/* Copyright (C) 1990, 1991 Free Software Foundation, Inc.
+/* BFD back-end for Motorola 88000 COFF "Binary Compatibility Standard" files.
+   Copyright (C) 1990-2018 Free Software Foundation, Inc.
+   Written by Cygnus Support.
 
-This file is part of BFD, the Binary File Diddler.
+   This file is part of BFD, the Binary File Descriptor library.
 
-BFD is free software; you can redistribute it and/or modify
-it under the terms of the GNU General Public License as published by
-the Free Software Foundation; either version 1, or (at your option)
-any later version.
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 3 of the License, or
+   (at your option) any later version.
 
-BFD is distributed in the hope that it will be useful,
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-GNU General Public License for more details.
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
 
-You should have received a copy of the GNU General Public License
-along with BFD; see the file COPYING.  If not, write to
-the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
+   MA 02110-1301, USA.  */
 
-
-/* $Id$ */
-
-#define M88 1
-#include <ansidecl.h>
-#include <sysdep.h>
+#define M88 1          /* Customize various include files */
+#include "sysdep.h"
 #include "bfd.h"
 #include "libbfd.h"
+#include "coff/m88k.h"
+#include "coff/internal.h"
 #include "libcoff.h"
-#include "m88k-bcs.h"
 
+static bfd_reloc_status_type m88k_special_reloc
+  (bfd *, arelent *, asymbol *, void *, asection *, bfd *, char **);
+
+#define COFF_DEFAULT_SECTION_ALIGNMENT_POWER (3)
+
+#define GET_SCNHDR_NRELOC H_GET_32
+#define GET_SCNHDR_NLNNO  H_GET_32
+
+/* On coff-m88k, local labels start with '@'.  */
+
+#define coff_bfd_is_local_label_name m88k_is_local_label_name
 
-static reloc_howto_type howto_table[] = 
+static bfd_boolean
+m88k_is_local_label_name (bfd *abfd ATTRIBUTE_UNUSED, const char *name)
 {
-/*     type    rtshift size    bitsize pc_rel  bitpos abs ovff sf  name partial inplace mask*/ 
-       R_PCR16L, 2,    1,      16,     true,   0, false, true, 0,"PCR16L", false, 0x0000ffff,
-       R_PCR26L, 2,    2,      26,     true,   0, false, true, 0,"PCR26L", false, 0x03ffffff,
-       R_VRT16, 0,     1,      16,     false,  0, false, true, 0,"VRT16", false, 0x0000ffff,
-       R_HVRT16,16,    1,      16,     false,  0, false, true, 0,"HVRT16", false, 0x0000ffff,
-       R_LVRT16, 0,    1,      16,     false,  0, false, true, 0,"LVRT16", false, 0x0000ffff,
-       R_VRT32,  0,    2,      32,     false,  0, false, true, 0,"VRT32", false, 0xffffffff,
-};
+  return name[0] == '@';
+}
 
+static bfd_reloc_status_type
+m88k_special_reloc (bfd *abfd,
+                   arelent *reloc_entry,
+                   asymbol *symbol,
+                   void * data,
+                   asection *input_section,
+                   bfd *output_bfd,
+                   char **error_message ATTRIBUTE_UNUSED)
+{
+  reloc_howto_type *howto = reloc_entry->howto;
 
+  switch (howto->type)
+    {
+    case R_HVRT16:
+    case R_LVRT16:
+      if (output_bfd != (bfd *) NULL)
+       {
+         /* This is a partial relocation, and we want to apply the
+            relocation to the reloc entry rather than the raw data.
+            Modify the reloc inplace to reflect what we now know.  */
 
-#define BADMAG(x) MC88BADMAG(x)
-#include "coff-code.h"
+         reloc_entry->address += input_section->output_offset;
+       }
+      else
+       {
+         bfd_vma output_base = 0;
+         bfd_vma addr = reloc_entry->address;
+         bfd_vma x;
+         asection *reloc_target_output_section;
+         long relocation = 0;
+
+         if (! bfd_reloc_offset_in_range (howto, abfd, input_section,
+                                          reloc_entry->address
+                                          * bfd_octets_per_byte (abfd)))
+           return bfd_reloc_outofrange;
+
+         x = bfd_get_16 (abfd, (bfd_byte *) data + addr);
+
+         /* Work out which section the relocation is targeted at and the
+            initial relocation command value.  */
+
+         /* Get symbol value.  (Common symbols are special.)  */
+         if (bfd_is_com_section (symbol->section))
+           relocation = 0;
+         else
+           relocation = symbol->value;
+
+         reloc_target_output_section = symbol->section->output_section;
+
+         /* Convert input-section-relative symbol value to absolute.  */
+         if (output_bfd)
+           output_base = 0;
+         else
+           output_base = reloc_target_output_section->vma;
+
+         relocation += output_base + symbol->section->output_offset;
+
+         /* Add in supplied addend.  */
+         relocation += ((reloc_entry->addend << howto->bitsize) + x);
+
+         reloc_entry->addend = 0;
+
+         relocation >>= (bfd_vma) howto->rightshift;
+
+         /* Shift everything up to where it's going to be used */
+
+         relocation <<= (bfd_vma) howto->bitpos;
+
+         if (relocation)
+           bfd_put_16 (abfd, (bfd_vma) relocation,
+                       (unsigned char *) data + addr);
+       }
+
+      /* If we are not producing relocatable output, return an error if
+        the symbol is not defined.  */
+      if (bfd_is_und_section (symbol->section) && output_bfd == (bfd *) NULL)
+       return bfd_reloc_undefined;
 
+      return bfd_reloc_ok;
 
+    default:
+      if (output_bfd != (bfd *) NULL)
+       {
+         /* This is a partial relocation, and we want to apply the
+            relocation to the reloc entry rather than the raw data.
+            Modify the reloc inplace to reflect what we now know.  */
 
-bfd_target m88k_bcs_vec =
+         reloc_entry->address += input_section->output_offset;
+         return bfd_reloc_ok;
+       }
+      break;
+    }
+
+  if (output_bfd == (bfd *) NULL)
+    return bfd_reloc_continue;
+
+  return bfd_reloc_ok;
+}
+
+static reloc_howto_type howto_table[] =
 {
-  "m88k-bcs",          /* name */
-  bfd_target_coff_flavour_enum,
-  false,                       /* data byte order is little */
-  true,                                /* header byte order is big */
-
-  (HAS_RELOC | EXEC_P |                /* object flags */
-   HAS_LINENO | HAS_DEBUG |
-   HAS_SYMS | HAS_LOCALS | DYNAMIC | WP_TEXT),
-
-  (SEC_HAS_CONTENTS | SEC_ALLOC | SEC_LOAD | SEC_RELOC), /* section flags */
-  0,                           /* valid reloc types */
-  '/',                         /* ar_pad_char */
-  15,                          /* ar_max_namelen */
-  coff_close_and_cleanup,      /* _close_and_cleanup */
-  coff_set_section_contents,   /* bfd_set_section_contents */
-  coff_get_section_contents,   /* bfd_get_section_contents */
-  coff_new_section_hook,       /* new_section_hook */
-  _bfd_dummy_core_file_failing_command, /* _core_file_failing_command */
-  _bfd_dummy_core_file_failing_signal, /* _core_file_failing_signal */
-  _bfd_dummy_core_file_matches_executable_p, /* _core_file_matches_ex...p */
-
-  bfd_slurp_coff_armap,                /* bfd_slurp_armap */
-  _bfd_slurp_extended_name_table, /* bfd_slurp_extended_name_table*/
-#if 0                          /*  */
-  bfd_dont_truncate_arname,    /* bfd_truncate_arname */
-#else
-  bfd_bsd_truncate_arname,
-#endif
+  HOWTO (R_PCR16L,                     /* type */
+        02,                            /* rightshift */
+        1,                             /* size (0 = byte, 1 = short, 2 = long) */
+        16,                            /* bitsize */
+        TRUE,                          /* pc_relative */
+        0,                             /* bitpos */
+        complain_overflow_signed,      /* complain_on_overflow */
+        m88k_special_reloc,            /* special_function */
+        "PCR16L",                      /* name */
+        FALSE,                         /* partial_inplace */
+        0x0000ffff,                    /* src_mask */
+        0x0000ffff,                    /* dst_mask */
+        TRUE),                         /* pcrel_offset */
+
+  HOWTO (R_PCR26L,                     /* type */
+        02,                            /* rightshift */
+        2,                             /* size (0 = byte, 1 = short, 2 = long) */
+        26,                            /* bitsize */
+        TRUE,                          /* pc_relative */
+        0,                             /* bitpos */
+        complain_overflow_signed,      /* complain_on_overflow */
+        m88k_special_reloc,            /* special_function */
+        "PCR26L",                      /* name */
+        FALSE,                         /* partial_inplace */
+        0x03ffffff,                    /* src_mask */
+        0x03ffffff,                    /* dst_mask */
+        TRUE),                         /* pcrel_offset */
+
+  HOWTO (R_VRT16,                      /* type */
+        00,                            /* rightshift */
+        1,                             /* size (0 = byte, 1 = short, 2 = long) */
+        16,                            /* bitsize */
+        FALSE,                         /* pc_relative */
+        0,                             /* bitpos */
+        complain_overflow_bitfield,    /* complain_on_overflow */
+        m88k_special_reloc,            /* special_function */
+        "VRT16",                       /* name */
+        FALSE,                         /* partial_inplace */
+        0x0000ffff,                    /* src_mask */
+        0x0000ffff,                    /* dst_mask */
+        TRUE),                         /* pcrel_offset */
+
+  HOWTO (R_HVRT16,                     /* type */
+        16,                            /* rightshift */
+        1,                             /* size (0 = byte, 1 = short, 2 = long) */
+        16,                            /* bitsize */
+        FALSE,                         /* pc_relative */
+        0,                             /* bitpos */
+        complain_overflow_dont,        /* complain_on_overflow */
+        m88k_special_reloc,            /* special_function */
+        "HVRT16",                      /* name */
+        FALSE,                         /* partial_inplace */
+        0x0000ffff,                    /* src_mask */
+        0x0000ffff,                    /* dst_mask */
+        TRUE),                         /* pcrel_offset */
+
+  HOWTO (R_LVRT16,                     /* type */
+        00,                            /* rightshift */
+        1,                             /* size (0 = byte, 1 = short, 2 = long) */
+        16,                            /* bitsize */
+        FALSE,                         /* pc_relative */
+        0,                             /* bitpos */
+        complain_overflow_dont,        /* complain_on_overflow */
+        m88k_special_reloc,            /* special_function */
+        "LVRT16",                      /* name */
+        FALSE,                         /* partial_inplace */
+        0x0000ffff,                    /* src_mask */
+        0x0000ffff,                    /* dst_mask */
+        TRUE),                         /* pcrel_offset */
+
+  HOWTO (R_VRT32,                      /* type */
+        00,                            /* rightshift */
+        2,                             /* size (0 = byte, 1 = short, 2 = long) */
+        32,                            /* bitsize */
+        FALSE,                         /* pc_relative */
+        0,                             /* bitpos */
+        complain_overflow_bitfield,    /* complain_on_overflow */
+        m88k_special_reloc,            /* special_function */
+        "VRT32",                       /* name */
+        FALSE,                         /* partial_inplace */
+        0xffffffff,                    /* src_mask */
+        0xffffffff,                    /* dst_mask */
+        TRUE),                         /* pcrel_offset */
+};
+
+/* Code to turn an external r_type into a pointer to an entry in the
+   above howto table.  */
+static void
+rtype2howto (arelent *cache_ptr, struct internal_reloc *dst)
+{
+  if (dst->r_type >= R_PCR16L && dst->r_type <= R_VRT32)
+    {
+      cache_ptr->howto = howto_table + dst->r_type - R_PCR16L;
+    }
+  else
+    {
+      BFD_ASSERT (0);
+    }
+}
+
+#define RTYPE2HOWTO(cache_ptr, dst) rtype2howto (cache_ptr, dst)
+
+/* Code to swap in the reloc offset */
+#define SWAP_IN_RELOC_OFFSET  H_GET_16
+#define SWAP_OUT_RELOC_OFFSET H_PUT_16
+
+#define RELOC_PROCESSING(relent,reloc,symbols,abfd,section)    \
+  reloc_processing(relent, reloc, symbols, abfd, section)
+
+static void
+reloc_processing (arelent *relent,
+                 struct internal_reloc *reloc,
+                 asymbol **symbols,
+                 bfd *abfd,
+                 asection *section)
+{
+  relent->address = reloc->r_vaddr;
+  rtype2howto (relent, reloc);
 
-  coff_get_symtab_upper_bound, /* get_symtab_upper_bound */
-  coff_get_symtab,             /* canonicalize_symtab */
-  (void (*)())bfd_false,       /* bfd_reclaim_symbol_table */
-  coff_get_reloc_upper_bound,  /* get_reloc_upper_bound */
-  coff_canonicalize_reloc,     /* bfd_canonicalize_reloc */
-  (void (*)())bfd_false,       /* bfd_reclaim_reloc */
-
-  coff_get_symcount_upper_bound, /* bfd_get_symcount_upper_bound */
-  coff_get_first_symbol,       /* bfd_get_first_symbol */
-  coff_get_next_symbol,                /* bfd_get_next_symbol */
-  coff_classify_symbol,                /* bfd_classify_symbol */
-  coff_symbol_hasclass,                /* bfd_symbol_hasclass */
-  coff_symbol_name,            /* bfd_symbol_name */
-  coff_symbol_value,           /* bfd_symbol_value */
-
-  _do_getllong, _do_putllong, _do_getlshort, _do_putlshort, /* data */
-  _do_getblong, _do_putblong, _do_getbshort, _do_putbshort, /* hdrs */
-
-  {_bfd_dummy_target, coff_object_p, /* bfd_check_format */
-     bfd_generic_archive_p, _bfd_dummy_target},
-  {bfd_false, coff_mkobject, _bfd_generic_mkarchive, /* bfd_set_format */
-     bfd_false},
-  coff_make_empty_symbol,
-  coff_print_symbol,
-  coff_get_lineno,
-  coff_set_arch_mach,
-  coff_write_armap,
-  bfd_generic_openr_next_archived_file,
-    coff_find_nearest_line,
-    bfd_generic_stat_arch_elt /* bfd_stat_arch_elt */
-  };
+  if (((int) reloc->r_symndx) > 0)
+    {
+      relent->sym_ptr_ptr = symbols + obj_convert (abfd)[reloc->r_symndx];
+    }
+  else
+    {
+      relent->sym_ptr_ptr = bfd_abs_section_ptr->symbol_ptr_ptr;
+    }
+
+  relent->addend = reloc->r_offset;
+  relent->address -= section->vma;
+}
+
+#define BADMAG(x) MC88BADMAG(x)
+
+#ifndef bfd_pe_print_pdata
+#define bfd_pe_print_pdata     NULL
+#endif
 
+#include "coffcode.h"
 
+#undef coff_write_armap
 
+CREATE_BIG_COFF_TARGET_VEC (m88k_coff_bcs_vec, "coff-m88kbcs", 0, 0, '_', NULL, COFF_SWAP_TABLE)
This page took 0.02673 seconds and 4 git commands to generate.