]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blobdiff - gas/config/tc-ns32k.c
Convert unmaintained files over to ISO-C90 and fix formatting.
[thirdparty/binutils-gdb.git] / gas / config / tc-ns32k.c
index 9d7ef82d977e6d623f41d74d8966f3ecc44b9d27..feae0fd36ef31c4798e04a4842947b5898983e29 100644 (file)
                                    encountered.  */
 
 #define IIF(ptr,a1,c1,e1,g1,i1,k1,m1,o1,q1,s1,u1)      \
-    iif.iifP[ptr].type= a1;                            \
-    iif.iifP[ptr].size= c1;                            \
-    iif.iifP[ptr].object= e1;                          \
-    iif.iifP[ptr].object_adjust= g1;                   \
-    iif.iifP[ptr].pcrel= i1;                           \
-    iif.iifP[ptr].pcrel_adjust= k1;                    \
-    iif.iifP[ptr].im_disp= m1;                         \
-    iif.iifP[ptr].relax_substate= o1;                  \
-    iif.iifP[ptr].bit_fixP= q1;                                \
-    iif.iifP[ptr].addr_mode= s1;                       \
-    iif.iifP[ptr].bsr= u1;
+    iif.iifP[ptr].type = a1;                           \
+    iif.iifP[ptr].size = c1;                           \
+    iif.iifP[ptr].object = e1;                         \
+    iif.iifP[ptr].object_adjust = g1;                  \
+    iif.iifP[ptr].pcrel = i1;                          \
+    iif.iifP[ptr].pcrel_adjust = k1;                   \
+    iif.iifP[ptr].im_disp = m1;                                \
+    iif.iifP[ptr].relax_substate = o1;                 \
+    iif.iifP[ptr].bit_fixP = q1;                       \
+    iif.iifP[ptr].addr_mode = s1;                      \
+    iif.iifP[ptr].bsr = u1;
 
 #ifdef SEQUENT_COMPATABILITY
 #define LINE_COMMENT_CHARS "|"
@@ -70,21 +70,21 @@ static int default_disp_size = 4; /* Displacement size for external refs.  */
 #endif
 
 struct addr_mode
-  {
-    signed char mode;          /* Addressing mode of operand (0-31).  */
-    signed char scaled_mode;   /* Mode combined with scaled mode.  */
-    char scaled_reg;           /* Register used in scaled+1 (1-8).  */
-    char float_flag;           /* Set if R0..R7 was F0..F7 ie a
+{
+  signed char mode;            /* Addressing mode of operand (0-31).  */
+  signed char scaled_mode;     /* Mode combined with scaled mode.  */
+  char scaled_reg;             /* Register used in scaled+1 (1-8).  */
+  char float_flag;             /* Set if R0..R7 was F0..F7 ie a
                                   floating-point-register.  */
-    char am_size;              /* Estimated max size of general addr-mode
+  char am_size;                        /* Estimated max size of general addr-mode
                                   parts.  */
-    char im_disp;              /* If im_disp==1 we have a displacement.  */
-    char pcrel;                        /* 1 if pcrel, this is really redundant info.  */
-    char disp_suffix[2];       /* Length of displacement(s), 0=undefined.  */
-    char *disp[2];             /* Pointer(s) at displacement(s)
+  char im_disp;                        /* If im_disp==1 we have a displacement.  */
+  char pcrel;                  /* 1 if pcrel, this is really redundant info.  */
+  char disp_suffix[2];         /* Length of displacement(s), 0=undefined.  */
+  char *disp[2];               /* Pointer(s) at displacement(s)
                                   or immediates(s)     (ascii).  */
-    char index_byte;           /* Index byte.  */
-  };
+  char index_byte;             /* Index byte.  */
+};
 typedef struct addr_mode addr_modeS;
 
 char *freeptr, *freeptr_static;        /* Points at some number of free bytes.  */
@@ -112,27 +112,27 @@ struct ns32k_option
 };
 
 typedef struct
-  {
-    int type;                  /* How to interpret object.  */
-    int size;                  /* Estimated max size of object.  */
-    unsigned long object;      /* Binary data.  */
-    int object_adjust;         /* Number added to object.  */
-    int pcrel;                 /* True if object is pcrel.  */
-    int pcrel_adjust;          /* Length in bytes from the instruction
+{
+  int type;                    /* How to interpret object.  */
+  int size;                    /* Estimated max size of object.  */
+  unsigned long object;                /* Binary data.  */
+  int object_adjust;           /* Number added to object.  */
+  int pcrel;                   /* True if object is pcrel.  */
+  int pcrel_adjust;            /* Length in bytes from the instruction
                                   start to the displacement.  */
-    int im_disp;               /* True if the object is a displacement.  */
-    relax_substateT relax_substate;    /* Initial relaxsubstate.  */
-    bit_fixS *bit_fixP;                /* Pointer at bit_fix struct.  */
-    int addr_mode;             /* What addrmode do we associate with this
+  int im_disp;                 /* True if the object is a displacement.  */
+  relax_substateT relax_substate;/*Initial relaxsubstate.  */
+  bit_fixS *bit_fixP;          /* Pointer at bit_fix struct.  */
+  int addr_mode;               /* What addrmode do we associate with this
                                   iif-entry.  */
-    char bsr;                  /* Sequent hack.  */
-  } iif_entryT;                        /* Internal Instruction Format.  */
+  char bsr;                    /* Sequent hack.  */
+} iif_entryT;                  /* Internal Instruction Format.  */
 
 struct int_ins_form
-  {
-    int instr_size;            /* Max size of instruction in bytes.  */
-    iif_entryT iifP[IIF_ENTRIES + 1];
-  };
+{
+  int instr_size;              /* Max size of instruction in bytes.  */
+  iif_entryT iifP[IIF_ENTRIES + 1];
+};
 
 struct int_ins_form iif;
 expressionS exprP;
@@ -367,11 +367,6 @@ char disp_test[] =
 char disp_size[] =
 {4, 1, 2, 0, 4};
 \f
-static void evaluate_expr PARAMS ((expressionS * resultP, char *));
-static void md_number_to_disp PARAMS ((char *, long, int));
-static void md_number_to_imm PARAMS ((char *, long, int));
-static void md_number_to_field PARAMS ((char *, long, bit_fixS *));
-
 /* Parse a general operand into an addressingmode struct
 
    In:  pointer at operand in ascii form
@@ -380,13 +375,10 @@ static void md_number_to_field PARAMS ((char *, long, bit_fixS *));
 
    Out: data in addr_mode struct.  */
 
-static int addr_mode PARAMS ((char *, addr_modeS *, int));
-
 static int
-addr_mode (operand, addr_modeP, recursive_level)
-     char *operand;
-     addr_modeS *addr_modeP;
-     int recursive_level;
+addr_mode (char *operand,
+          addr_modeS *addr_modeP,
+          int recursive_level)
 {
   char *str;
   int i;
@@ -657,17 +649,25 @@ addr_mode (operand, addr_modeP, recursive_level)
   return -1;
 }
 \f
+static void
+evaluate_expr (expressionS *resultP, char *ptr)
+{
+  char *tmp_line;
+
+  tmp_line = input_line_pointer;
+  input_line_pointer = ptr;
+  expression (resultP);
+  input_line_pointer = tmp_line;
+}
+
 /* ptr points at string addr_modeP points at struct with result This
    routine calls addr_mode to determine the general addr.mode of the
    operand. When this is ready it parses the displacements for size
    specifying suffixes and determines size of immediate mode via
    ns32k-opcode.  Also builds index bytes if needed.  */
 
-static int get_addr_mode PARAMS ((char *, addr_modeS *));
 static int
-get_addr_mode (ptr, addr_modeP)
-     char *ptr;
-     addr_modeS *addr_modeP;
+get_addr_mode (char *ptr, addr_modeS *addr_modeP)
 {
   int tmp;
 
@@ -801,12 +801,10 @@ get_addr_mode (ptr, addr_modeP)
 
 /* Read an optionlist.  */
 
-static void optlist PARAMS ((char *, struct ns32k_option *, unsigned long *));
 static void
-optlist (str, optionP, default_map)
-     char *str;                        /* The string to extract options from.  */
-     struct ns32k_option *optionP;     /* How to search the string.  */
-     unsigned long *default_map;       /* Default pattern and output.  */
+optlist (char *str,                    /* The string to extract options from.  */
+        struct ns32k_option *optionP,  /* How to search the string.  */
+        unsigned long *default_map)    /* Default pattern and output.  */
 {
   int i, j, k, strlen1, strlen2;
   char *patternP, *strP;
@@ -845,13 +843,10 @@ optlist (str, optionP, default_map)
    the instructions lmr, smr, lpr, spr return true if str is found in
    list.  */
 
-static int list_search PARAMS ((char *, struct ns32k_option *, unsigned long *));
-
 static int
-list_search (str, optionP, default_map)
-     char *str;                                /* The string to match.  */
-     struct ns32k_option *optionP;     /* List to search.  */
-     unsigned long *default_map;       /* Default pattern and output.  */
+list_search (char *str,                                /* The string to match.  */
+            struct ns32k_option *optionP,      /* List to search.  */
+            unsigned long *default_map)        /* Default pattern and output.  */
 {
   int i;
 
@@ -870,37 +865,49 @@ list_search (str, optionP, default_map)
   as_bad (_("No such entry in list. (cpu/mmu register)"));
   return 0;
 }
+\f
+/* Create a bit_fixS in obstack 'notes'.
+   This struct is used to profile the normal fix. If the bit_fixP is a
+   valid pointer (not NULL) the bit_fix data will be used to format
+   the fix.  */
 
-static void
-evaluate_expr (resultP, ptr)
-     expressionS *resultP;
-     char *ptr;
+static bit_fixS *
+bit_fix_new (int size,         /* Length of bitfield.  */
+            int offset,        /* Bit offset to bitfield.  */
+            long min,          /* Signextended min for bitfield.  */
+            long max,          /* Signextended max for bitfield.  */
+            long add,          /* Add mask, used for huffman prefix.  */
+            long base_type,    /* 0 or 1, if 1 it's exploded to opcode ptr.  */
+            long base_adj)
 {
-  char *tmp_line;
+  bit_fixS *bit_fixP;
 
-  tmp_line = input_line_pointer;
-  input_line_pointer = ptr;
-  expression (resultP);
-  input_line_pointer = tmp_line;
+  bit_fixP = (bit_fixS *) obstack_alloc (&notes, sizeof (bit_fixS));
+
+  bit_fixP->fx_bit_size = size;
+  bit_fixP->fx_bit_offset = offset;
+  bit_fixP->fx_bit_base = base_type;
+  bit_fixP->fx_bit_base_adj = base_adj;
+  bit_fixP->fx_bit_max = max;
+  bit_fixP->fx_bit_min = min;
+  bit_fixP->fx_bit_add = add;
+
+  return bit_fixP;
 }
-\f
+
 /* Convert operands to iif-format and adds bitfields to the opcode.
    Operands are parsed in such an order that the opcode is updated from
    its most significant bit, that is when the operand need to alter the
    opcode.
    Be careful not to put to objects in the same iif-slot.  */
 
-static void encode_operand
-  PARAMS ((int, char **, const char *, const char *, char, char));
-
 static void
-encode_operand (argc, argv, operandsP, suffixP, im_size, opcode_bit_ptr)
-     int argc;
-     char **argv;
-     const char *operandsP;
-     const char *suffixP;
-     char im_size ATTRIBUTE_UNUSED;
-     char opcode_bit_ptr;
+encode_operand (int argc,
+               char **argv,
+               const char *operandsP,
+               const char *suffixP,
+               char im_size ATTRIBUTE_UNUSED,
+               char opcode_bit_ptr)
 {
   int i, j;
   char d;
@@ -1079,12 +1086,8 @@ encode_operand (argc, argv, operandsP, suffixP, im_size, opcode_bit_ptr)
    Return-value = recursive_level.  */
 /* Build iif of one assembly text line.  */
 
-static int parse PARAMS ((const char *, int));
-
 static int
-parse (line, recursive_level)
-     const char *line;
-     int recursive_level;
+parse (const char *line, int recursive_level)
 {
   const char *lineptr;
   char c, suffix_separator;
@@ -1109,9 +1112,7 @@ parse (line, recursive_level)
       *(char *) lineptr = c;
     }
   else
-    {
-      lineptr = line;
-    }
+    lineptr = line;
 
   argc = 0;
 
@@ -1208,9 +1209,7 @@ parse (line, recursive_level)
                  lineptr += 1;
                }
              else
-               {
-                 as_fatal (_("Too many operands passed to instruction"));
-               }
+               as_fatal (_("Too many operands passed to instruction"));
            }
        }
     }
@@ -1225,9 +1224,7 @@ parse (line, recursive_level)
            as_fatal (_("Wrong numbers of operands in default, check ns32k-opcodes.h"));
        }
       else
-       {
-         as_fatal (_("Wrong number of operands"));
-       }
+       as_fatal (_("Wrong number of operands"));
     }
 
   for (i = 0; i < IIF_ENTRIES; i++)
@@ -1240,183 +1237,531 @@ parse (line, recursive_level)
 
   /* This call encodes operands to iif format.  */
   if (argc)
-    {
-      encode_operand (argc,
-                     argv,
-                     &desc->operands[0],
-                     &suffix[0],
-                     desc->im_size,
-                     desc->opcode_size);
-    }
+    encode_operand (argc, argv, &desc->operands[0],
+                   &suffix[0], desc->im_size, desc->opcode_size);
+
   return recursive_level;
 }
 \f
-/* Convert iif to fragments.  From this point we start to dribble with
-   functions in other files than this one.(Except hash.c) So, if it's
-   possible to make an iif for an other CPU, you don't need to know
-   what frags, relax, obstacks, etc is in order to port this
-   assembler. You only need to know if it's possible to reduce your
-   cpu-instruction to iif-format (takes some work) and adopt the other
-   md_? parts according to given instructions Note that iif was
-   invented for the clean ns32k`s architecture.  */
-
-/* GAS for the ns32k has a problem. PC relative displacements are
-   relative to the address of the opcode, not the address of the
-   operand. We used to keep track of the offset between the operand
-   and the opcode in pcrel_adjust for each frag and each fix. However,
-   we get into trouble where there are two or more pc-relative
-   operands and the size of the first one can't be determined. Then in
-   the relax phase, the size of the first operand will change and
-   pcrel_adjust will no longer be correct.  The current solution is
-   keep a pointer to the frag with the opcode in it and the offset in
-   that frag for each frag and each fix. Then, when needed, we can
-   always figure out how far it is between the opcode and the pcrel
-   object.  See also md_pcrel_adjust and md_fix_pcrel_adjust.  For
-   objects not part of an instruction, the pointer to the opcode frag
-   is always zero.  */
+#ifdef BFD_ASSEMBLER
+/* This functionality should really be in the bfd library.  */
 
-static void convert_iif PARAMS ((void));
-static void
-convert_iif ()
+static bfd_reloc_code_real_type
+reloc (int size, int pcrel, int type)
 {
-  int i;
-  bit_fixS *j;
-  fragS *inst_frag;
-  unsigned int inst_offset;
-  char *inst_opcode;
-  char *memP;
-  int l;
-  int k;
-  char type;
-  char size = 0;
+  int length, index;
+  bfd_reloc_code_real_type relocs[] =
+  {
+    BFD_RELOC_NS32K_IMM_8,
+    BFD_RELOC_NS32K_IMM_16,
+    BFD_RELOC_NS32K_IMM_32,
+    BFD_RELOC_NS32K_IMM_8_PCREL,
+    BFD_RELOC_NS32K_IMM_16_PCREL,
+    BFD_RELOC_NS32K_IMM_32_PCREL,
 
-  frag_grow (iif.instr_size);  /* This is important.  */
-  memP = frag_more (0);
-  inst_opcode = memP;
-  inst_offset = (memP - frag_now->fr_literal);
-  inst_frag = frag_now;
+    /* ns32k displacements.  */
+    BFD_RELOC_NS32K_DISP_8,
+    BFD_RELOC_NS32K_DISP_16,
+    BFD_RELOC_NS32K_DISP_32,
+    BFD_RELOC_NS32K_DISP_8_PCREL,
+    BFD_RELOC_NS32K_DISP_16_PCREL,
+    BFD_RELOC_NS32K_DISP_32_PCREL,
 
-  for (i = 0; i < IIF_ENTRIES; i++)
+    /* Normal 2's complement.  */
+    BFD_RELOC_8,
+    BFD_RELOC_16,
+    BFD_RELOC_32,
+    BFD_RELOC_8_PCREL,
+    BFD_RELOC_16_PCREL,
+    BFD_RELOC_32_PCREL
+  };
+
+  switch (size)
     {
-      if ((type = iif.iifP[i].type))
-       {
-         /* The object exist, so handle it.  */
-         switch (size = iif.iifP[i].size)
-           {
-           case 42:
-             size = 0;
-             /* It's a bitfix that operates on an existing object.  */
-             if (iif.iifP[i].bit_fixP->fx_bit_base)
-               /* Expand fx_bit_base to point at opcode.  */
-               iif.iifP[i].bit_fixP->fx_bit_base = (long) inst_opcode;
-             /* Fall through.  */
+    case 1:
+      length = 0;
+      break;
+    case 2:
+      length = 1;
+      break;
+    case 4:
+      length = 2;
+      break;
+    default:
+      length = -1;
+      break;
+    }
 
-           case 8:             /* bignum or doublefloat.  */
-           case 1:
-           case 2:
-           case 3:
-           case 4:
-             /* The final size in objectmemory is known.  */
-             memP = frag_more (size);
-             j = iif.iifP[i].bit_fixP;
+  index = length + 3 * pcrel + 6 * type;
 
-             switch (type)
-               {
-               case 1: /* The object is pure binary.  */
-                 if (j)
-                   {
-                     md_number_to_field(memP, exprP.X_add_number, j);
-                   }
-                 else if (iif.iifP[i].pcrel)
-                   {
-                     fix_new_ns32k (frag_now,
-                                    (long) (memP - frag_now->fr_literal),
-                                    size,
-                                    0,
-                                    iif.iifP[i].object,
-                                    iif.iifP[i].pcrel,
-                                    iif.iifP[i].im_disp,
-                                    0,
-                                    iif.iifP[i].bsr,   /* Sequent hack.  */
-                                    inst_frag, inst_offset);
-                   }
-                 else
-                   {
-                     /* Good, just put them bytes out.  */
-                     switch (iif.iifP[i].im_disp)
-                       {
-                       case 0:
-                         md_number_to_chars (memP, iif.iifP[i].object, size);
-                         break;
-                       case 1:
-                         md_number_to_disp (memP, iif.iifP[i].object, size);
-                         break;
-                       default:
-                         as_fatal (_("iif convert internal pcrel/binary"));
-                       }
-                   }
-                 break;
+  if (index >= 0 && (unsigned int) index < sizeof (relocs) / sizeof (relocs[0]))
+    return relocs[index];
 
-               case 2:
-                 /* The object is a pointer at an expression, so
-                     unpack it, note that bignums may result from the
-                     expression.  */
-                 evaluate_expr (&exprP, (char *) iif.iifP[i].object);
-                 if (exprP.X_op == O_big || size == 8)
-                   {
-                     if ((k = exprP.X_add_number) > 0)
-                       {
-                         /* We have a bignum ie a quad. This can only
-                             happens in a long suffixed instruction.  */
-                         if (k * 2 > size)
-                           as_bad (_("Bignum too big for long"));
+  if (pcrel)
+    as_bad (_("Can not do %d byte pc-relative relocation for storage type %d"),
+           size, type);
+  else
+    as_bad (_("Can not do %d byte relocation for storage type %d"),
+           size, type);
 
-                         if (k == 3)
-                           memP += 2;
+  return BFD_RELOC_NONE;
 
-                         for (l = 0; k > 0; k--, l += 2)
-                           md_number_to_chars (memP + l,
-                                               generic_bignum[l >> 1],
-                                               sizeof (LITTLENUM_TYPE));
-                       }
-                     else
-                       {
-                         /* flonum.  */
-                         LITTLENUM_TYPE words[4];
+}
+#endif
 
-                         switch (size)
-                           {
-                           case 4:
-                             gen_to_words (words, 2, 8);
-                             md_number_to_imm (memP, (long) words[0],
-                                               sizeof (LITTLENUM_TYPE));
-                             md_number_to_imm (memP + sizeof (LITTLENUM_TYPE),
-                                               (long) words[1],
-                                               sizeof (LITTLENUM_TYPE));
-                             break;
-                           case 8:
-                             gen_to_words (words, 4, 11);
-                             md_number_to_imm (memP, (long) words[0],
-                                               sizeof (LITTLENUM_TYPE));
-                             md_number_to_imm (memP + sizeof (LITTLENUM_TYPE),
-                                               (long) words[1],
-                                               sizeof (LITTLENUM_TYPE));
-                             md_number_to_imm ((memP + 2
-                                                * sizeof (LITTLENUM_TYPE)),
-                                               (long) words[2],
-                                               sizeof (LITTLENUM_TYPE));
-                             md_number_to_imm ((memP + 3
-                                                * sizeof (LITTLENUM_TYPE)),
-                                               (long) words[3],
-                                               sizeof (LITTLENUM_TYPE));
-                             break;
-                           }
-                       }
-                     break;
-                   }
-                 if (exprP.X_add_symbol ||
-                     exprP.X_op_symbol ||
-                     iif.iifP[i].pcrel)
+static void
+fix_new_ns32k (fragS *frag,            /* Which frag? */
+              int where,               /* Where in that frag? */
+              int size,                /* 1, 2  or 4 usually.  */
+              symbolS *add_symbol,     /* X_add_symbol.  */
+              long offset,             /* X_add_number.  */
+              int pcrel,               /* True if PC-relative relocation.  */
+              char im_disp,            /* True if the value to write is a
+                                          displacement.  */
+              bit_fixS *bit_fixP,      /* Pointer at struct of bit_fix's, ignored if
+                                          NULL.  */
+              char bsr,                /* Sequent-linker-hack: 1 when relocobject is
+                                          a bsr.  */
+              fragS *opcode_frag,
+              unsigned int opcode_offset)
+{
+  fixS *fixP = fix_new (frag, where, size, add_symbol,
+                       offset, pcrel,
+#ifdef BFD_ASSEMBLER
+                       bit_fixP ? NO_RELOC : reloc (size, pcrel, im_disp)
+#else
+                       NO_RELOC
+#endif
+                       );
+
+  fix_opcode_frag (fixP) = opcode_frag;
+  fix_opcode_offset (fixP) = opcode_offset;
+  fix_im_disp (fixP) = im_disp;
+  fix_bsr (fixP) = bsr;
+  fix_bit_fixP (fixP) = bit_fixP;
+  /* We have a MD overflow check for displacements.  */
+  fixP->fx_no_overflow = (im_disp != 0);
+}
+
+static void
+fix_new_ns32k_exp (fragS *frag,                /* Which frag? */
+                  int where,           /* Where in that frag? */
+                  int size,            /* 1, 2  or 4 usually.  */
+                  expressionS *exp,    /* Expression.  */
+                  int pcrel,           /* True if PC-relative relocation.  */
+                  char im_disp,        /* True if the value to write is a
+                                          displacement.  */
+                  bit_fixS *bit_fixP,  /* Pointer at struct of bit_fix's, ignored if
+                                          NULL.  */
+                  char bsr,            /* Sequent-linker-hack: 1 when relocobject is
+                                          a bsr.  */
+                  fragS *opcode_frag,
+                  unsigned int opcode_offset)
+{
+  fixS *fixP = fix_new_exp (frag, where, size, exp, pcrel,
+#ifdef BFD_ASSEMBLER
+                           bit_fixP ? NO_RELOC : reloc (size, pcrel, im_disp)
+#else
+                           NO_RELOC
+#endif
+                           );
+
+  fix_opcode_frag (fixP) = opcode_frag;
+  fix_opcode_offset (fixP) = opcode_offset;
+  fix_im_disp (fixP) = im_disp;
+  fix_bsr (fixP) = bsr;
+  fix_bit_fixP (fixP) = bit_fixP;
+  /* We have a MD overflow check for displacements.  */
+  fixP->fx_no_overflow = (im_disp != 0);
+}
+
+/* Convert number to chars in correct order.  */
+
+void
+md_number_to_chars (char *buf, valueT value, int nbytes)
+{
+  number_to_chars_littleendian (buf, value, nbytes);
+}
+
+/* This is a variant of md_numbers_to_chars. The reason for its'
+   existence is the fact that ns32k uses Huffman coded
+   displacements. This implies that the bit order is reversed in
+   displacements and that they are prefixed with a size-tag.
+
+   binary: msb -> lsb
+   0xxxxxxx                            byte
+   10xxxxxx xxxxxxxx                   word
+   11xxxxxx xxxxxxxx xxxxxxxx xxxxxxxx double word
+
+   This must be taken care of and we do it here!  */
+
+static void
+md_number_to_disp (char *buf, long val, int n)
+{
+  switch (n)
+    {
+    case 1:
+      if (val < -64 || val > 63)
+       as_bad (_("value of %ld out of byte displacement range."), val);
+      val &= 0x7f;
+#ifdef SHOW_NUM
+      printf ("%x ", val & 0xff);
+#endif
+      *buf++ = val;
+      break;
+
+    case 2:
+      if (val < -8192 || val > 8191)
+       as_bad (_("value of %ld out of word displacement range."), val);
+      val &= 0x3fff;
+      val |= 0x8000;
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 8 & 0xff);
+#endif
+      *buf++ = (val >> 8);
+#ifdef SHOW_NUM
+      printf ("%x ", val & 0xff);
+#endif
+      *buf++ = val;
+      break;
+
+    case 4:
+      if (val < -0x20000000 || val >= 0x20000000)
+       as_bad (_("value of %ld out of double word displacement range."), val);
+      val |= 0xc0000000;
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 24 & 0xff);
+#endif
+      *buf++ = (val >> 24);
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 16 & 0xff);
+#endif
+      *buf++ = (val >> 16);
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 8 & 0xff);
+#endif
+      *buf++ = (val >> 8);
+#ifdef SHOW_NUM
+      printf ("%x ", val & 0xff);
+#endif
+      *buf++ = val;
+      break;
+
+    default:
+      as_fatal (_("Internal logic error.  line %d, file \"%s\""),
+               __LINE__, __FILE__);
+    }
+}
+
+static void
+md_number_to_imm (char *buf, long val, int n)
+{
+  switch (n)
+    {
+    case 1:
+#ifdef SHOW_NUM
+      printf ("%x ", val & 0xff);
+#endif
+      *buf++ = val;
+      break;
+
+    case 2:
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 8 & 0xff);
+#endif
+      *buf++ = (val >> 8);
+#ifdef SHOW_NUM
+      printf ("%x ", val & 0xff);
+#endif
+      *buf++ = val;
+      break;
+
+    case 4:
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 24 & 0xff);
+#endif
+      *buf++ = (val >> 24);
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 16 & 0xff);
+#endif
+      *buf++ = (val >> 16);
+#ifdef SHOW_NUM
+      printf ("%x ", val >> 8 & 0xff);
+#endif
+      *buf++ = (val >> 8);
+#ifdef SHOW_NUM
+      printf ("%x ", val & 0xff);
+#endif
+      *buf++ = val;
+      break;
+
+    default:
+      as_fatal (_("Internal logic error. line %d, file \"%s\""),
+               __LINE__, __FILE__);
+    }
+}
+
+/* Fast bitfiddling support.  */
+/* Mask used to zero bitfield before oring in the true field.  */
+
+static unsigned long l_mask[] =
+{
+  0xffffffff, 0xfffffffe, 0xfffffffc, 0xfffffff8,
+  0xfffffff0, 0xffffffe0, 0xffffffc0, 0xffffff80,
+  0xffffff00, 0xfffffe00, 0xfffffc00, 0xfffff800,
+  0xfffff000, 0xffffe000, 0xffffc000, 0xffff8000,
+  0xffff0000, 0xfffe0000, 0xfffc0000, 0xfff80000,
+  0xfff00000, 0xffe00000, 0xffc00000, 0xff800000,
+  0xff000000, 0xfe000000, 0xfc000000, 0xf8000000,
+  0xf0000000, 0xe0000000, 0xc0000000, 0x80000000,
+};
+static unsigned long r_mask[] =
+{
+  0x00000000, 0x00000001, 0x00000003, 0x00000007,
+  0x0000000f, 0x0000001f, 0x0000003f, 0x0000007f,
+  0x000000ff, 0x000001ff, 0x000003ff, 0x000007ff,
+  0x00000fff, 0x00001fff, 0x00003fff, 0x00007fff,
+  0x0000ffff, 0x0001ffff, 0x0003ffff, 0x0007ffff,
+  0x000fffff, 0x001fffff, 0x003fffff, 0x007fffff,
+  0x00ffffff, 0x01ffffff, 0x03ffffff, 0x07ffffff,
+  0x0fffffff, 0x1fffffff, 0x3fffffff, 0x7fffffff,
+};
+#define MASK_BITS 31
+/* Insert bitfield described by field_ptr and val at buf
+   This routine is written for modification of the first 4 bytes pointed
+   to by buf, to yield speed.
+   The ifdef stuff is for selection between a ns32k-dependent routine
+   and a general version. (My advice: use the general version!).  */
+
+static void
+md_number_to_field (char *buf, long val, bit_fixS *field_ptr)
+{
+  unsigned long object;
+  unsigned long mask;
+  /* Define ENDIAN on a ns32k machine.  */
+#ifdef ENDIAN
+  unsigned long *mem_ptr;
+#else
+  char *mem_ptr;
+#endif
+
+  if (field_ptr->fx_bit_min <= val && val <= field_ptr->fx_bit_max)
+    {
+#ifdef ENDIAN
+      if (field_ptr->fx_bit_base)
+       /* Override buf.  */
+       mem_ptr = (unsigned long *) field_ptr->fx_bit_base;
+      else
+       mem_ptr = (unsigned long *) buf;
+
+      mem_ptr = ((unsigned long *)
+                ((char *) mem_ptr + field_ptr->fx_bit_base_adj));
+#else
+      if (field_ptr->fx_bit_base)
+       mem_ptr = (char *) field_ptr->fx_bit_base;
+      else
+       mem_ptr = buf;
+
+      mem_ptr += field_ptr->fx_bit_base_adj;
+#endif
+#ifdef ENDIAN
+      /* We have a nice ns32k machine with lowbyte at low-physical mem.  */
+      object = *mem_ptr;       /* get some bytes */
+#else /* OVE Goof! the machine is a m68k or dito.  */
+      /* That takes more byte fiddling.  */
+      object = 0;
+      object |= mem_ptr[3] & 0xff;
+      object <<= 8;
+      object |= mem_ptr[2] & 0xff;
+      object <<= 8;
+      object |= mem_ptr[1] & 0xff;
+      object <<= 8;
+      object |= mem_ptr[0] & 0xff;
+#endif
+      mask = 0;
+      mask |= (r_mask[field_ptr->fx_bit_offset]);
+      mask |= (l_mask[field_ptr->fx_bit_offset + field_ptr->fx_bit_size]);
+      object &= mask;
+      val += field_ptr->fx_bit_add;
+      object |= ((val << field_ptr->fx_bit_offset) & (mask ^ 0xffffffff));
+#ifdef ENDIAN
+      *mem_ptr = object;
+#else
+      mem_ptr[0] = (char) object;
+      object >>= 8;
+      mem_ptr[1] = (char) object;
+      object >>= 8;
+      mem_ptr[2] = (char) object;
+      object >>= 8;
+      mem_ptr[3] = (char) object;
+#endif
+    }
+  else
+    as_bad (_("Bit field out of range"));
+}
+
+/* Convert iif to fragments.  From this point we start to dribble with
+   functions in other files than this one.(Except hash.c) So, if it's
+   possible to make an iif for an other CPU, you don't need to know
+   what frags, relax, obstacks, etc is in order to port this
+   assembler. You only need to know if it's possible to reduce your
+   cpu-instruction to iif-format (takes some work) and adopt the other
+   md_? parts according to given instructions Note that iif was
+   invented for the clean ns32k`s architecture.  */
+
+/* GAS for the ns32k has a problem. PC relative displacements are
+   relative to the address of the opcode, not the address of the
+   operand. We used to keep track of the offset between the operand
+   and the opcode in pcrel_adjust for each frag and each fix. However,
+   we get into trouble where there are two or more pc-relative
+   operands and the size of the first one can't be determined. Then in
+   the relax phase, the size of the first operand will change and
+   pcrel_adjust will no longer be correct.  The current solution is
+   keep a pointer to the frag with the opcode in it and the offset in
+   that frag for each frag and each fix. Then, when needed, we can
+   always figure out how far it is between the opcode and the pcrel
+   object.  See also md_pcrel_adjust and md_fix_pcrel_adjust.  For
+   objects not part of an instruction, the pointer to the opcode frag
+   is always zero.  */
+
+static void
+convert_iif (void)
+{
+  int i;
+  bit_fixS *j;
+  fragS *inst_frag;
+  unsigned int inst_offset;
+  char *inst_opcode;
+  char *memP;
+  int l;
+  int k;
+  char type;
+  char size = 0;
+
+  frag_grow (iif.instr_size);  /* This is important.  */
+  memP = frag_more (0);
+  inst_opcode = memP;
+  inst_offset = (memP - frag_now->fr_literal);
+  inst_frag = frag_now;
+
+  for (i = 0; i < IIF_ENTRIES; i++)
+    {
+      if ((type = iif.iifP[i].type))
+       {
+         /* The object exist, so handle it.  */
+         switch (size = iif.iifP[i].size)
+           {
+           case 42:
+             size = 0;
+             /* It's a bitfix that operates on an existing object.  */
+             if (iif.iifP[i].bit_fixP->fx_bit_base)
+               /* Expand fx_bit_base to point at opcode.  */
+               iif.iifP[i].bit_fixP->fx_bit_base = (long) inst_opcode;
+             /* Fall through.  */
+
+           case 8:             /* bignum or doublefloat.  */
+           case 1:
+           case 2:
+           case 3:
+           case 4:
+             /* The final size in objectmemory is known.  */
+             memP = frag_more (size);
+             j = iif.iifP[i].bit_fixP;
+
+             switch (type)
+               {
+               case 1: /* The object is pure binary.  */
+                 if (j)
+                   md_number_to_field (memP, exprP.X_add_number, j);
+
+                 else if (iif.iifP[i].pcrel)
+                   fix_new_ns32k (frag_now,
+                                  (long) (memP - frag_now->fr_literal),
+                                  size,
+                                  0,
+                                  iif.iifP[i].object,
+                                  iif.iifP[i].pcrel,
+                                  iif.iifP[i].im_disp,
+                                  0,
+                                  iif.iifP[i].bsr,     /* Sequent hack.  */
+                                  inst_frag, inst_offset);
+                 else
+                   {
+                     /* Good, just put them bytes out.  */
+                     switch (iif.iifP[i].im_disp)
+                       {
+                       case 0:
+                         md_number_to_chars (memP, iif.iifP[i].object, size);
+                         break;
+                       case 1:
+                         md_number_to_disp (memP, iif.iifP[i].object, size);
+                         break;
+                       default:
+                         as_fatal (_("iif convert internal pcrel/binary"));
+                       }
+                   }
+                 break;
+
+               case 2:
+                 /* The object is a pointer at an expression, so
+                     unpack it, note that bignums may result from the
+                     expression.  */
+                 evaluate_expr (&exprP, (char *) iif.iifP[i].object);
+                 if (exprP.X_op == O_big || size == 8)
+                   {
+                     if ((k = exprP.X_add_number) > 0)
+                       {
+                         /* We have a bignum ie a quad. This can only
+                             happens in a long suffixed instruction.  */
+                         if (k * 2 > size)
+                           as_bad (_("Bignum too big for long"));
+
+                         if (k == 3)
+                           memP += 2;
+
+                         for (l = 0; k > 0; k--, l += 2)
+                           md_number_to_chars (memP + l,
+                                               generic_bignum[l >> 1],
+                                               sizeof (LITTLENUM_TYPE));
+                       }
+                     else
+                       {
+                         /* flonum.  */
+                         LITTLENUM_TYPE words[4];
+
+                         switch (size)
+                           {
+                           case 4:
+                             gen_to_words (words, 2, 8);
+                             md_number_to_imm (memP, (long) words[0],
+                                               sizeof (LITTLENUM_TYPE));
+                             md_number_to_imm (memP + sizeof (LITTLENUM_TYPE),
+                                               (long) words[1],
+                                               sizeof (LITTLENUM_TYPE));
+                             break;
+                           case 8:
+                             gen_to_words (words, 4, 11);
+                             md_number_to_imm (memP, (long) words[0],
+                                               sizeof (LITTLENUM_TYPE));
+                             md_number_to_imm (memP + sizeof (LITTLENUM_TYPE),
+                                               (long) words[1],
+                                               sizeof (LITTLENUM_TYPE));
+                             md_number_to_imm ((memP + 2
+                                                * sizeof (LITTLENUM_TYPE)),
+                                               (long) words[2],
+                                               sizeof (LITTLENUM_TYPE));
+                             md_number_to_imm ((memP + 3
+                                                * sizeof (LITTLENUM_TYPE)),
+                                               (long) words[3],
+                                               sizeof (LITTLENUM_TYPE));
+                             break;
+                           }
+                       }
+                     break;
+                   }
+                 if (exprP.X_add_symbol ||
+                     exprP.X_op_symbol ||
+                     iif.iifP[i].pcrel)
                    {
                      /* The expression was undefined due to an
                          undefined label. Create a fix so we can fix
@@ -1433,9 +1778,7 @@ convert_iif ()
                                         inst_frag, inst_offset);
                    }
                  else if (j)
-                   {
-                     md_number_to_field(memP, exprP.X_add_number, j);
-                   }
+                   md_number_to_field (memP, exprP.X_add_number, j);
                  else
                    {
                      /* Good, just put them bytes out.  */
@@ -1488,10 +1831,8 @@ convert_iif ()
                  {
                    /* Frag it.  */
                    if (exprP.X_op_symbol)
-                     {
-                       /* We cant relax this case.  */
-                       as_fatal (_("Can't relax difference"));
-                     }
+                     /* We cant relax this case.  */
+                     as_fatal (_("Can't relax difference"));
                    else
                      {
                        /* Size is not important.  This gets fixed by
@@ -1519,23 +1860,17 @@ convert_iif ()
                  {
                    /* This duplicates code in md_number_to_disp.  */
                    if (-64 <= exprP.X_add_number && exprP.X_add_number <= 63)
-                     {
-                       size = 1;
-                     }
+                     size = 1;
                    else
                      {
                        if (-8192 <= exprP.X_add_number
                            && exprP.X_add_number <= 8191)
-                         {
-                           size = 2;
-                         }
+                         size = 2;
                        else
                          {
                            if (-0x20000000 <= exprP.X_add_number
                                && exprP.X_add_number<=0x1fffffff)
-                             {
-                               size = 4;
-                             }
+                             size = 4;
                            else
                              {
                                as_bad (_("Displacement to large for :d"));
@@ -1554,392 +1889,89 @@ convert_iif ()
              as_fatal (_("Internal logic error in iif.iifP[].type"));
            }
        }
-    }
-}
-\f
-#ifdef BFD_ASSEMBLER
-/* This functionality should really be in the bfd library.  */
-static bfd_reloc_code_real_type
-reloc (int size, int pcrel, int type)
-{
-  int length, index;
-  bfd_reloc_code_real_type relocs[] =
-  {
-    BFD_RELOC_NS32K_IMM_8,
-    BFD_RELOC_NS32K_IMM_16,
-    BFD_RELOC_NS32K_IMM_32,
-    BFD_RELOC_NS32K_IMM_8_PCREL,
-    BFD_RELOC_NS32K_IMM_16_PCREL,
-    BFD_RELOC_NS32K_IMM_32_PCREL,
-
-    /* ns32k displacements.  */
-    BFD_RELOC_NS32K_DISP_8,
-    BFD_RELOC_NS32K_DISP_16,
-    BFD_RELOC_NS32K_DISP_32,
-    BFD_RELOC_NS32K_DISP_8_PCREL,
-    BFD_RELOC_NS32K_DISP_16_PCREL,
-    BFD_RELOC_NS32K_DISP_32_PCREL,
-
-    /* Normal 2's complement.  */
-    BFD_RELOC_8,
-    BFD_RELOC_16,
-    BFD_RELOC_32,
-    BFD_RELOC_8_PCREL,
-    BFD_RELOC_16_PCREL,
-    BFD_RELOC_32_PCREL
-  };
-
-  switch (size)
-    {
-    case 1:
-      length = 0;
-      break;
-    case 2:
-      length = 1;
-      break;
-    case 4:
-      length = 2;
-      break;
-    default:
-      length = -1;
-      break;
-    }
-
-  index = length + 3 * pcrel + 6 * type;
-
-  if (index >= 0 && (unsigned int) index < sizeof (relocs) / sizeof (relocs[0]))
-    return relocs[index];
-
-  if (pcrel)
-    as_bad (_("Can not do %d byte pc-relative relocation for storage type %d"),
-           size, type);
-  else
-    as_bad (_("Can not do %d byte relocation for storage type %d"),
-           size, type);
-
-  return BFD_RELOC_NONE;
-
-}
-#endif
-
-void
-md_assemble (line)
-     char *line;
-{
-  freeptr = freeptr_static;
-  parse (line, 0);             /* Explode line to more fix form in iif.  */
-  convert_iif ();              /* Convert iif to frags, fix's etc.  */
-#ifdef SHOW_NUM
-  printf (" \t\t\t%s\n", line);
-#endif
-}
-
-void
-md_begin ()
-{
-  /* Build a hashtable of the instructions.  */
-  const struct ns32k_opcode *ptr;
-  const char *stat;
-  const struct ns32k_opcode *endop;
-
-  inst_hash_handle = hash_new ();
-
-  endop = ns32k_opcodes + sizeof (ns32k_opcodes) / sizeof (ns32k_opcodes[0]);
-  for (ptr = ns32k_opcodes; ptr < endop; ptr++)
-    {
-      if ((stat = hash_insert (inst_hash_handle, ptr->name, (char *) ptr)))
-       /* Fatal.  */
-       as_fatal (_("Can't hash %s: %s"), ptr->name, stat);
-    }
-
-  /* Some private space please!  */
-  freeptr_static = (char *) malloc (PRIVATE_SIZE);
-}
-
-/* Must be equal to MAX_PRECISON in atof-ieee.c.  */
-#define MAX_LITTLENUMS 6
-
-/* Turn the string pointed to by litP into a floating point constant
-   of type TYPE, and emit the appropriate bytes.  The number of
-   LITTLENUMS emitted is stored in *SIZEP.  An error message is
-   returned, or NULL on OK.  */
-
-char *
-md_atof (type, litP, sizeP)
-     char type;
-     char *litP;
-     int *sizeP;
-{
-  int prec;
-  LITTLENUM_TYPE words[MAX_LITTLENUMS];
-  LITTLENUM_TYPE *wordP;
-  char *t;
-
-  switch (type)
-    {
-    case 'f':
-      prec = 2;
-      break;
-
-    case 'd':
-      prec = 4;
-      break;
-    default:
-      *sizeP = 0;
-      return _("Bad call to MD_ATOF()");
-    }
-
-  t = atof_ieee (input_line_pointer, type, words);
-  if (t)
-    input_line_pointer = t;
-
-  *sizeP = prec * sizeof (LITTLENUM_TYPE);
-
-  for (wordP = words + prec; prec--;)
-    {
-      md_number_to_chars (litP, (long) (*--wordP), sizeof (LITTLENUM_TYPE));
-      litP += sizeof (LITTLENUM_TYPE);
-    }
-
-  return 0;
-}
-\f
-/* Convert number to chars in correct order.  */
-
-void
-md_number_to_chars (buf, value, nbytes)
-     char *buf;
-     valueT value;
-     int nbytes;
-{
-  number_to_chars_littleendian (buf, value, nbytes);
-}
-
-/* This is a variant of md_numbers_to_chars. The reason for its'
-   existence is the fact that ns32k uses Huffman coded
-   displacements. This implies that the bit order is reversed in
-   displacements and that they are prefixed with a size-tag.
-
-   binary: msb -> lsb
-   0xxxxxxx                            byte
-   10xxxxxx xxxxxxxx                   word
-   11xxxxxx xxxxxxxx xxxxxxxx xxxxxxxx double word
-
-   This must be taken care of and we do it here!  */
-
-static void
-md_number_to_disp (buf, val, n)
-     char *buf;
-     long val;
-     char n;
-{
-  switch (n)
-    {
-    case 1:
-      if (val < -64 || val > 63)
-       as_bad (_("value of %ld out of byte displacement range."), val);
-      val &= 0x7f;
-#ifdef SHOW_NUM
-      printf ("%x ", val & 0xff);
-#endif
-      *buf++ = val;
-      break;
-    case 2:
-      if (val < -8192 || val > 8191)
-       as_bad (_("value of %ld out of word displacement range."), val);
-      val &= 0x3fff;
-      val |= 0x8000;
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 8 & 0xff);
-#endif
-      *buf++ = (val >> 8);
-#ifdef SHOW_NUM
-      printf ("%x ", val & 0xff);
-#endif
-      *buf++ = val;
-      break;
-    case 4:
-      if (val < -0x20000000 || val >= 0x20000000)
-       as_bad (_("value of %ld out of double word displacement range."), val);
-      val |= 0xc0000000;
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 24 & 0xff);
-#endif
-      *buf++ = (val >> 24);
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 16 & 0xff);
-#endif
-      *buf++ = (val >> 16);
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 8 & 0xff);
-#endif
-      *buf++ = (val >> 8);
+    }
+}
+\f
+void
+md_assemble (char *line)
+{
+  freeptr = freeptr_static;
+  parse (line, 0);             /* Explode line to more fix form in iif.  */
+  convert_iif ();              /* Convert iif to frags, fix's etc.  */
 #ifdef SHOW_NUM
-      printf ("%x ", val & 0xff);
+  printf (" \t\t\t%s\n", line);
 #endif
-      *buf++ = val;
-      break;
-    default:
-      as_fatal (_("Internal logic error.  line %d, file \"%s\""),
-               __LINE__, __FILE__);
-    }
 }
 
-static void
-md_number_to_imm (buf, val, n)
-     char *buf;
-     long val;
-     char n;
+void
+md_begin (void)
 {
-  switch (n)
+  /* Build a hashtable of the instructions.  */
+  const struct ns32k_opcode *ptr;
+  const char *stat;
+  const struct ns32k_opcode *endop;
+
+  inst_hash_handle = hash_new ();
+
+  endop = ns32k_opcodes + sizeof (ns32k_opcodes) / sizeof (ns32k_opcodes[0]);
+  for (ptr = ns32k_opcodes; ptr < endop; ptr++)
     {
-    case 1:
-#ifdef SHOW_NUM
-      printf ("%x ", val & 0xff);
-#endif
-      *buf++ = val;
-      break;
-    case 2:
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 8 & 0xff);
-#endif
-      *buf++ = (val >> 8);
-#ifdef SHOW_NUM
-      printf ("%x ", val & 0xff);
-#endif
-      *buf++ = val;
-      break;
-    case 4:
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 24 & 0xff);
-#endif
-      *buf++ = (val >> 24);
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 16 & 0xff);
-#endif
-      *buf++ = (val >> 16);
-#ifdef SHOW_NUM
-      printf ("%x ", val >> 8 & 0xff);
-#endif
-      *buf++ = (val >> 8);
-#ifdef SHOW_NUM
-      printf ("%x ", val & 0xff);
-#endif
-      *buf++ = val;
-      break;
-    default:
-      as_fatal (_("Internal logic error. line %d, file \"%s\""),
-               __LINE__, __FILE__);
+      if ((stat = hash_insert (inst_hash_handle, ptr->name, (char *) ptr)))
+       /* Fatal.  */
+       as_fatal (_("Can't hash %s: %s"), ptr->name, stat);
     }
+
+  /* Some private space please!  */
+  freeptr_static = (char *) malloc (PRIVATE_SIZE);
 }
 
-/* Fast bitfiddling support.  */
-/* Mask used to zero bitfield before oring in the true field.  */
+/* Must be equal to MAX_PRECISON in atof-ieee.c.  */
+#define MAX_LITTLENUMS 6
 
-static unsigned long l_mask[] =
-{
-  0xffffffff, 0xfffffffe, 0xfffffffc, 0xfffffff8,
-  0xfffffff0, 0xffffffe0, 0xffffffc0, 0xffffff80,
-  0xffffff00, 0xfffffe00, 0xfffffc00, 0xfffff800,
-  0xfffff000, 0xffffe000, 0xffffc000, 0xffff8000,
-  0xffff0000, 0xfffe0000, 0xfffc0000, 0xfff80000,
-  0xfff00000, 0xffe00000, 0xffc00000, 0xff800000,
-  0xff000000, 0xfe000000, 0xfc000000, 0xf8000000,
-  0xf0000000, 0xe0000000, 0xc0000000, 0x80000000,
-};
-static unsigned long r_mask[] =
-{
-  0x00000000, 0x00000001, 0x00000003, 0x00000007,
-  0x0000000f, 0x0000001f, 0x0000003f, 0x0000007f,
-  0x000000ff, 0x000001ff, 0x000003ff, 0x000007ff,
-  0x00000fff, 0x00001fff, 0x00003fff, 0x00007fff,
-  0x0000ffff, 0x0001ffff, 0x0003ffff, 0x0007ffff,
-  0x000fffff, 0x001fffff, 0x003fffff, 0x007fffff,
-  0x00ffffff, 0x01ffffff, 0x03ffffff, 0x07ffffff,
-  0x0fffffff, 0x1fffffff, 0x3fffffff, 0x7fffffff,
-};
-#define MASK_BITS 31
-/* Insert bitfield described by field_ptr and val at buf
-   This routine is written for modification of the first 4 bytes pointed
-   to by buf, to yield speed.
-   The ifdef stuff is for selection between a ns32k-dependent routine
-   and a general version. (My advice: use the general version!).  */
+/* Turn the string pointed to by litP into a floating point constant
+   of type TYPE, and emit the appropriate bytes.  The number of
+   LITTLENUMS emitted is stored in *SIZEP.  An error message is
+   returned, or NULL on OK.  */
 
-static void
-md_number_to_field (buf, val, field_ptr)
-     char *buf;
-     long val;
-     bit_fixS *field_ptr;
+char *
+md_atof (int type, char *litP, int *sizeP)
 {
-  unsigned long object;
-  unsigned long mask;
-  /* Define ENDIAN on a ns32k machine.  */
-#ifdef ENDIAN
-  unsigned long *mem_ptr;
-#else
-  char *mem_ptr;
-#endif
+  int prec;
+  LITTLENUM_TYPE words[MAX_LITTLENUMS];
+  LITTLENUM_TYPE *wordP;
+  char *t;
 
-  if (field_ptr->fx_bit_min <= val && val <= field_ptr->fx_bit_max)
+  switch (type)
     {
-#ifdef ENDIAN
-      if (field_ptr->fx_bit_base)
-       /* Override buf.  */
-       mem_ptr = (unsigned long *) field_ptr->fx_bit_base;
-      else
-       mem_ptr = (unsigned long *) buf;
-
-      mem_ptr = ((unsigned long *)
-                ((char *) mem_ptr + field_ptr->fx_bit_base_adj));
-#else
-      if (field_ptr->fx_bit_base)
-       mem_ptr = (char *) field_ptr->fx_bit_base;
-      else
-       mem_ptr = buf;
+    case 'f':
+      prec = 2;
+      break;
 
-      mem_ptr += field_ptr->fx_bit_base_adj;
-#endif
-#ifdef ENDIAN
-      /* We have a nice ns32k machine with lowbyte at low-physical mem.  */
-      object = *mem_ptr;       /* get some bytes */
-#else /* OVE Goof! the machine is a m68k or dito.  */
-      /* That takes more byte fiddling.  */
-      object = 0;
-      object |= mem_ptr[3] & 0xff;
-      object <<= 8;
-      object |= mem_ptr[2] & 0xff;
-      object <<= 8;
-      object |= mem_ptr[1] & 0xff;
-      object <<= 8;
-      object |= mem_ptr[0] & 0xff;
-#endif
-      mask = 0;
-      mask |= (r_mask[field_ptr->fx_bit_offset]);
-      mask |= (l_mask[field_ptr->fx_bit_offset + field_ptr->fx_bit_size]);
-      object &= mask;
-      val += field_ptr->fx_bit_add;
-      object |= ((val << field_ptr->fx_bit_offset) & (mask ^ 0xffffffff));
-#ifdef ENDIAN
-      *mem_ptr = object;
-#else
-      mem_ptr[0] = (char) object;
-      object >>= 8;
-      mem_ptr[1] = (char) object;
-      object >>= 8;
-      mem_ptr[2] = (char) object;
-      object >>= 8;
-      mem_ptr[3] = (char) object;
-#endif
+    case 'd':
+      prec = 4;
+      break;
+    default:
+      *sizeP = 0;
+      return _("Bad call to MD_ATOF()");
     }
-  else
+
+  t = atof_ieee (input_line_pointer, type, words);
+  if (t)
+    input_line_pointer = t;
+
+  *sizeP = prec * sizeof (LITTLENUM_TYPE);
+
+  for (wordP = words + prec; prec--;)
     {
-      as_bad (_("Bit field out of range"));
+      md_number_to_chars (litP, (long) (*--wordP), sizeof (LITTLENUM_TYPE));
+      litP += sizeof (LITTLENUM_TYPE);
     }
-}
 
+  return 0;
+}
+\f
 int
-md_pcrel_adjust (fragP)
-     fragS *fragP;
+md_pcrel_adjust (fragS *fragP)
 {
   fragS *opcode_frag;
   addressT opcode_address;
@@ -1955,10 +1987,8 @@ md_pcrel_adjust (fragP)
   return fragP->fr_address + fragP->fr_fix - opcode_address;
 }
 
-static int md_fix_pcrel_adjust PARAMS ((fixS *fixP));
 static int
-md_fix_pcrel_adjust (fixP)
-     fixS *fixP;
+md_fix_pcrel_adjust (fixS *fixP)
 {
   fragS *opcode_frag;
   addressT opcode_address;
@@ -1982,19 +2012,14 @@ md_fix_pcrel_adjust (fixP)
    They all get called from here.  */
 
 void
-md_apply_fix3 (fixP, valP, seg)
-     fixS *fixP;
-     valueT * valP;
-     segT seg ATTRIBUTE_UNUSED;
+md_apply_fix3 (fixS *fixP, valueT * valP, segT seg ATTRIBUTE_UNUSED)
 {
   long val = * (long *) valP;
   char *buf = fixP->fx_where + fixP->fx_frag->fr_literal;
 
   if (fix_bit_fixP (fixP))
-    {
-      /* Bitfields to fix, sigh.  */
-      md_number_to_field (buf, val, fix_bit_fixP (fixP));
-    }
+    /* Bitfields to fix, sigh.  */
+    md_number_to_field (buf, val, fix_bit_fixP (fixP));
   else switch (fix_im_disp (fixP))
     {
     case 0:
@@ -2024,16 +2049,14 @@ md_apply_fix3 (fixP, valP, seg)
 
 #ifndef BFD_ASSEMBLER
 void
-md_convert_frag (headers, sec, fragP)
-     object_headers *headers;
-     segT sec;
-     fragS *fragP;
+md_convert_frag (object_headers *headers,
+                segT sec,
+                fragS *fragP)
 #else
 void
-md_convert_frag (abfd, sec, fragP)
-     bfd *abfd ATTRIBUTE_UNUSED;
-     segT sec ATTRIBUTE_UNUSED;
-     fragS *fragP;
+md_convert_frag (bfd *abfd ATTRIBUTE_UNUSED,
+                segT sec ATTRIBUTE_UNUSED,
+                fragS *fragP)
 #endif
 {
   long disp;
@@ -2076,9 +2099,7 @@ md_convert_frag (abfd, sec, fragP)
    actually know it.  */
 
 int
-md_estimate_size_before_relax (fragP, segment)
-     fragS *fragP;
-     segT segment;
+md_estimate_size_before_relax (fragS *fragP, segT segment)
 {
   if (fragP->fr_subtype == IND (BRANCH, UNDEF))
     {
@@ -2119,11 +2140,11 @@ int md_long_jump_size = 5;
 const int md_reloc_size = 8;   /* Size of relocation record.  */
 
 void
-md_create_short_jump (ptr, from_addr, to_addr, frag, to_symbol)
-     char *ptr;
-     addressT from_addr, to_addr;
-     fragS *frag ATTRIBUTE_UNUSED;
-     symbolS *to_symbol ATTRIBUTE_UNUSED;
+md_create_short_jump (char *ptr,
+                     addressT from_addr,
+                     addressT to_addr,
+                     fragS *frag ATTRIBUTE_UNUSED,
+                     symbolS *to_symbol ATTRIBUTE_UNUSED)
 {
   valueT offset;
 
@@ -2133,11 +2154,11 @@ md_create_short_jump (ptr, from_addr, to_addr, frag, to_symbol)
 }
 
 void
-md_create_long_jump (ptr, from_addr, to_addr, frag, to_symbol)
-     char *ptr;
-     addressT from_addr, to_addr;
-     fragS *frag ATTRIBUTE_UNUSED;
-     symbolS *to_symbol ATTRIBUTE_UNUSED;
+md_create_long_jump (char *ptr,
+                    addressT from_addr,
+                    addressT to_addr,
+                    fragS *frag ATTRIBUTE_UNUSED,
+                    symbolS *to_symbol ATTRIBUTE_UNUSED)
 {
   valueT offset;
 
@@ -2158,9 +2179,7 @@ struct option md_longopts[] =
 size_t md_longopts_size = sizeof (md_longopts);
 
 int
-md_parse_option (c, arg)
-     int c;
-     char *arg;
+md_parse_option (int c, char *arg)
 {
   switch (c)
     {
@@ -2204,8 +2223,7 @@ md_parse_option (c, arg)
 }
 
 void
-md_show_usage (stream)
-     FILE *stream;
+md_show_usage (FILE *stream)
 {
   fprintf (stream, _("\
 NS32K options:\n\
@@ -2213,114 +2231,13 @@ NS32K options:\n\
 --disp-size-default=<1|2|4>\n"));
 }
 \f
-/* Create a bit_fixS in obstack 'notes'.
-   This struct is used to profile the normal fix. If the bit_fixP is a
-   valid pointer (not NULL) the bit_fix data will be used to format
-   the fix.  */
-
-bit_fixS *
-bit_fix_new (size, offset, min, max, add, base_type, base_adj)
-     char size;                        /* Length of bitfield.  */
-     char offset;              /* Bit offset to bitfield.  */
-     long min;                 /* Signextended min for bitfield.  */
-     long max;                 /* Signextended max for bitfield.  */
-     long add;                 /* Add mask, used for huffman prefix.  */
-     long base_type;           /* 0 or 1, if 1 it's exploded to opcode ptr.  */
-     long base_adj;
-{
-  bit_fixS *bit_fixP;
-
-  bit_fixP = (bit_fixS *) obstack_alloc (&notes, sizeof (bit_fixS));
-
-  bit_fixP->fx_bit_size = size;
-  bit_fixP->fx_bit_offset = offset;
-  bit_fixP->fx_bit_base = base_type;
-  bit_fixP->fx_bit_base_adj = base_adj;
-  bit_fixP->fx_bit_max = max;
-  bit_fixP->fx_bit_min = min;
-  bit_fixP->fx_bit_add = add;
-
-  return bit_fixP;
-}
-
-void
-fix_new_ns32k (frag, where, size, add_symbol, offset, pcrel,
-              im_disp, bit_fixP, bsr, opcode_frag, opcode_offset)
-     fragS *frag;              /* Which frag? */
-     int where;                        /* Where in that frag? */
-     int size;                 /* 1, 2  or 4 usually.  */
-     symbolS *add_symbol;      /* X_add_symbol.  */
-     long offset;              /* X_add_number.  */
-     int pcrel;                        /* True if PC-relative relocation.  */
-     char im_disp;             /* True if the value to write is a
-                                  displacement.  */
-     bit_fixS *bit_fixP;       /* Pointer at struct of bit_fix's, ignored if
-                                  NULL.  */
-     char bsr;                 /* Sequent-linker-hack: 1 when relocobject is
-                                  a bsr.  */
-     fragS *opcode_frag;
-     unsigned int opcode_offset;
-{
-  fixS *fixP = fix_new (frag, where, size, add_symbol,
-                       offset, pcrel,
-#ifdef BFD_ASSEMBLER
-                       bit_fixP ? NO_RELOC : reloc (size, pcrel, im_disp)
-#else
-                       NO_RELOC
-#endif
-                       );
-
-  fix_opcode_frag (fixP) = opcode_frag;
-  fix_opcode_offset (fixP) = opcode_offset;
-  fix_im_disp (fixP) = im_disp;
-  fix_bsr (fixP) = bsr;
-  fix_bit_fixP (fixP) = bit_fixP;
-  /* We have a MD overflow check for displacements.  */
-  fixP->fx_no_overflow = (im_disp != 0);
-}
-
-void
-fix_new_ns32k_exp (frag, where, size, exp, pcrel,
-                  im_disp, bit_fixP, bsr, opcode_frag, opcode_offset)
-     fragS *frag;              /* Which frag? */
-     int where;                        /* Where in that frag? */
-     int size;                 /* 1, 2  or 4 usually.  */
-     expressionS *exp;         /* Expression.  */
-     int pcrel;                        /* True if PC-relative relocation.  */
-     char im_disp;             /* True if the value to write is a
-                                  displacement.  */
-     bit_fixS *bit_fixP;       /* Pointer at struct of bit_fix's, ignored if
-                                  NULL.  */
-     char bsr;                 /* Sequent-linker-hack: 1 when relocobject is
-                                  a bsr.  */
-     fragS *opcode_frag;
-     unsigned int opcode_offset;
-{
-  fixS *fixP = fix_new_exp (frag, where, size, exp, pcrel,
-#ifdef BFD_ASSEMBLER
-                           bit_fixP ? NO_RELOC : reloc (size, pcrel, im_disp)
-#else
-                           NO_RELOC
-#endif
-                           );
-
-  fix_opcode_frag (fixP) = opcode_frag;
-  fix_opcode_offset (fixP) = opcode_offset;
-  fix_im_disp (fixP) = im_disp;
-  fix_bsr (fixP) = bsr;
-  fix_bit_fixP (fixP) = bit_fixP;
-  /* We have a MD overflow check for displacements.  */
-  fixP->fx_no_overflow = (im_disp != 0);
-}
-
 /* This is TC_CONS_FIX_NEW, called by emit_expr in read.c.  */
 
 void
-cons_fix_new_ns32k (frag, where, size, exp)
-     fragS *frag;              /* Which frag? */
-     int where;                        /* Where in that frag? */
-     int size;                 /* 1, 2  or 4 usually.  */
-     expressionS *exp;         /* Expression.  */
+cons_fix_new_ns32k (fragS *frag,       /* Which frag? */
+                   int where,          /* Where in that frag? */
+                   int size,           /* 1, 2  or 4 usually.  */
+                   expressionS *exp)   /* Expression.  */
 {
   fix_new_ns32k_exp (frag, where, size, exp,
                     0, 2, 0, 0, 0, 0);
@@ -2329,8 +2246,7 @@ cons_fix_new_ns32k (frag, where, size, exp)
 /* We have no need to default values of symbols.  */
 
 symbolS *
-md_undefined_symbol (name)
-     char *name ATTRIBUTE_UNUSED;
+md_undefined_symbol (char *name ATTRIBUTE_UNUSED)
 {
   return 0;
 }
@@ -2338,9 +2254,7 @@ md_undefined_symbol (name)
 /* Round up a section size to the appropriate boundary.  */
 
 valueT
-md_section_align (segment, size)
-     segT segment ATTRIBUTE_UNUSED;
-     valueT size;
+md_section_align (segT segment ATTRIBUTE_UNUSED, valueT size)
 {
   return size;                 /* Byte alignment is fine.  */
 }
@@ -2349,8 +2263,7 @@ md_section_align (segment, size)
    ns32k, they're relative to the start of the instruction.  */
 
 long
-md_pcrel_from (fixP)
-     fixS *fixP;
+md_pcrel_from (fixS *fixP)
 {
   long res;
 
@@ -2365,17 +2278,15 @@ md_pcrel_from (fixP)
 #ifdef BFD_ASSEMBLER
 
 arelent *
-tc_gen_reloc (section, fixp)
-     asection *section ATTRIBUTE_UNUSED;
-     fixS *fixp;
+tc_gen_reloc (asection *section ATTRIBUTE_UNUSED, fixS *fixp)
 {
   arelent *rel;
   bfd_reloc_code_real_type code;
 
   code = reloc (fixp->fx_size, fixp->fx_pcrel, fix_im_disp (fixp));
 
-  rel = (arelent *) xmalloc (sizeof (arelent));
-  rel->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
+  rel = xmalloc (sizeof (arelent));
+  rel->sym_ptr_ptr = xmalloc (sizeof (asymbol *));
   *rel->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
   rel->address = fixp->fx_frag->fr_address + fixp->fx_where;
   if (fixp->fx_pcrel)
@@ -2401,10 +2312,9 @@ tc_gen_reloc (section, fixp)
 
 #ifdef OBJ_AOUT
 void
-cons_fix_new_ns32k (where, fixP, segment_address_in_file)
-     char *where;
-     struct fix *fixP;
-     relax_addressT segment_address_in_file;
+cons_fix_new_ns32k (char *where,
+                   struct fix *fixP,
+                   relax_addressT segment_address_in_file)
 {
   /* In:  Length of relocation (or of address) in chars: 1, 2 or 4.
      Out: GNU LD relocation length code: 0, 1, or 2.  */