use xstrdup, xmemdup0 and concat more
[deliverable/binutils-gdb.git] / gas / config / tc-hppa.c
index f87b57a12df3f50134bf54deffc37e2e9ba7ef49..aea78dc953688035809ffd224b564e0bf5cacb0b 100644 (file)
@@ -1,12 +1,11 @@
 /* tc-hppa.c -- Assemble for the PA
 /* tc-hppa.c -- Assemble for the PA
-   Copyright 1989, 93, 94, 95, 96, 97, 98, 99, 2000, 2001
-   Free Software Foundation, Inc.
+   Copyright (C) 1989-2016 Free Software Foundation, Inc.
 
    This file is part of GAS, the GNU Assembler.
 
    GAS is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
 
    This file is part of GAS, the GNU Assembler.
 
    GAS is free software; you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
-   the Free Software Foundation; either version 2, or (at your option)
+   the Free Software Foundation; either version 3, or (at your option)
    any later version.
 
    GAS is distributed in the hope that it will be useful,
    any later version.
 
    GAS is distributed in the hope that it will be useful,
 
    You should have received a copy of the GNU General Public License
    along with GAS; see the file COPYING.  If not, write to the Free
 
    You should have received a copy of the GNU General Public License
    along with GAS; see the file COPYING.  If not, write to the Free
-   Software Foundation, 59 Temple Place - Suite 330, Boston, MA
-   02111-1307, USA.  */
+   Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
+   02110-1301, USA.  */
 
 /* HP PA-RISC support was contributed by the Center for Software Science
    at the University of Utah.  */
 
 
 /* HP PA-RISC support was contributed by the Center for Software Science
    at the University of Utah.  */
 
-#include <stdio.h>
-#include <ctype.h>
-
 #include "as.h"
 #include "as.h"
+#include "safe-ctype.h"
 #include "subsegs.h"
 #include "subsegs.h"
+#include "dw2gencfi.h"
 
 #include "bfd/libhppa.h"
 
 
 #include "bfd/libhppa.h"
 
@@ -43,7 +41,7 @@ error only one of OBJ_ELF and OBJ_SOM can be defined
 #ifdef OBJ_ELF
 #include "dwarf2dbg.h"
 
 #ifdef OBJ_ELF
 #include "dwarf2dbg.h"
 
-/* A "convient" place to put object file dependencies which do
+/* A "convenient" place to put object file dependencies which do
    not need to be seen outside of tc-hppa.c.  */
 
 /* Object file formats specify relocation types.  */
    not need to be seen outside of tc-hppa.c.  */
 
 /* Object file formats specify relocation types.  */
@@ -57,8 +55,10 @@ typedef elf_symbol_type obj_symbol_type;
 #if TARGET_ARCH_SIZE == 64
 /* How to generate a relocation.  */
 #define hppa_gen_reloc_type _bfd_elf64_hppa_gen_reloc_type
 #if TARGET_ARCH_SIZE == 64
 /* How to generate a relocation.  */
 #define hppa_gen_reloc_type _bfd_elf64_hppa_gen_reloc_type
+#define elf_hppa_reloc_final_type elf64_hppa_reloc_final_type
 #else
 #define hppa_gen_reloc_type _bfd_elf32_hppa_gen_reloc_type
 #else
 #define hppa_gen_reloc_type _bfd_elf32_hppa_gen_reloc_type
+#define elf_hppa_reloc_final_type elf32_hppa_reloc_final_type
 #endif
 
 /* ELF objects can have versions, but apparently do not have anywhere
 #endif
 
 /* ELF objects can have versions, but apparently do not have anywhere
@@ -105,6 +105,12 @@ typedef som_symbol_type obj_symbol_type;
 #endif
 #endif /* OBJ_SOM */
 
 #endif
 #endif /* OBJ_SOM */
 
+#if TARGET_ARCH_SIZE == 64
+#define DEFAULT_LEVEL 25
+#else
+#define DEFAULT_LEVEL 10
+#endif
+
 /* Various structures and types used internally in tc-hppa.c.  */
 
 /* Unwind table and descriptor.  FIXME: Sync this with GDB version.  */
 /* Various structures and types used internally in tc-hppa.c.  */
 
 /* Unwind table and descriptor.  FIXME: Sync this with GDB version.  */
@@ -133,6 +139,31 @@ struct unwind_desc
     unsigned int frame_size:27;
   };
 
     unsigned int frame_size:27;
   };
 
+/* We can't rely on compilers placing bitfields in any particular
+   place, so use these macros when dumping unwind descriptors to
+   object files.  */
+#define UNWIND_LOW32(U) \
+  (((U)->cannot_unwind << 31)          \
+   | ((U)->millicode << 30)            \
+   | ((U)->millicode_save_rest << 29)  \
+   | ((U)->region_desc << 27)          \
+   | ((U)->save_sr << 25)              \
+   | ((U)->entry_fr << 21)             \
+   | ((U)->entry_gr << 16)             \
+   | ((U)->args_stored << 15)          \
+   | ((U)->call_fr << 10)              \
+   | ((U)->call_gr << 5)               \
+   | ((U)->save_sp << 4)               \
+   | ((U)->save_rp << 3)               \
+   | ((U)->save_rp_in_frame << 2)      \
+   | ((U)->extn_ptr_defined << 1)      \
+   | ((U)->cleanup_defined << 0))
+
+#define UNWIND_HIGH32(U) \
+  (((U)->hpe_interrupt_marker << 31)   \
+   | ((U)->hpux_interrupt_marker << 30)        \
+   | ((U)->frame_size << 0))
+
 struct unwind_table
   {
     /* Starting and ending offsets of the region described by
 struct unwind_table
   {
     /* Starting and ending offsets of the region described by
@@ -319,7 +350,7 @@ typedef struct space_dictionary_chain sd_chain_struct;
 struct default_subspace_dict
   {
     /* Name of the subspace.  */
 struct default_subspace_dict
   {
     /* Name of the subspace.  */
-    char *name;
+    const char *name;
 
     /* FIXME.  Is this still needed?  */
     char defined;
 
     /* FIXME.  Is this still needed?  */
     char defined;
@@ -330,6 +361,9 @@ struct default_subspace_dict
     /* Nonzero if this subspace contains only code.  */
     char code_only;
 
     /* Nonzero if this subspace contains only code.  */
     char code_only;
 
+    /* Nonzero if this is a comdat subspace.  */
+    char comdat;
+
     /* Nonzero if this is a common subspace.  */
     char common;
 
     /* Nonzero if this is a common subspace.  */
     char common;
 
@@ -369,7 +403,7 @@ struct default_subspace_dict
 struct default_space_dict
   {
     /* Name of the space.  */
 struct default_space_dict
   {
     /* Name of the space.  */
-    char *name;
+    const char *name;
 
     /* Space number.  It is possible to identify spaces within
        assembly code numerically!  */
 
     /* Space number.  It is possible to identify spaces within
        assembly code numerically!  */
@@ -431,7 +465,7 @@ struct hppa_fix_struct
 
 struct pd_reg
   {
 
 struct pd_reg
   {
-    char *name;
+    const char *name;
     int value;
   };
 
     int value;
   };
 
@@ -439,7 +473,7 @@ struct pd_reg
    to a condition number which can be recorded in an instruction.  */
 struct fp_cond_map
   {
    to a condition number which can be recorded in an instruction.  */
 struct fp_cond_map
   {
-    char *string;
+    const char *string;
     int cond;
   };
 
     int cond;
   };
 
@@ -447,123 +481,52 @@ struct fp_cond_map
    string to a field selector type.  */
 struct selector_entry
   {
    string to a field selector type.  */
 struct selector_entry
   {
-    char *prefix;
+    const char *prefix;
     int field_selector;
   };
 
 /* Prototypes for functions local to tc-hppa.c.  */
 
 #ifdef OBJ_SOM
     int field_selector;
   };
 
 /* Prototypes for functions local to tc-hppa.c.  */
 
 #ifdef OBJ_SOM
-static void pa_check_current_space_and_subspace PARAMS ((void));
+static void pa_check_current_space_and_subspace (void);
 #endif
 
 #endif
 
-#if !(defined (OBJ_ELF) && defined (TE_LINUX))
-static void pa_text PARAMS ((int));
-static void pa_data PARAMS ((int));
-static void pa_comm PARAMS ((int));
+#if !(defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD)))
+static void pa_text (int);
+static void pa_data (int);
+static void pa_comm (int);
 #endif
 #endif
-static fp_operand_format pa_parse_fp_format PARAMS ((char **s));
-static void pa_cons PARAMS ((int));
-static void pa_float_cons PARAMS ((int));
-static void pa_fill PARAMS ((int));
-static void pa_lcomm PARAMS ((int));
-static void pa_lsym PARAMS ((int));
-static void pa_stringer PARAMS ((int));
-static void pa_version PARAMS ((int));
-static int pa_parse_fp_cmp_cond PARAMS ((char **));
-static int get_expression PARAMS ((char *));
-static int pa_get_absolute_expression PARAMS ((struct pa_it *, char **));
-static int evaluate_absolute PARAMS ((struct pa_it *));
-static unsigned int pa_build_arg_reloc PARAMS ((char *));
-static unsigned int pa_align_arg_reloc PARAMS ((unsigned int, unsigned int));
-static int pa_parse_nullif PARAMS ((char **));
-static int pa_parse_nonneg_cmpsub_cmpltr PARAMS ((char **, int));
-static int pa_parse_neg_cmpsub_cmpltr PARAMS ((char **, int));
-static int pa_parse_neg_add_cmpltr PARAMS ((char **, int));
-static int pa_parse_nonneg_add_cmpltr PARAMS ((char **, int));
-static int pa_parse_cmpb_64_cmpltr PARAMS ((char **));
-static int pa_parse_cmpib_64_cmpltr PARAMS ((char **));
-static int pa_parse_addb_64_cmpltr PARAMS ((char **));
-static void pa_block PARAMS ((int));
-static void pa_brtab PARAMS ((int));
-static void pa_try PARAMS ((int));
-static void pa_call PARAMS ((int));
-static void pa_call_args PARAMS ((struct call_desc *));
-static void pa_callinfo PARAMS ((int));
-static void pa_copyright PARAMS ((int));
-static void pa_end PARAMS ((int));
-static void pa_enter PARAMS ((int));
-static void pa_entry PARAMS ((int));
-static void pa_equ PARAMS ((int));
-static void pa_exit PARAMS ((int));
-static void pa_export PARAMS ((int));
-static void pa_type_args PARAMS ((symbolS *, int));
-static void pa_import PARAMS ((int));
-static void pa_label PARAMS ((int));
-static void pa_leave PARAMS ((int));
-static void pa_level PARAMS ((int));
-static void pa_origin PARAMS ((int));
-static void pa_proc PARAMS ((int));
-static void pa_procend PARAMS ((int));
-static void pa_param PARAMS ((int));
-static void pa_undefine_label PARAMS ((void));
-static int need_pa11_opcode PARAMS ((void));
-static int pa_parse_number PARAMS ((char **, int));
-static label_symbol_struct *pa_get_label PARAMS ((void));
 #ifdef OBJ_SOM
 #ifdef OBJ_SOM
-static int log2 PARAMS ((int));
-static void pa_compiler PARAMS ((int));
-static void pa_align PARAMS ((int));
-static void pa_space PARAMS ((int));
-static void pa_spnum PARAMS ((int));
-static void pa_subspace PARAMS ((int));
-static sd_chain_struct *create_new_space PARAMS ((char *, int, int,
-                                                 int, int, int,
-                                                 asection *, int));
-static ssd_chain_struct *create_new_subspace PARAMS ((sd_chain_struct *,
-                                                     char *, int, int,
-                                                     int, int, int,
-                                                     int, int, int, int,
-                                                     int, asection *));
-static ssd_chain_struct *update_subspace PARAMS ((sd_chain_struct *,
-                                                 char *, int, int, int,
-                                                 int, int, int, int,
-                                                 int, int, int,
-                                                 asection *));
-static sd_chain_struct *is_defined_space PARAMS ((char *));
-static ssd_chain_struct *is_defined_subspace PARAMS ((char *));
-static sd_chain_struct *pa_segment_to_space PARAMS ((asection *));
-static ssd_chain_struct *pa_subsegment_to_subspace PARAMS ((asection *,
-                                                           subsegT));
-static sd_chain_struct *pa_find_space_by_number PARAMS ((int));
-static unsigned int pa_subspace_start PARAMS ((sd_chain_struct *, int));
-static sd_chain_struct *pa_parse_space_stmt PARAMS ((char *, int));
-static int pa_next_subseg PARAMS ((sd_chain_struct *));
-static void pa_spaces_begin PARAMS ((void));
-#endif
-static void pa_ip PARAMS ((char *));
-static void fix_new_hppa PARAMS ((fragS *, int, int, symbolS *,
-                                 offsetT, expressionS *, int,
-                                 bfd_reloc_code_real_type,
-                                 enum hppa_reloc_field_selector_type_alt,
-                                 int, unsigned int, int *));
-static int is_end_of_statement PARAMS ((void));
-static int reg_name_search PARAMS ((char *));
-static int pa_chk_field_selector PARAMS ((char **));
-static int is_same_frag PARAMS ((fragS *, fragS *));
-static void process_exit PARAMS ((void));
-static unsigned int pa_stringer_aux PARAMS ((char *));
-static fp_operand_format pa_parse_fp_cnv_format PARAMS ((char **s));
-static int pa_parse_ftest_gfx_completer PARAMS ((char **));
-
-#ifdef OBJ_ELF
-static void hppa_elf_mark_end_of_function PARAMS ((void));
-static void pa_build_unwind_subspace PARAMS ((struct call_info *));
-static void pa_vtable_entry PARAMS ((int));
-static void pa_vtable_inherit  PARAMS ((int));
+static int exact_log2 (int);
+static void pa_compiler (int);
+static void pa_align (int);
+static void pa_space (int);
+static void pa_spnum (int);
+static void pa_subspace (int);
+static sd_chain_struct *create_new_space (const char *, int, int,
+                                         int, int, int,
+                                         asection *, int);
+static ssd_chain_struct *create_new_subspace (sd_chain_struct *,
+                                             const char *, int, int,
+                                             int, int, int, int,
+                                             int, int, int, int,
+                                             int, asection *);
+static ssd_chain_struct *update_subspace (sd_chain_struct *,
+                                         char *, int, int, int,
+                                         int, int, int, int,
+                                         int, int, int, int,
+                                         asection *);
+static sd_chain_struct *is_defined_space (const char *);
+static ssd_chain_struct *is_defined_subspace (const char *);
+static sd_chain_struct *pa_segment_to_space (asection *);
+static ssd_chain_struct *pa_subsegment_to_subspace (asection *,
+                                                           subsegT);
+static sd_chain_struct *pa_find_space_by_number (int);
+static unsigned int pa_subspace_start (sd_chain_struct *, int);
+static sd_chain_struct *pa_parse_space_stmt (const char *, int);
 #endif
 
 #endif
 
-/* File and gloally scoped variable declarations.  */
+/* File and globally scoped variable declarations.  */
 
 #ifdef OBJ_SOM
 /* Root and final entry in the space chain.  */
 
 #ifdef OBJ_SOM
 /* Root and final entry in the space chain.  */
@@ -589,101 +552,10 @@ static struct call_desc last_call_desc;
 /* handle of the OPCODE hash table */
 static struct hash_control *op_hash = NULL;
 
 /* handle of the OPCODE hash table */
 static struct hash_control *op_hash = NULL;
 
-/* Table of pseudo ops for the PA.  FIXME -- how many of these
-   are now redundant with the overall GAS and the object file
-   dependent tables?  */
-const pseudo_typeS md_pseudo_table[] =
-{
-  /* align pseudo-ops on the PA specify the actual alignment requested,
-     not the log2 of the requested alignment.  */
-#ifdef OBJ_SOM
-  {"align", pa_align, 8},
-#endif
-#ifdef OBJ_ELF
-  {"align", s_align_bytes, 8},
-#endif
-  {"begin_brtab", pa_brtab, 1},
-  {"begin_try", pa_try, 1},
-  {"block", pa_block, 1},
-  {"blockz", pa_block, 0},
-  {"byte", pa_cons, 1},
-  {"call", pa_call, 0},
-  {"callinfo", pa_callinfo, 0},
-#if defined (OBJ_ELF) && defined (TE_LINUX)
-  {"code", obj_elf_text, 0},
-#else
-  {"code", pa_text, 0},
-  {"comm", pa_comm, 0},
-#endif
-#ifdef OBJ_SOM
-  {"compiler", pa_compiler, 0},
-#endif
-  {"copyright", pa_copyright, 0},
-#if !(defined (OBJ_ELF) && defined (TE_LINUX))
-  {"data", pa_data, 0},
-#endif
-  {"double", pa_float_cons, 'd'},
-  {"dword", pa_cons, 8},
-  {"end", pa_end, 0},
-  {"end_brtab", pa_brtab, 0},
-#if !(defined (OBJ_ELF) && defined (TE_LINUX))
-  {"end_try", pa_try, 0},
-#endif
-  {"enter", pa_enter, 0},
-  {"entry", pa_entry, 0},
-  {"equ", pa_equ, 0},
-  {"exit", pa_exit, 0},
-  {"export", pa_export, 0},
-#ifdef OBJ_ELF
-  {"file", dwarf2_directive_file, 0 },
-#endif
-  {"fill", pa_fill, 0},
-  {"float", pa_float_cons, 'f'},
-  {"half", pa_cons, 2},
-  {"import", pa_import, 0},
-  {"int", pa_cons, 4},
-  {"label", pa_label, 0},
-  {"lcomm", pa_lcomm, 0},
-  {"leave", pa_leave, 0},
-  {"level", pa_level, 0},
-#ifdef OBJ_ELF
-  {"loc", dwarf2_directive_loc, 0 },
-#endif
-  {"long", pa_cons, 4},
-  {"lsym", pa_lsym, 0},
-#ifdef OBJ_SOM
-  {"nsubspa", pa_subspace, 1},
-#endif
-  {"octa", pa_cons, 16},
-  {"org", pa_origin, 0},
-  {"origin", pa_origin, 0},
-  {"param", pa_param, 0},
-  {"proc", pa_proc, 0},
-  {"procend", pa_procend, 0},
-  {"quad", pa_cons, 8},
-  {"reg", pa_equ, 1},
-  {"short", pa_cons, 2},
-  {"single", pa_float_cons, 'f'},
-#ifdef OBJ_SOM
-  {"space", pa_space, 0},
-  {"spnum", pa_spnum, 0},
-#endif
-  {"string", pa_stringer, 0},
-  {"stringz", pa_stringer, 1},
-#ifdef OBJ_SOM
-  {"subspa", pa_subspace, 0},
-#endif
-#if !(defined (OBJ_ELF) && defined (TE_LINUX))
-  {"text", pa_text, 0},
-#endif
-  {"version", pa_version, 0},
-#ifdef OBJ_ELF
-  {"vtable_entry", pa_vtable_entry, 0},
-  {"vtable_inherit", pa_vtable_inherit, 0},
-#endif
-  {"word", pa_cons, 4},
-  {NULL, 0, 0}
-};
+/* These characters can be suffixes of opcode names and they may be
+   followed by meaningful whitespace.  We don't include `,' and `!'
+   as they never appear followed by meaningful whitespace.  */
+const char hppa_symbol_chars[] = "*?=<>";
 
 /* This array holds the chars that only start a comment at the beginning of
    a line.  If the line seems to have the form '# 123 filename'
 
 /* This array holds the chars that only start a comment at the beginning of
    a line.  If the line seems to have the form '# 123 filename'
@@ -710,13 +582,13 @@ const char EXP_CHARS[] = "eE";
    As in 0f12.456 or 0d1.2345e12.
 
    Be aware that MAXIMUM_NUMBER_OF_CHARS_FOR_FLOAT may have to be
    As in 0f12.456 or 0d1.2345e12.
 
    Be aware that MAXIMUM_NUMBER_OF_CHARS_FOR_FLOAT may have to be
-   changed in read.c.  Ideally it shouldn't hae to know abou it at
-   all, but nothing is ideal around here.  */
+   changed in read.c.  Ideally it shouldn't have to know about it
+   at all, but nothing is ideal around here.  */
 const char FLT_CHARS[] = "rRsSfFdDxXpP";
 
 static struct pa_it the_insn;
 
 const char FLT_CHARS[] = "rRsSfFdDxXpP";
 
 static struct pa_it the_insn;
 
-/* Points to the end of an expression just parsed by get_expressoin
+/* Points to the end of an expression just parsed by get_expression
    and friends.  FIXME.  This shouldn't be handled with a file-global
    variable.  */
 static char *expr_end;
    and friends.  FIXME.  This shouldn't be handled with a file-global
    variable.  */
 static char *expr_end;
@@ -734,14 +606,18 @@ static int within_procedure;
    seen in each subspace.  */
 static label_symbol_struct *label_symbols_rootp = NULL;
 
    seen in each subspace.  */
 static label_symbol_struct *label_symbols_rootp = NULL;
 
-/* Holds the last field selector.  */
-static int hppa_field_selector;
+/* Last label symbol */
+static label_symbol_struct last_label_symbol;
+
+/* Nonzero when strict matching is enabled.  Zero otherwise.
 
 
-/* Nonzero when strict syntax checking is enabled.  Zero otherwise.
+   Each opcode in the table has a flag which indicates whether or
+   not strict matching should be enabled for that instruction.
 
 
-   Each opcode in the table has a flag which indicates whether or not
-   strict syntax checking should be enabled for that instruction.  */
-static int strict = 0;
+   Mainly, strict causes errors to be ignored when a match failure
+   occurs.  However, it also affects the parsing of register fields
+   by pa_parse_number.  */
+static int strict;
 
 /* pa_parse_number returns values in `pa_number'.  Mostly
    pa_parse_number is used to return a register number, with floating
 
 /* pa_parse_number returns values in `pa_number'.  Mostly
    pa_parse_number is used to return a register number, with floating
@@ -777,6 +653,7 @@ static int print_errors = 1;
 
    %r26 - %r23 have %arg0 - %arg3 as synonyms
    %r28 - %r29 have %ret0 - %ret1 as synonyms
 
    %r26 - %r23 have %arg0 - %arg3 as synonyms
    %r28 - %r29 have %ret0 - %ret1 as synonyms
+   %fr4 - %fr7 have %farg0 - %farg3 as synonyms
    %r30 has %sp as a synonym
    %r27 has %dp as a synonym
    %r2  has %rp as a synonym
    %r30 has %sp as a synonym
    %r27 has %dp as a synonym
    %r2  has %rp as a synonym
@@ -820,6 +697,10 @@ static const struct pd_reg pre_defined_registers[] =
   {"%dp",    27},
   {"%eiem",  15},
   {"%eirr",  23},
   {"%dp",    27},
   {"%eiem",  15},
   {"%eirr",  23},
+  {"%farg0",  4 + FP_REG_BASE},
+  {"%farg1",  5 + FP_REG_BASE},
+  {"%farg2",  6 + FP_REG_BASE},
+  {"%farg3",  7 + FP_REG_BASE},
   {"%fr0",    0 + FP_REG_BASE},
   {"%fr0l",   0 + FP_REG_BASE},
   {"%fr0r",   0 + FP_REG_BASE + FP_REG_RSEL},
   {"%fr0",    0 + FP_REG_BASE},
   {"%fr0l",   0 + FP_REG_BASE},
   {"%fr0r",   0 + FP_REG_BASE + FP_REG_RSEL},
@@ -916,6 +797,7 @@ static const struct pd_reg pre_defined_registers[] =
   {"%fr9",    9 + FP_REG_BASE},
   {"%fr9l",   9 + FP_REG_BASE},
   {"%fr9r",   9 + FP_REG_BASE + FP_REG_RSEL},
   {"%fr9",    9 + FP_REG_BASE},
   {"%fr9l",   9 + FP_REG_BASE},
   {"%fr9r",   9 + FP_REG_BASE + FP_REG_RSEL},
+  {"%fret",   4},
   {"%hta",   25},
   {"%iir",   19},
   {"%ior",   21},
   {"%hta",   25},
   {"%iir",   19},
   {"%ior",   21},
@@ -923,6 +805,11 @@ static const struct pd_reg pre_defined_registers[] =
   {"%isr",   20},
   {"%itmr",  16},
   {"%iva",   14},
   {"%isr",   20},
   {"%itmr",  16},
   {"%iva",   14},
+#if TARGET_ARCH_SIZE == 64
+  {"%mrp",    2},
+#else
+  {"%mrp",   31},
+#endif
   {"%pcoq",  18},
   {"%pcsq",  17},
   {"%pidr1",  8},
   {"%pcoq",  18},
   {"%pcsq",  17},
   {"%pidr1",  8},
@@ -976,6 +863,14 @@ static const struct pd_reg pre_defined_registers[] =
   {"%sr5",    5},
   {"%sr6",    6},
   {"%sr7",    7},
   {"%sr5",    5},
   {"%sr6",    6},
   {"%sr7",    7},
+  {"%t1",    22},
+  {"%t2",    21},
+  {"%t3",    20},
+  {"%t4",    19},
+  {"%tf1",   11},
+  {"%tf2",   10},
+  {"%tf3",    9},
+  {"%tf4",    8},
   {"%tr0",   24},
   {"%tr1",   25},
   {"%tr2",   26},
   {"%tr0",   24},
   {"%tr1",   25},
   {"%tr2",   26},
@@ -1052,8 +947,8 @@ static const struct selector_entry selector_table[] =
 #ifdef OBJ_SOM
 /* default space and subspace dictionaries */
 
 #ifdef OBJ_SOM
 /* default space and subspace dictionaries */
 
-#define GDB_SYMBOLS          GDB_SYMBOLS_SUBSPACE_NAME
-#define GDB_STRINGS          GDB_STRINGS_SUBSPACE_NAME
+#define GDB_SYMBOLS    GDB_SYMBOLS_SUBSPACE_NAME
+#define GDB_STRINGS    GDB_STRINGS_SUBSPACE_NAME
 
 /* pre-defined subsegments (subspaces) for the HPPA.  */
 #define SUBSEG_CODE   0
 
 /* pre-defined subsegments (subspaces) for the HPPA.  */
 #define SUBSEG_CODE   0
@@ -1067,12 +962,12 @@ static const struct selector_entry selector_table[] =
 
 static struct default_subspace_dict pa_def_subspaces[] =
 {
 
 static struct default_subspace_dict pa_def_subspaces[] =
 {
-  {"$CODE$", 1, 1, 1, 0, 0, 0, 24, 0x2c, 0, 8, 0, 0, SUBSEG_CODE},
-  {"$DATA$", 1, 1, 0, 0, 0, 0, 24, 0x1f, 1, 8, 1, 1, SUBSEG_DATA},
-  {"$LIT$", 1, 1, 0, 0, 0, 0, 16, 0x2c, 0, 8, 0, 0, SUBSEG_LIT},
-  {"$MILLICODE$", 1, 1, 0, 0, 0, 0, 8, 0x2c, 0, 8, 0, 0, SUBSEG_MILLI},
-  {"$BSS$", 1, 1, 0, 0, 0, 1, 80, 0x1f, 1, 8, 1, 1, SUBSEG_BSS},
-  {NULL, 0, 1, 0, 0, 0, 0, 255, 0x1f, 0, 4, 0, 0, 0}
+  {"$CODE$", 1, 1, 1, 0, 0, 0, 0, 24, 0x2c, 0, 8, 0, 0, SUBSEG_CODE},
+  {"$DATA$", 1, 1, 0, 0, 0, 0, 0, 24, 0x1f, 1, 8, 1, 1, SUBSEG_DATA},
+  {"$LIT$", 1, 1, 0, 0, 0, 0, 0, 16, 0x2c, 0, 8, 0, 0, SUBSEG_LIT},
+  {"$MILLICODE$", 1, 1, 0, 0, 0, 0, 0, 8, 0x2c, 0, 8, 0, 0, SUBSEG_MILLI},
+  {"$BSS$", 1, 1, 0, 0, 0, 0, 1, 80, 0x1f, 1, 8, 1, 1, SUBSEG_BSS},
+  {NULL, 0, 1, 0, 0, 0, 0, 0, 255, 0x1f, 0, 4, 0, 0, 0}
 };
 
 static struct default_space_dict pa_def_spaces[] =
 };
 
 static struct default_space_dict pa_def_spaces[] =
@@ -1099,6 +994,19 @@ static struct default_space_dict pa_def_spaces[] =
 #define IS_R_SELECT(S)   (*(S) == 'R' || *(S) == 'r')
 #define IS_L_SELECT(S)   (*(S) == 'L' || *(S) == 'l')
 
 #define IS_R_SELECT(S)   (*(S) == 'R' || *(S) == 'r')
 #define IS_L_SELECT(S)   (*(S) == 'L' || *(S) == 'l')
 
+/* Store immediate values of shift/deposit/extract functions.  */
+
+#define SAVE_IMMEDIATE(VALUE) \
+  { \
+    if (immediate_check) \
+      { \
+       if (pos == -1) \
+         pos = (VALUE); \
+       else if (len == -1) \
+         len = (VALUE); \
+      } \
+  }
+
 /* Insert FIELD into OPCODE starting at bit START.  Continue pa_ip
    main loop after insertion.  */
 
 /* Insert FIELD into OPCODE starting at bit START.  Continue pa_ip
    main loop after insertion.  */
 
@@ -1108,7 +1016,7 @@ static struct default_space_dict pa_def_spaces[] =
     continue; \
   }
 
     continue; \
   }
 
-/* Simple range checking for FIELD againt HIGH and LOW bounds.
+/* Simple range checking for FIELD against HIGH and LOW bounds.
    IGNORE is used to suppress the error message.  */
 
 #define CHECK_FIELD(FIELD, HIGH, LOW, IGNORE) \
    IGNORE is used to suppress the error message.  */
 
 #define CHECK_FIELD(FIELD, HIGH, LOW, IGNORE) \
@@ -1116,13 +1024,27 @@ static struct default_space_dict pa_def_spaces[] =
     if ((FIELD) > (HIGH) || (FIELD) < (LOW)) \
       { \
        if (! IGNORE) \
     if ((FIELD) > (HIGH) || (FIELD) < (LOW)) \
       { \
        if (! IGNORE) \
-          as_bad (_("Field out of range [%d..%d] (%d)."), (LOW), (HIGH), \
+         as_bad (_("Field out of range [%d..%d] (%d)."), (LOW), (HIGH), \
                  (int) (FIELD));\
                  (int) (FIELD));\
-        break; \
+       break; \
+      } \
+  }
+
+/* Variant of CHECK_FIELD for use in md_apply_fix and other places where
+   the current file and line number are not valid.  */
+
+#define CHECK_FIELD_WHERE(FIELD, HIGH, LOW, FILENAME, LINE) \
+  { \
+    if ((FIELD) > (HIGH) || (FIELD) < (LOW)) \
+      { \
+       as_bad_where ((FILENAME), (LINE), \
+                     _("Field out of range [%d..%d] (%d)."), (LOW), (HIGH), \
+                     (int) (FIELD));\
+       break; \
       } \
   }
 
       } \
   }
 
-/* Simple alignment checking for FIELD againt ALIGN (a power of two).
+/* Simple alignment checking for FIELD against ALIGN (a power of two).
    IGNORE is used to suppress the error message.  */
 
 #define CHECK_ALIGN(FIELD, ALIGN, IGNORE) \
    IGNORE is used to suppress the error message.  */
 
 #define CHECK_ALIGN(FIELD, ALIGN, IGNORE) \
@@ -1130,9 +1052,9 @@ static struct default_space_dict pa_def_spaces[] =
     if ((FIELD) & ((ALIGN) - 1)) \
       { \
        if (! IGNORE) \
     if ((FIELD) & ((ALIGN) - 1)) \
       { \
        if (! IGNORE) \
-          as_bad (_("Field not properly aligned [%d] (%d)."), (ALIGN), \
+         as_bad (_("Field not properly aligned [%d] (%d)."), (ALIGN), \
                  (int) (FIELD));\
                  (int) (FIELD));\
-        break; \
+       break; \
       } \
   }
 
       } \
   }
 
@@ -1140,10 +1062,34 @@ static struct default_space_dict pa_def_spaces[] =
   ((exp).X_op == O_subtract                    \
    && strcmp (S_GET_NAME ((exp).X_op_symbol), "$global$") == 0)
 
   ((exp).X_op == O_subtract                    \
    && strcmp (S_GET_NAME ((exp).X_op_symbol), "$global$") == 0)
 
+#define is_SB_relative(exp)                    \
+  ((exp).X_op == O_subtract                    \
+   && strcmp (S_GET_NAME ((exp).X_op_symbol), "$segrel$") == 0)
+
 #define is_PC_relative(exp)                    \
   ((exp).X_op == O_subtract                    \
    && strcmp (S_GET_NAME ((exp).X_op_symbol), "$PIC_pcrel$0") == 0)
 
 #define is_PC_relative(exp)                    \
   ((exp).X_op == O_subtract                    \
    && strcmp (S_GET_NAME ((exp).X_op_symbol), "$PIC_pcrel$0") == 0)
 
+#define is_tls_gdidx(exp)                      \
+  ((exp).X_op == O_subtract                    \
+   && strcmp (S_GET_NAME ((exp).X_op_symbol), "$tls_gdidx$") == 0)
+
+#define is_tls_ldidx(exp)                      \
+  ((exp).X_op == O_subtract                    \
+   && strcmp (S_GET_NAME ((exp).X_op_symbol), "$tls_ldidx$") == 0)
+
+#define is_tls_dtpoff(exp)                     \
+  ((exp).X_op == O_subtract                    \
+   && strcmp (S_GET_NAME ((exp).X_op_symbol), "$tls_dtpoff$") == 0)
+
+#define is_tls_ieoff(exp)                      \
+  ((exp).X_op == O_subtract                    \
+   && strcmp (S_GET_NAME ((exp).X_op_symbol), "$tls_ieoff$") == 0)
+
+#define is_tls_leoff(exp)                      \
+  ((exp).X_op == O_subtract                    \
+   && strcmp (S_GET_NAME ((exp).X_op_symbol), "$tls_leoff$") == 0)
+
 /* We need some complex handling for stabs (sym1 - sym2).  Luckily, we'll
    always be able to reduce the expression to a constant, so we don't
    need real complex handling yet.  */
 /* We need some complex handling for stabs (sym1 - sym2).  Luckily, we'll
    always be able to reduce the expression to a constant, so we don't
    need real complex handling yet.  */
@@ -1156,7 +1102,7 @@ static struct default_space_dict pa_def_spaces[] =
    proc/procend pairs match.  */
 
 void
    proc/procend pairs match.  */
 
 void
-pa_check_eof ()
+pa_check_eof (void)
 {
   if (within_entry_exit)
     as_fatal (_("Missing .exit\n"));
 {
   if (within_entry_exit)
     as_fatal (_("Missing .exit\n"));
@@ -1169,21 +1115,19 @@ pa_check_eof ()
    or NULL if no label_symbol_struct exists for the current space.  */
 
 static label_symbol_struct *
    or NULL if no label_symbol_struct exists for the current space.  */
 
 static label_symbol_struct *
-pa_get_label ()
+pa_get_label (void)
 {
 {
-  label_symbol_struct *label_chain;
+  label_symbol_struct *label_chain = label_symbols_rootp;
 
 
-  for (label_chain = label_symbols_rootp;
-       label_chain;
-       label_chain = label_chain->lss_next)
+  if (label_chain)
     {
 #ifdef OBJ_SOM
     {
 #ifdef OBJ_SOM
-    if (current_space == label_chain->lss_space && label_chain->lss_label)
-      return label_chain;
+      if (current_space == label_chain->lss_space && label_chain->lss_label)
+       return label_chain;
 #endif
 #ifdef OBJ_ELF
 #endif
 #ifdef OBJ_ELF
-    if (now_seg == label_chain->lss_segment && label_chain->lss_label)
-      return label_chain;
+      if (now_seg == label_chain->lss_segment && label_chain->lss_label)
+       return label_chain;
 #endif
     }
 
 #endif
     }
 
@@ -1194,67 +1138,38 @@ pa_get_label ()
    this function will replace it with the new label.  */
 
 void
    this function will replace it with the new label.  */
 
 void
-pa_define_label (symbol)
-     symbolS *symbol;
+pa_define_label (symbolS *symbol)
 {
 {
-  label_symbol_struct *label_chain = pa_get_label ();
+  label_symbol_struct *label_chain = label_symbols_rootp;
 
 
-  if (label_chain)
-    label_chain->lss_label = symbol;
-  else
-    {
-      /* Create a new label entry and add it to the head of the chain.  */
-      label_chain
-       = (label_symbol_struct *) xmalloc (sizeof (label_symbol_struct));
-      label_chain->lss_label = symbol;
+  if (!label_chain)
+    label_chain = &last_label_symbol;
+
+  label_chain->lss_label = symbol;
 #ifdef OBJ_SOM
 #ifdef OBJ_SOM
-      label_chain->lss_space = current_space;
+  label_chain->lss_space = current_space;
 #endif
 #ifdef OBJ_ELF
 #endif
 #ifdef OBJ_ELF
-      label_chain->lss_segment = now_seg;
+  label_chain->lss_segment = now_seg;
 #endif
 #endif
-      label_chain->lss_next = NULL;
 
 
-      if (label_symbols_rootp)
-       label_chain->lss_next = label_symbols_rootp;
+  /* Not used.  */
+  label_chain->lss_next = NULL;
 
 
-      label_symbols_rootp = label_chain;
-    }
+  label_symbols_rootp = label_chain;
+
+#ifdef OBJ_ELF
+  dwarf2_emit_label (symbol);
+#endif
 }
 
 /* Removes a label definition for the current space.
    If there is no label_symbol_struct entry, then no action is taken.  */
 
 static void
 }
 
 /* Removes a label definition for the current space.
    If there is no label_symbol_struct entry, then no action is taken.  */
 
 static void
-pa_undefine_label ()
+pa_undefine_label (void)
 {
 {
-  label_symbol_struct *label_chain;
-  label_symbol_struct *prev_label_chain = NULL;
-
-  for (label_chain = label_symbols_rootp;
-       label_chain;
-       label_chain = label_chain->lss_next)
-    {
-      if (1
-#ifdef OBJ_SOM
-         && current_space == label_chain->lss_space && label_chain->lss_label
-#endif
-#ifdef OBJ_ELF
-         && now_seg == label_chain->lss_segment && label_chain->lss_label
-#endif
-         )
-       {
-         /* Remove the label from the chain and free its memory.  */
-         if (prev_label_chain)
-           prev_label_chain->lss_next = label_chain->lss_next;
-         else
-           label_symbols_rootp = label_chain->lss_next;
-
-         free (label_chain);
-         break;
-       }
-      prev_label_chain = label_chain;
-    }
+  label_symbols_rootp = NULL;
 }
 
 /* An HPPA-specific version of fix_new.  This is required because the HPPA
 }
 
 /* An HPPA-specific version of fix_new.  This is required because the HPPA
@@ -1265,25 +1180,21 @@ pa_undefine_label ()
    tc_fix_data field.  */
 
 static void
    tc_fix_data field.  */
 
 static void
-fix_new_hppa (frag, where, size, add_symbol, offset, exp, pcrel,
-             r_type, r_field, r_format, arg_reloc, unwind_bits)
-     fragS *frag;
-     int where;
-     int size;
-     symbolS *add_symbol;
-     offsetT offset;
-     expressionS *exp;
-     int pcrel;
-     bfd_reloc_code_real_type r_type;
-     enum hppa_reloc_field_selector_type_alt r_field;
-     int r_format;
-     unsigned int arg_reloc;
-     int* unwind_bits ATTRIBUTE_UNUSED;
+fix_new_hppa (fragS *frag,
+             int where,
+             int size,
+             symbolS *add_symbol,
+             offsetT offset,
+             expressionS *exp,
+             int pcrel,
+             bfd_reloc_code_real_type r_type,
+             enum hppa_reloc_field_selector_type_alt r_field,
+             int r_format,
+             unsigned int arg_reloc,
+             int unwind_bits ATTRIBUTE_UNUSED)
 {
   fixS *new_fix;
 {
   fixS *new_fix;
-
-  struct hppa_fix_struct *hppa_fix = (struct hppa_fix_struct *)
-  obstack_alloc (&notes, sizeof (struct hppa_fix_struct));
+  struct hppa_fix_struct *hppa_fix = XOBNEW (&notes, struct hppa_fix_struct);
 
   if (exp != NULL)
     new_fix = fix_new_exp (frag, where, size, exp, pcrel, r_type);
 
   if (exp != NULL)
     new_fix = fix_new_exp (frag, where, size, exp, pcrel, r_type);
@@ -1297,7 +1208,7 @@ fix_new_hppa (frag, where, size, add_symbol, offset, exp, pcrel,
   hppa_fix->segment = now_seg;
 #ifdef OBJ_SOM
   if (r_type == R_ENTRY || r_type == R_EXIT)
   hppa_fix->segment = now_seg;
 #ifdef OBJ_SOM
   if (r_type == R_ENTRY || r_type == R_EXIT)
-    new_fix->fx_offset = *unwind_bits;
+    new_fix->fx_offset = unwind_bits;
 #endif
 
   /* foo-$global$ is used to access non-automatic storage.  $global$
 #endif
 
   /* foo-$global$ is used to access non-automatic storage.  $global$
@@ -1305,36 +1216,44 @@ fix_new_hppa (frag, where, size, add_symbol, offset, exp, pcrel,
      it now so as not to confuse write.c.  Ditto for $PIC_pcrel$0.  */
   if (new_fix->fx_subsy
       && (strcmp (S_GET_NAME (new_fix->fx_subsy), "$global$") == 0
      it now so as not to confuse write.c.  Ditto for $PIC_pcrel$0.  */
   if (new_fix->fx_subsy
       && (strcmp (S_GET_NAME (new_fix->fx_subsy), "$global$") == 0
-         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$PIC_pcrel$0") == 0))
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$segrel$") == 0
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$PIC_pcrel$0") == 0
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$tls_gdidx$") == 0
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$tls_ldidx$") == 0
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$tls_dtpoff$") == 0
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$tls_ieoff$") == 0
+         || strcmp (S_GET_NAME (new_fix->fx_subsy), "$tls_leoff$") == 0))
     new_fix->fx_subsy = NULL;
 }
 
     new_fix->fx_subsy = NULL;
 }
 
-/* Parse a .byte, .word, .long expression for the HPPA.  Called by
-   cons via the TC_PARSE_CONS_EXPRESSION macro.  */
-
-void
-parse_cons_expression_hppa (exp)
-     expressionS *exp;
-{
-  hppa_field_selector = pa_chk_field_selector (&input_line_pointer);
-  expression (exp);
-}
-
 /* This fix_new is called by cons via TC_CONS_FIX_NEW.
    hppa_field_selector is set by the parse_cons_expression_hppa.  */
 
 void
 /* This fix_new is called by cons via TC_CONS_FIX_NEW.
    hppa_field_selector is set by the parse_cons_expression_hppa.  */
 
 void
-cons_fix_new_hppa (frag, where, size, exp)
-     fragS *frag;
-     int where;
-     int size;
-     expressionS *exp;
+cons_fix_new_hppa (fragS *frag, int where, int size, expressionS *exp,
+                  int hppa_field_selector)
 {
   unsigned int rel_type;
 
   /* Get a base relocation type.  */
   if (is_DP_relative (*exp))
     rel_type = R_HPPA_GOTOFF;
 {
   unsigned int rel_type;
 
   /* Get a base relocation type.  */
   if (is_DP_relative (*exp))
     rel_type = R_HPPA_GOTOFF;
+  else if (is_PC_relative (*exp))
+    rel_type = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+  else if (is_SB_relative (*exp))
+    rel_type = R_PARISC_SEGREL32;
+  else if (is_tls_gdidx (*exp))
+    rel_type = R_PARISC_TLS_GD21L;
+  else if (is_tls_ldidx (*exp))
+    rel_type = R_PARISC_TLS_LDM21L;
+  else if (is_tls_dtpoff (*exp))
+    rel_type = R_PARISC_TLS_LDO21L;
+  else if (is_tls_ieoff (*exp))
+    rel_type = R_PARISC_TLS_IE21L;
+  else if (is_tls_leoff (*exp))
+    rel_type = R_PARISC_TLS_LE21L;
+#endif
   else if (is_complex (*exp))
     rel_type = R_HPPA_COMPLEX;
   else
   else if (is_complex (*exp))
     rel_type = R_HPPA_COMPLEX;
   else
@@ -1348,4517 +1267,4539 @@ cons_fix_new_hppa (frag, where, size, exp)
 
   fix_new_hppa (frag, where, size,
                (symbolS *) NULL, (offsetT) 0, exp, 0, rel_type,
 
   fix_new_hppa (frag, where, size,
                (symbolS *) NULL, (offsetT) 0, exp, 0, rel_type,
-               hppa_field_selector, size * 8, 0, NULL);
-
-  /* Reset field selector to its default state.  */
-  hppa_field_selector = 0;
+               hppa_field_selector, size * 8, 0, 0);
 }
 
 }
 
-/* This function is called once, at assembler startup time.  It should
-   set up all the tables, etc. that the MD part of the assembler will need.  */
+/* Mark (via expr_end) the end of an expression (I think).  FIXME.  */
 
 
-void
-md_begin ()
+static void
+get_expression (char *str)
 {
 {
-  const char *retval = NULL;
-  int lose = 0;
-  unsigned int i = 0;
-
-  last_call_info = NULL;
-  call_info_root = NULL;
-
-  /* Set the default machine type.  */
-  if (!bfd_set_arch_mach (stdoutput, bfd_arch_hppa, 10))
-    as_warn (_("could not set architecture and machine"));
+  char *save_in;
+  asection *seg;
 
 
-  /* Folding of text and data segments fails miserably on the PA.
-     Warn user and disable "-R" option.  */
-  if (flag_readonly_data_in_text)
+  save_in = input_line_pointer;
+  input_line_pointer = str;
+  seg = expression (&the_insn.exp);
+  if (!(seg == absolute_section
+       || seg == undefined_section
+       || SEG_NORMAL (seg)))
     {
     {
-      as_warn (_("-R option not supported on this target."));
-      flag_readonly_data_in_text = 0;
+      as_warn (_("Bad segment in expression."));
+      expr_end = input_line_pointer;
+      input_line_pointer = save_in;
+      return;
     }
     }
+  expr_end = input_line_pointer;
+  input_line_pointer = save_in;
+}
 
 
-#ifdef OBJ_SOM
-  pa_spaces_begin ();
-#endif
+/* Parse a PA nullification completer (,n).  Return nonzero if the
+   completer was found; return zero if no completer was found.  */
 
 
-  op_hash = hash_new ();
+static int
+pa_parse_nullif (char **s)
+{
+  int nullif;
 
 
-  while (i < NUMOPCODES)
+  nullif = 0;
+  if (**s == ',')
     {
     {
-      const char *name = pa_opcodes[i].name;
-      retval = hash_insert (op_hash, name, (struct pa_opcode *) &pa_opcodes[i]);
-      if (retval != NULL && *retval != '\0')
-       {
-         as_fatal (_("Internal error: can't hash `%s': %s\n"), name, retval);
-         lose = 1;
-       }
-      do
+      *s = *s + 1;
+      if (strncasecmp (*s, "n", 1) == 0)
+       nullif = 1;
+      else
        {
        {
-         if ((pa_opcodes[i].match & pa_opcodes[i].mask)
-             != pa_opcodes[i].match)
-           {
-             fprintf (stderr, _("internal error: losing opcode: `%s' \"%s\"\n"),
-                      pa_opcodes[i].name, pa_opcodes[i].args);
-             lose = 1;
-           }
-         ++i;
+         as_bad (_("Invalid Nullification: (%c)"), **s);
+         nullif = 0;
        }
        }
-      while (i < NUMOPCODES && !strcmp (pa_opcodes[i].name, name));
+      *s = *s + 1;
     }
 
     }
 
-  if (lose)
-    as_fatal (_("Broken assembler.  No assembly attempted."));
-
-#ifdef OBJ_SOM
-  /* SOM will change text_section.  To make sure we never put
-     anything into the old one switch to the new one now.  */
-  subseg_set (text_section, 0);
-#endif
+  return nullif;
+}
 
 
-#ifdef OBJ_SOM
-  dummy_symbol = symbol_find_or_make ("L$dummy");
-  S_SET_SEGMENT (dummy_symbol, text_section);
-  /* Force the symbol to be converted to a real symbol.  */
-  (void) symbol_get_bfdsym (dummy_symbol);
-#endif
+const char *
+md_atof (int type, char *litP, int *sizeP)
+{
+  return ieee_md_atof (type, litP, sizeP, TRUE);
 }
 
 }
 
-/* Assemble a single instruction storing it into a frag.  */
+/* Write out big-endian.  */
+
 void
 void
-md_assemble (str)
-     char *str;
+md_number_to_chars (char *buf, valueT val, int n)
 {
 {
-  char *to;
-
-  /* The had better be something to assemble.  */
-  assert (str);
+  number_to_chars_bigendian (buf, val, n);
+}
 
 
-  /* If we are within a procedure definition, make sure we've
-     defined a label for the procedure; handle case where the
-     label was defined after the .PROC directive.
-
-     Note there's not need to diddle with the segment or fragment
-     for the label symbol in this case.  We have already switched
-     into the new $CODE$ subspace at this point.  */
-  if (within_procedure && last_call_info->start_symbol == NULL)
-    {
-      label_symbol_struct *label_symbol = pa_get_label ();
-
-      if (label_symbol)
-       {
-         if (label_symbol->lss_label)
-           {
-             last_call_info->start_symbol = label_symbol->lss_label;
-             symbol_get_bfdsym (label_symbol->lss_label)->flags
-               |= BSF_FUNCTION;
-#ifdef OBJ_SOM
-             /* Also handle allocation of a fixup to hold the unwind
-                information when the label appears after the proc/procend.  */
-             if (within_entry_exit)
-               {
-                 char *where = frag_more (0);
+/* Translate internal representation of relocation info to BFD target
+   format.  */
 
 
-                 fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
-                               NULL, (offsetT) 0, NULL,
-                               0, R_HPPA_ENTRY, e_fsel, 0, 0,
-                               (int *)&last_call_info->ci_unwind.descriptor);
-               }
-#endif
-           }
-         else
-           as_bad (_("Missing function name for .PROC (corrupted label chain)"));
-       }
-      else
-       as_bad (_("Missing function name for .PROC"));
-    }
+arelent **
+tc_gen_reloc (asection *section, fixS *fixp)
+{
+  arelent *reloc;
+  struct hppa_fix_struct *hppa_fixp;
+  static arelent *no_relocs = NULL;
+  arelent **relocs;
+  reloc_type **codes;
+  reloc_type code;
+  int n_relocs;
+  int i;
 
 
-  /* Assemble the instruction.  Results are saved into "the_insn".  */
-  pa_ip (str);
+  hppa_fixp = (struct hppa_fix_struct *) fixp->tc_fix_data;
+  if (fixp->fx_addsy == 0)
+    return &no_relocs;
 
 
-  /* Get somewhere to put the assembled instrution.  */
-  to = frag_more (4);
+  gas_assert (hppa_fixp != 0);
+  gas_assert (section != 0);
 
 
-  /* Output the opcode.  */
-  md_number_to_chars (to, the_insn.opcode, 4);
+  reloc = XNEW (arelent);
 
 
-  /* If necessary output more stuff.  */
-  if (the_insn.reloc != R_HPPA_NONE)
-    fix_new_hppa (frag_now, (to - frag_now->fr_literal), 4, NULL,
-                 (offsetT) 0, &the_insn.exp, the_insn.pcrel,
-                 the_insn.reloc, the_insn.field_selector,
-                 the_insn.format, the_insn.arg_reloc, NULL);
+  reloc->sym_ptr_ptr = XNEW (asymbol *);
+  *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
 
 
-#ifdef OBJ_ELF
-  dwarf2_emit_insn (4);
-#endif
-}
+  /* Allow fixup_segment to recognize hand-written pc-relative relocations.
+     When we went through cons_fix_new_hppa, we classified them as complex.  */
+  /* ??? It might be better to hide this +8 stuff in tc_cfi_emit_pcrel_expr,
+     undefine DIFF_EXPR_OK, and let these sorts of complex expressions fail
+     when R_HPPA_COMPLEX == R_PARISC_UNIMPLEMENTED.  */
+  if (fixp->fx_r_type == (bfd_reloc_code_real_type) R_HPPA_COMPLEX
+      && fixp->fx_pcrel)
+    {
+      fixp->fx_r_type = R_HPPA_PCREL_CALL;
+      fixp->fx_offset += 8;
+    }
 
 
-/* Do the real work for assembling a single instruction.  Store results
-   into the global "the_insn" variable.  */
+  codes = hppa_gen_reloc_type (stdoutput,
+                              fixp->fx_r_type,
+                              hppa_fixp->fx_r_format,
+                              hppa_fixp->fx_r_field,
+                              fixp->fx_subsy != NULL,
+                              symbol_get_bfdsym (fixp->fx_addsy));
 
 
-static void
-pa_ip (str)
-     char *str;
-{
-  char *error_message = "";
-  char *s, c, *argstart, *name, *save_s;
-  const char *args;
-  int match = FALSE;
-  int comma = 0;
-  int cmpltr, nullif, flag, cond, num;
-  unsigned long opcode;
-  struct pa_opcode *insn;
+  if (codes == NULL)
+    {
+      as_bad_where (fixp->fx_file, fixp->fx_line, _("Cannot handle fixup"));
+      abort ();
+    }
 
 
-#ifdef OBJ_SOM
-  /* We must have a valid space and subspace.  */
-  pa_check_current_space_and_subspace ();
-#endif
+  for (n_relocs = 0; codes[n_relocs]; n_relocs++)
+    ;
 
 
-  /* Convert everything up to the first whitespace character into lower
-     case.  */
-  for (s = str; *s != ' ' && *s != '\t' && *s != '\n' && *s != '\0'; s++)
-    if (isupper (*s))
-      *s = tolower (*s);
+  relocs = XNEWVEC (arelent *, n_relocs + 1);
+  reloc = XNEWVEC (arelent, n_relocs);
+  for (i = 0; i < n_relocs; i++)
+    relocs[i] = &reloc[i];
 
 
-  /* Skip to something interesting.  */
-  for (s = str; isupper (*s) || islower (*s) || (*s >= '0' && *s <= '3'); ++s)
-    ;
+  relocs[n_relocs] = NULL;
 
 
-  switch (*s)
+#ifdef OBJ_ELF
+  switch (fixp->fx_r_type)
     {
     {
+    default:
+      gas_assert (n_relocs == 1);
 
 
-    case '\0':
-      break;
+      code = *codes[0];
 
 
-    case ',':
-      comma = 1;
+      /* Now, do any processing that is dependent on the relocation type.  */
+      switch (code)
+       {
+       case R_PARISC_DLTREL21L:
+       case R_PARISC_DLTREL14R:
+       case R_PARISC_DLTREL14F:
+       case R_PARISC_PLABEL32:
+       case R_PARISC_PLABEL21L:
+       case R_PARISC_PLABEL14R:
+         /* For plabel relocations, the addend of the
+            relocation should be either 0 (no static link) or 2
+            (static link required).  This adjustment is done in
+            bfd/elf32-hppa.c:elf32_hppa_relocate_section.
 
 
-      /*FALLTHROUGH */
+            We also slam a zero addend into the DLT relative relocs;
+            it doesn't make a lot of sense to use any addend since
+            it gets you a different (eg unknown) DLT entry.  */
+         reloc->addend = 0;
+         break;
 
 
-    case ' ':
-      *s++ = '\0';
-      break;
+#ifdef ELF_ARG_RELOC
+       case R_PARISC_PCREL17R:
+       case R_PARISC_PCREL17F:
+       case R_PARISC_PCREL17C:
+       case R_PARISC_DIR17R:
+       case R_PARISC_DIR17F:
+       case R_PARISC_PCREL21L:
+       case R_PARISC_DIR21L:
+         reloc->addend = HPPA_R_ADDEND (hppa_fixp->fx_arg_reloc,
+                                        fixp->fx_offset);
+         break;
+#endif
 
 
-    default:
-      as_fatal (_("Unknown opcode: `%s'"), str);
-    }
+       case R_PARISC_DIR32:
+         /* Facilitate hand-crafted unwind info.  */
+         if (strcmp (section->name, UNWIND_SECTION_NAME) == 0)
+           code = R_PARISC_SEGREL32;
+         /* Fall thru */
 
 
-  save_s = str;
+       default:
+         reloc->addend = fixp->fx_offset;
+         break;
+       }
 
 
-  /* Look up the opcode in the has table.  */
-  if ((insn = (struct pa_opcode *) hash_find (op_hash, str)) == NULL)
-    {
-      as_bad ("Unknown opcode: `%s'", str);
-      return;
-    }
+      reloc->sym_ptr_ptr = XNEW (asymbol *);
+      *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
+      reloc->howto = bfd_reloc_type_lookup (stdoutput,
+                                           (bfd_reloc_code_real_type) code);
+      reloc->address = fixp->fx_frag->fr_address + fixp->fx_where;
 
 
-  if (comma)
-    {
-      *--s = ',';
+      gas_assert (reloc->howto && (unsigned int) code == reloc->howto->type);
+      break;
     }
     }
+#else /* OBJ_SOM */
 
 
-  /* Mark the location where arguments for the instruction start, then
-     start processing them.  */
-  argstart = s;
-  for (;;)
+  /* Walk over reach relocation returned by the BFD backend.  */
+  for (i = 0; i < n_relocs; i++)
     {
     {
-      /* Do some initialization.  */
-      opcode = insn->match;
-      strict = (insn->flags & FLAG_STRICT);
-      memset (&the_insn, 0, sizeof (the_insn));
-
-      the_insn.reloc = R_HPPA_NONE;
+      code = *codes[i];
 
 
-      /* If this instruction is specific to a particular architecture,
-        then set a new architecture.  */
-      /* But do not automatically promote to pa2.0.  The automatic promotion
-        crud is for compatability with HP's old assemblers only.  */
-      if (insn->arch < 20
-         && bfd_get_mach (stdoutput) < insn->arch)
-       {
-         if (!bfd_set_arch_mach (stdoutput, bfd_arch_hppa, insn->arch))
-           as_warn (_("could not update architecture and machine"));
-       }
-      else if (bfd_get_mach (stdoutput) < insn->arch)
-       {
-         match = FALSE;
-         goto failed;
-       }
+      relocs[i]->sym_ptr_ptr = XNEW (asymbol *);
+      *relocs[i]->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
+      relocs[i]->howto =
+       bfd_reloc_type_lookup (stdoutput,
+                              (bfd_reloc_code_real_type) code);
+      relocs[i]->address = fixp->fx_frag->fr_address + fixp->fx_where;
 
 
-      /* Build the opcode, checking as we go to make
-         sure that the operands match.  */
-      for (args = insn->args;; ++args)
+      switch (code)
        {
        {
-         /* Absorb white space in instruction.  */
-         while (*s == ' ' || *s == '\t')
-           s++;
+       case R_COMP2:
+         /* The only time we ever use a R_COMP2 fixup is for the difference
+            of two symbols.  With that in mind we fill in all four
+            relocs now and break out of the loop.  */
+         gas_assert (i == 1);
+         relocs[0]->sym_ptr_ptr
+           = (asymbol **) bfd_abs_section_ptr->symbol_ptr_ptr;
+         relocs[0]->howto
+           = bfd_reloc_type_lookup (stdoutput,
+                                    (bfd_reloc_code_real_type) *codes[0]);
+         relocs[0]->address = fixp->fx_frag->fr_address + fixp->fx_where;
+         relocs[0]->addend = 0;
+         relocs[1]->sym_ptr_ptr = XNEW (asymbol *);
+         *relocs[1]->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
+         relocs[1]->howto
+           = bfd_reloc_type_lookup (stdoutput,
+                                    (bfd_reloc_code_real_type) *codes[1]);
+         relocs[1]->address = fixp->fx_frag->fr_address + fixp->fx_where;
+         relocs[1]->addend = 0;
+         relocs[2]->sym_ptr_ptr = XNEW (asymbol *);
+         *relocs[2]->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_subsy);
+         relocs[2]->howto
+           = bfd_reloc_type_lookup (stdoutput,
+                                    (bfd_reloc_code_real_type) *codes[2]);
+         relocs[2]->address = fixp->fx_frag->fr_address + fixp->fx_where;
+         relocs[2]->addend = 0;
+         relocs[3]->sym_ptr_ptr
+           = (asymbol **) bfd_abs_section_ptr->symbol_ptr_ptr;
+         relocs[3]->howto
+           = bfd_reloc_type_lookup (stdoutput,
+                                    (bfd_reloc_code_real_type) *codes[3]);
+         relocs[3]->address = fixp->fx_frag->fr_address + fixp->fx_where;
+         relocs[3]->addend = 0;
+         relocs[4]->sym_ptr_ptr
+           = (asymbol **) bfd_abs_section_ptr->symbol_ptr_ptr;
+         relocs[4]->howto
+           = bfd_reloc_type_lookup (stdoutput,
+                                    (bfd_reloc_code_real_type) *codes[4]);
+         relocs[4]->address = fixp->fx_frag->fr_address + fixp->fx_where;
+         relocs[4]->addend = 0;
+         goto done;
+       case R_PCREL_CALL:
+       case R_ABS_CALL:
+         relocs[i]->addend = HPPA_R_ADDEND (hppa_fixp->fx_arg_reloc, 0);
+         break;
 
 
-         switch (*args)
-           {
+       case R_DLT_REL:
+       case R_DATA_PLABEL:
+       case R_CODE_PLABEL:
+         /* For plabel relocations, the addend of the
+            relocation should be either 0 (no static link) or 2
+            (static link required).
 
 
-           /* End of arguments.  */
-           case '\0':
-             if (*s == '\0')
-               match = TRUE;
-             break;
+            FIXME: We always assume no static link!
 
 
-           case '+':
-             if (*s == '+')
-               {
-                 ++s;
-                 continue;
-               }
-             if (*s == '-')
-               continue;
-             break;
+            We also slam a zero addend into the DLT relative relocs;
+            it doesn't make a lot of sense to use any addend since
+            it gets you a different (eg unknown) DLT entry.  */
+         relocs[i]->addend = 0;
+         break;
 
 
-           /* These must match exactly.  */
-           case '(':
-           case ')':
-           case ',':
-           case ' ':
-             if (*s++ == *args)
-               continue;
-             break;
+       case R_N_MODE:
+       case R_S_MODE:
+       case R_D_MODE:
+       case R_R_MODE:
+       case R_FSEL:
+       case R_LSEL:
+       case R_RSEL:
+       case R_BEGIN_BRTAB:
+       case R_END_BRTAB:
+       case R_BEGIN_TRY:
+       case R_N0SEL:
+       case R_N1SEL:
+         /* There is no symbol or addend associated with these fixups.  */
+         relocs[i]->sym_ptr_ptr = XNEW (asymbol *);
+         *relocs[i]->sym_ptr_ptr = symbol_get_bfdsym (dummy_symbol);
+         relocs[i]->addend = 0;
+         break;
 
 
-           /* Handle a 5 bit register or control register field at 10.  */
-           case 'b':
-           case '^':
-             if (!pa_parse_number (&s, 0))
-               break;
-             num = pa_number;
-             CHECK_FIELD (num, 31, 0, 0);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
+       case R_END_TRY:
+       case R_ENTRY:
+       case R_EXIT:
+         /* There is no symbol associated with these fixups.  */
+         relocs[i]->sym_ptr_ptr = XNEW (asymbol *);
+         *relocs[i]->sym_ptr_ptr = symbol_get_bfdsym (dummy_symbol);
+         relocs[i]->addend = fixp->fx_offset;
+         break;
 
 
-           /* Handle %sar or %cr11.  No bits get set, we just verify that it
-              is there.  */
-           case '!':
-             /* Skip whitespace before register.  */
-             while (*s == ' ' || *s == '\t')
-               s = s + 1;
+       default:
+         relocs[i]->addend = fixp->fx_offset;
+       }
+    }
 
 
-             if (!strncasecmp(s, "%sar", 4))
-               {
-                 s += 4;
-                 continue;
-               }
-             else if (!strncasecmp(s, "%cr11", 5))
-               {
-                 s += 5;
-                 continue;
-               }
-             break;
+ done:
+#endif
 
 
-           /* Handle a 5 bit register field at 15.  */
-           case 'x':
-             if (!pa_parse_number (&s, 0))
-               break;
-             num = pa_number;
-             CHECK_FIELD (num, 31, 0, 0);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
+  return relocs;
+}
 
 
-           /* Handle a 5 bit register field at 31.  */
-           case 't':
-             if (!pa_parse_number (&s, 0))
-               break;
-             num = pa_number;
-             CHECK_FIELD (num, 31, 0, 0);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+/* Process any machine dependent frag types.  */
 
 
-           /* Handle a 5 bit register field at 10 and 15.  */
-           case 'a':
-             if (!pa_parse_number (&s, 0))
-               break;
-             num = pa_number;
-             CHECK_FIELD (num, 31, 0, 0);
-             opcode |= num << 16;
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
+void
+md_convert_frag (bfd *abfd ATTRIBUTE_UNUSED,
+                asection *sec ATTRIBUTE_UNUSED,
+                fragS *fragP)
+{
+  unsigned int address;
 
 
-           /* Handle a 5 bit field length at 31.  */
-           case 'T':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 32, 1, 0);
-             INSERT_FIELD_AND_CONTINUE (opcode, 32 - num, 0);
+  if (fragP->fr_type == rs_machine_dependent)
+    {
+      switch ((int) fragP->fr_subtype)
+       {
+       case 0:
+         fragP->fr_type = rs_fill;
+         know (fragP->fr_var == 1);
+         know (fragP->fr_next);
+         address = fragP->fr_address + fragP->fr_fix;
+         if (address % fragP->fr_offset)
+           {
+             fragP->fr_offset =
+               fragP->fr_next->fr_address
+               - fragP->fr_address
+               - fragP->fr_fix;
+           }
+         else
+           fragP->fr_offset = 0;
+         break;
+       }
+    }
+}
 
 
-           /* Handle a 5 bit immediate at 15.  */
-           case '5':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             /* When in strict mode, we want to just reject this
-                match instead of giving an out of range error.  */
-             CHECK_FIELD (num, 15, -16, strict);
-             num = low_sign_unext (num, 5);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
+/* Round up a section size to the appropriate boundary.  */
 
 
-           /* Handle a 5 bit immediate at 31.  */
-           case 'V':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             /* When in strict mode, we want to just reject this
-                match instead of giving an out of range error.  */
-             CHECK_FIELD (num, 15, -16, strict);
-             num = low_sign_unext (num, 5);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+valueT
+md_section_align (asection *segment, valueT size)
+{
+  int align = bfd_get_section_alignment (stdoutput, segment);
+  int align2 = (1 << align) - 1;
 
 
-           /* Handle an unsigned 5 bit immediate at 31.  */
-           case 'r':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 31, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+  return (size + align2) & ~align2;
+}
 
 
-           /* Handle an unsigned 5 bit immediate at 15.  */
-           case 'R':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 31, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
+/* Return the approximate size of a frag before relaxation has occurred.  */
 
 
-           /* Handle an unsigned 10 bit immediate at 15.  */
-           case 'U':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 1023, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
+int
+md_estimate_size_before_relax (fragS *fragP, asection *segment ATTRIBUTE_UNUSED)
+{
+  int size;
 
 
-           /* Handle a 2 bit space identifier at 17.  */
-           case 's':
-             if (!pa_parse_number (&s, 0))
-               break;
-             num = pa_number;
-             CHECK_FIELD (num, 3, 0, 1);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 14);
+  size = 0;
 
 
-           /* Handle a 3 bit space identifier at 18.  */
-           case 'S':
-             if (!pa_parse_number (&s, 0))
-               break;
-             num = pa_number;
-             CHECK_FIELD (num, 7, 0, 1);
-             opcode |= re_assemble_3 (num);
-             continue;
+  while ((fragP->fr_fix + size) % fragP->fr_offset)
+    size++;
 
 
-           /* Handle all completers.  */
-           case 'c':
-             switch (*++args)
-               {
+  return size;
+}
+\f
+#ifdef OBJ_ELF
+# ifdef WARN_COMMENTS
+const char *md_shortopts = "Vc";
+# else
+const char *md_shortopts = "V";
+# endif
+#else
+# ifdef WARN_COMMENTS
+const char *md_shortopts = "c";
+# else
+const char *md_shortopts = "";
+# endif
+#endif
 
 
-               /* Handle a completer for an indexing load or store.  */
-               case 'x':
-                 {
-                   int uu = 0;
-                   int m = 0;
-                   int i = 0;
-                   while (*s == ',' && i < 2)
-                     {
-                       s++;
-                       if (strncasecmp (s, "sm", 2) == 0)
-                         {
-                           uu = 1;
-                           m = 1;
-                           s++;
-                           i++;
-                         }
-                       else if (strncasecmp (s, "m", 1) == 0)
-                         m = 1;
-                       else if ((strncasecmp (s, "s ", 2) == 0)
-                                || (strncasecmp (s, "s,", 2) == 0))
-                         uu = 1;
-                       /* When in strict mode this is a match failure.  */
-                       else if (strict)
-                         {
-                           s--;
-                           break;
-                         }
-                       else
-                         as_bad (_("Invalid Indexed Load Completer."));
-                       s++;
-                       i++;
-                     }
-                   if (i > 2)
-                     as_bad (_("Invalid Indexed Load Completer Syntax."));
-                   opcode |= m << 5;
-                   INSERT_FIELD_AND_CONTINUE (opcode, uu, 13);
-                 }
+struct option md_longopts[] =
+{
+#ifdef WARN_COMMENTS
+  {"warn-comment", no_argument, NULL, 'c'},
+#endif
+  {NULL, no_argument, NULL, 0}
+};
+size_t md_longopts_size = sizeof (md_longopts);
 
 
-               /* Handle a short load/store completer.  */
-               case 'm':
-               case 'q':
-               case 'J':
-               case 'e':
-                 {
-                   int a = 0;
-                   int m = 0;
-                   if (*s == ',')
-                     {
-                       int found = 0;
-                       s++;
-                       if (strncasecmp (s, "ma", 2) == 0)
-                         {
-                           a = 0;
-                           m = 1;
-                           found = 1;
-                         }
-                       else if (strncasecmp (s, "mb", 2) == 0)
-                         {
-                           a = 1;
-                           m = 1;
-                           found = 1;
-                         }
-
-                       /* When in strict mode, pass through for cache op.  */
-                       if (!found && strict)
-                         s--;
-                       else
-                         {
-                           if (!found)
-                             as_bad (_("Invalid Short Load/Store Completer."));
-                           s += 2;
-                         }
-                     }
-                   /* If we did not get a ma/mb completer, then we do not
-                      consider this a positive match for 'ce'.  */
-                   else if (*args == 'e')
-                     break;
-
-                  /* 'J', 'm' and 'q' are the same, except for where they
-                      encode the before/after field.  */
-                  if (*args == 'm')
-                     {
-                       opcode |= m << 5;
-                       INSERT_FIELD_AND_CONTINUE (opcode, a, 13);
-                     }
-                   else if (*args == 'q')
-                     {
-                       opcode |= m << 3;
-                       INSERT_FIELD_AND_CONTINUE (opcode, a, 2);
-                     }
-                   else if (*args == 'J')
-                     {
-                       /* M bit is explicit in the major opcode.  */
-                       INSERT_FIELD_AND_CONTINUE (opcode, a, 2);
-                     }
-                   else if (*args == 'e')
-                     {
-                       /* Stash the ma/mb flag temporarily in the
-                          instruction.  We will use (and remove it)
-                          later when handling 'J', 'K', '<' & '>'.  */
-                       opcode |= a;
-                       continue;
-                     }
-                 }
-
-               /* Handle a stbys completer.  */
-               case 's':
-                 {
-                   int a = 0;
-                   int m = 0;
-                   int i = 0;
-                   while (*s == ',' && i < 2)
-                     {
-                       s++;
-                       if (strncasecmp (s, "m", 1) == 0)
-                         m = 1;
-                       else if ((strncasecmp (s, "b ", 2) == 0)
-                                || (strncasecmp (s, "b,", 2) == 0))
-                         a = 0;
-                       else if (strncasecmp (s, "e", 1) == 0)
-                         a = 1;
-                       /* When in strict mode this is a match failure.  */
-                       else if (strict)
-                         {
-                           s--;
-                           break;
-                         }
-                       else
-                         as_bad (_("Invalid Store Bytes Short Completer"));
-                       s++;
-                       i++;
-                     }
-                   if (i > 2)
-                     as_bad (_("Invalid Store Bytes Short Completer"));
-                   opcode |= m << 5;
-                   INSERT_FIELD_AND_CONTINUE (opcode, a, 13);
-                 }
+int
+md_parse_option (int c, const char *arg ATTRIBUTE_UNUSED)
+{
+  switch (c)
+    {
+    default:
+      return 0;
 
 
-               /* Handle load cache hint completer.  */
-               case 'c':
-                 cmpltr = 0;
-                 if (!strncmp(s, ",sl", 3))
-                   {
-                     s += 3;
-                     cmpltr = 2;
-                   }
-                 INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 10);
+#ifdef OBJ_ELF
+    case 'V':
+      print_version_id ();
+      break;
+#endif
+#ifdef WARN_COMMENTS
+    case 'c':
+      warn_comment = 1;
+      break;
+#endif
+    }
 
 
-               /* Handle store cache hint completer.  */
-               case 'C':
-                 cmpltr = 0;
-                 if (!strncmp(s, ",sl", 3))
-                   {
-                     s += 3;
-                     cmpltr = 2;
-                   }
-                 else if (!strncmp(s, ",bc", 3))
-                   {
-                     s += 3;
-                     cmpltr = 1;
-                   }
-                 INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 10);
+  return 1;
+}
 
 
-               /* Handle load and clear cache hint completer.  */
-               case 'd':
-                 cmpltr = 0;
-                 if (!strncmp(s, ",co", 3))
-                   {
-                     s += 3;
-                     cmpltr = 1;
-                   }
-                 INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 10);
+void
+md_show_usage (FILE *stream ATTRIBUTE_UNUSED)
+{
+#ifdef OBJ_ELF
+  fprintf (stream, _("\
+  -Q                      ignored\n"));
+#endif
+#ifdef WARN_COMMENTS
+  fprintf (stream, _("\
+  -c                      print a warning if a comment is found\n"));
+#endif
+}
+\f
+/* We have no need to default values of symbols.  */
 
 
-               /* Handle load ordering completer.  */
-               case 'o':
-                 if (strncmp(s, ",o", 2) != 0)
-                   break;
-                 s += 2;
-                 continue;
+symbolS *
+md_undefined_symbol (char *name ATTRIBUTE_UNUSED)
+{
+  return NULL;
+}
 
 
-               /* Handle a branch gate completer.  */
-               case 'g':
-                 if (strncasecmp (s, ",gate", 5) != 0)
-                   break;
-                 s += 5;
-                 continue;
+#if defined (OBJ_SOM) || defined (ELF_ARG_RELOC)
+#define nonzero_dibits(x) \
+  ((x) | (((x) & 0x55555555) << 1) | (((x) & 0xAAAAAAAA) >> 1))
+#define arg_reloc_stub_needed(CALLER, CALLEE) \
+  (((CALLER) ^ (CALLEE)) & nonzero_dibits (CALLER) & nonzero_dibits (CALLEE))
+#else
+#define arg_reloc_stub_needed(CALLER, CALLEE) 0
+#endif
 
 
-               /* Handle a branch link and push completer.  */
-               case 'p':
-                 if (strncasecmp (s, ",l,push", 7) != 0)
-                   break;
-                 s += 7;
-                 continue;
+/* Apply a fixup to an instruction.  */
 
 
-               /* Handle a branch link completer.  */
-               case 'l':
-                 if (strncasecmp (s, ",l", 2) != 0)
-                   break;
-                 s += 2;
-                 continue;
+void
+md_apply_fix (fixS *fixP, valueT *valP, segT seg ATTRIBUTE_UNUSED)
+{
+  char *fixpos;
+  struct hppa_fix_struct *hppa_fixP;
+  offsetT new_val;
+  int insn, val, fmt;
 
 
-               /* Handle a branch pop completer.  */
-               case 'P':
-                 if (strncasecmp (s, ",pop", 4) != 0)
-                   break;
-                 s += 4;
-                 continue;
+  /* SOM uses R_HPPA_ENTRY and R_HPPA_EXIT relocations which can
+     never be "applied" (they are just markers).  Likewise for
+     R_HPPA_BEGIN_BRTAB and R_HPPA_END_BRTAB.  */
+#ifdef OBJ_SOM
+  if (fixP->fx_r_type == R_HPPA_ENTRY
+      || fixP->fx_r_type == R_HPPA_EXIT
+      || fixP->fx_r_type == R_HPPA_BEGIN_BRTAB
+      || fixP->fx_r_type == R_HPPA_END_BRTAB
+      || fixP->fx_r_type == R_HPPA_BEGIN_TRY)
+    return;
 
 
-               /* Handle a local processor completer.  */
-               case 'L':
-                 if (strncasecmp (s, ",l", 2) != 0)
-                   break;
-                 s += 2;
-                 continue;
+  /* Disgusting.  We must set fx_offset ourselves -- R_HPPA_END_TRY
+     fixups are considered not adjustable, which in turn causes
+     adjust_reloc_syms to not set fx_offset.  Ugh.  */
+  if (fixP->fx_r_type == R_HPPA_END_TRY)
+    {
+      fixP->fx_offset = * valP;
+      return;
+    }
+#endif
+#ifdef OBJ_ELF
+  if (fixP->fx_r_type == (int) R_PARISC_GNU_VTENTRY
+      || fixP->fx_r_type == (int) R_PARISC_GNU_VTINHERIT)
+    return;
+#endif
 
 
-               /* Handle a PROBE read/write completer.  */
-               case 'w':
-                 flag = 0;
-                 if (!strncasecmp (s, ",w", 2))
-                   {
-                     flag = 1;
-                     s += 2;
-                   }
-                 else if (!strncasecmp (s, ",r", 2))
-                   {
-                     flag = 0;
-                     s += 2;
-                   }
+  if (fixP->fx_addsy == NULL && fixP->fx_pcrel == 0)
+    fixP->fx_done = 1;
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 6);
+  /* There should be a HPPA specific fixup associated with the GAS fixup.  */
+  hppa_fixP = (struct hppa_fix_struct *) fixP->tc_fix_data;
+  if (hppa_fixP == NULL)
+    {
+      as_bad_where (fixP->fx_file, fixP->fx_line,
+                   _("no hppa_fixup entry for fixup type 0x%x"),
+                   fixP->fx_r_type);
+      return;
+    }
 
 
-               /* Handle MFCTL wide completer.  */
-               case 'W':
-                 if (strncasecmp (s, ",w", 2) != 0)
-                   break;
-                 s += 2;
-                 continue;
+  fixpos = fixP->fx_frag->fr_literal + fixP->fx_where;
 
 
-               /* Handle an RFI restore completer.  */
-               case 'r':
-                 flag = 0;
-                 if (!strncasecmp (s, ",r", 2))
-                   {
-                     flag = 5;
-                     s += 2;
-                   }
+  if (fixP->fx_size != 4 || hppa_fixP->fx_r_format == 32)
+    {
+      /* Handle constant output. */
+      number_to_chars_bigendian (fixpos, *valP, fixP->fx_size);
+      return;
+    }
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 5);
+  insn = bfd_get_32 (stdoutput, fixpos);
+  fmt = bfd_hppa_insn2fmt (stdoutput, insn);
 
 
-               /* Handle a system control completer.  */
-               case 'Z':
-                 if (*s == ',' && (*(s + 1) == 'm' || *(s + 1) == 'M'))
-                   {
-                     flag = 1;
-                     s += 2;
-                   }
-                 else
-                   flag = 0;
-
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 5);
+  /* If there is a symbol associated with this fixup, then it's something
+     which will need a SOM relocation (except for some PC-relative relocs).
+     In such cases we should treat the "val" or "addend" as zero since it
+     will be added in as needed from fx_offset in tc_gen_reloc.  */
+  if ((fixP->fx_addsy != NULL
+       || fixP->fx_r_type == (int) R_HPPA_NONE)
+#ifdef OBJ_SOM
+      && fmt != 32
+#endif
+      )
+    new_val = ((fmt == 12 || fmt == 17 || fmt == 22) ? 8 : 0);
+#ifdef OBJ_SOM
+  /* These field selectors imply that we do not want an addend.  */
+  else if (hppa_fixP->fx_r_field == e_psel
+          || hppa_fixP->fx_r_field == e_rpsel
+          || hppa_fixP->fx_r_field == e_lpsel
+          || hppa_fixP->fx_r_field == e_tsel
+          || hppa_fixP->fx_r_field == e_rtsel
+          || hppa_fixP->fx_r_field == e_ltsel)
+    new_val = ((fmt == 12 || fmt == 17 || fmt == 22) ? 8 : 0);
+#endif
+  else
+    new_val = hppa_field_adjust (* valP, 0, hppa_fixP->fx_r_field);
 
 
-               /* Handle intermediate/final completer for DCOR.  */
-               case 'i':
-                 flag = 0;
-                 if (!strncasecmp (s, ",i", 2))
-                   {
-                     flag = 1;
-                     s += 2;
-                   }
+  /* Handle pc-relative exceptions from above.  */
+  if ((fmt == 12 || fmt == 17 || fmt == 22)
+      && fixP->fx_addsy
+      && fixP->fx_pcrel
+      && !arg_reloc_stub_needed (symbol_arg_reloc_info (fixP->fx_addsy),
+                                hppa_fixP->fx_arg_reloc)
+#ifdef OBJ_ELF
+      && (* valP - 8 + 8192 < 16384
+         || (fmt == 17 && * valP - 8 + 262144 < 524288)
+         || (fmt == 22 && * valP - 8 + 8388608 < 16777216))
+#endif
+#ifdef OBJ_SOM
+      && (* valP - 8 + 262144 < 524288
+         || (fmt == 22 && * valP - 8 + 8388608 < 16777216))
+#endif
+      && !S_IS_EXTERNAL (fixP->fx_addsy)
+      && !S_IS_WEAK (fixP->fx_addsy)
+      && S_GET_SEGMENT (fixP->fx_addsy) == hppa_fixP->segment
+      && !(fixP->fx_subsy
+          && S_GET_SEGMENT (fixP->fx_subsy) != hppa_fixP->segment))
+    {
+      new_val = hppa_field_adjust (* valP, 0, hppa_fixP->fx_r_field);
+    }
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 6);
+  switch (fmt)
+    {
+    case 10:
+      CHECK_FIELD_WHERE (new_val, 8191, -8192,
+                        fixP->fx_file, fixP->fx_line);
+      val = new_val;
 
 
-               /* Handle zero/sign extension completer.  */
-               case 'z':
-                 flag = 1;
-                 if (!strncasecmp (s, ",z", 2))
-                   {
-                     flag = 0;
-                     s += 2;
-                   }
+      insn = (insn & ~ 0x3ff1) | (((val & 0x1ff8) << 1)
+                                 | ((val & 0x2000) >> 13));
+      break;
+    case -11:
+      CHECK_FIELD_WHERE (new_val, 8191, -8192,
+                        fixP->fx_file, fixP->fx_line);
+      val = new_val;
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 10);
+      insn = (insn & ~ 0x3ff9) | (((val & 0x1ffc) << 1)
+                                 | ((val & 0x2000) >> 13));
+      break;
+      /* Handle all opcodes with the 'j' operand type.  */
+    case 14:
+      CHECK_FIELD_WHERE (new_val, 8191, -8192,
+                        fixP->fx_file, fixP->fx_line);
+      val = new_val;
 
 
-               /* Handle add completer.  */
-               case 'a':
-                 flag = 1;
-                 if (!strncasecmp (s, ",l", 2))
-                   {
-                     flag = 2;
-                     s += 2;
-                   }
-                 else if (!strncasecmp (s, ",tsv", 4))
-                   {
-                     flag = 3;
-                     s += 4;
-                   }
+      insn = ((insn & ~ 0x3fff) | low_sign_unext (val, 14));
+      break;
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 10);
+      /* Handle all opcodes with the 'k' operand type.  */
+    case 21:
+      CHECK_FIELD_WHERE (new_val, 1048575, -1048576,
+                        fixP->fx_file, fixP->fx_line);
+      val = new_val;
 
 
-               /* Handle 64 bit carry for ADD.  */
-               case 'Y':
-                 flag = 0;
-                 if (!strncasecmp (s, ",dc,tsv", 7) ||
-                     !strncasecmp (s, ",tsv,dc", 7))
-                   {
-                     flag = 1;
-                     s += 7;
-                   }
-                 else if (!strncasecmp (s, ",dc", 3))
-                   {
-                     flag = 0;
-                     s += 3;
-                   }
-                 else
-                   break;
+      insn = (insn & ~ 0x1fffff) | re_assemble_21 (val);
+      break;
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+      /* Handle all the opcodes with the 'i' operand type.  */
+    case 11:
+      CHECK_FIELD_WHERE (new_val, 1023, -1024,
+                        fixP->fx_file, fixP->fx_line);
+      val = new_val;
 
 
-               /* Handle 32 bit carry for ADD.  */
-               case 'y':
-                 flag = 0;
-                 if (!strncasecmp (s, ",c,tsv", 6) ||
-                     !strncasecmp (s, ",tsv,c", 6))
-                   {
-                     flag = 1;
-                     s += 6;
-                   }
-                 else if (!strncasecmp (s, ",c", 2))
-                   {
-                     flag = 0;
-                     s += 2;
-                   }
-                 else
-                   break;
+      insn = (insn & ~ 0x7ff) | low_sign_unext (val, 11);
+      break;
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+      /* Handle all the opcodes with the 'w' operand type.  */
+    case 12:
+      CHECK_FIELD_WHERE (new_val - 8, 8191, -8192,
+                        fixP->fx_file, fixP->fx_line);
+      val = new_val - 8;
 
 
-               /* Handle trap on signed overflow.  */
-               case 'v':
-                 flag = 0;
-                 if (!strncasecmp (s, ",tsv", 4))
-                   {
-                     flag = 1;
-                     s += 4;
-                   }
+      insn = (insn & ~ 0x1ffd) | re_assemble_12 (val >> 2);
+      break;
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+      /* Handle some of the opcodes with the 'W' operand type.  */
+    case 17:
+      {
+       offsetT distance = * valP;
 
 
-               /* Handle trap on condition and overflow.  */
-               case 't':
-                 flag = 0;
-                 if (!strncasecmp (s, ",tc,tsv", 7) ||
-                     !strncasecmp (s, ",tsv,tc", 7))
-                   {
-                     flag = 1;
-                     s += 7;
-                   }
-                 else if (!strncasecmp (s, ",tc", 3))
-                   {
-                     flag = 0;
-                     s += 3;
-                   }
-                 else
-                   break;
+       /* If this is an absolute branch (ie no link) with an out of
+          range target, then we want to complain.  */
+       if (fixP->fx_r_type == (int) R_HPPA_PCREL_CALL
+           && (insn & 0xffe00000) == 0xe8000000)
+         CHECK_FIELD_WHERE (distance - 8, 262143, -262144,
+                            fixP->fx_file, fixP->fx_line);
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+       CHECK_FIELD_WHERE (new_val - 8, 262143, -262144,
+                          fixP->fx_file, fixP->fx_line);
+       val = new_val - 8;
 
 
-               /* Handle 64 bit borrow for SUB.  */
-               case 'B':
-                 flag = 0;
-                 if (!strncasecmp (s, ",db,tsv", 7) ||
-                     !strncasecmp (s, ",tsv,db", 7))
-                   {
-                     flag = 1;
-                     s += 7;
-                   }
-                 else if (!strncasecmp (s, ",db", 3))
-                   {
-                     flag = 0;
-                     s += 3;
-                   }
-                 else
-                   break;
+       insn = (insn & ~ 0x1f1ffd) | re_assemble_17 (val >> 2);
+       break;
+      }
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+    case 22:
+      {
+       offsetT distance = * valP;
 
 
-               /* Handle 32 bit borrow for SUB.  */
-               case 'b':
-                 flag = 0;
-                 if (!strncasecmp (s, ",b,tsv", 6) ||
-                     !strncasecmp (s, ",tsv,b", 6))
-                   {
-                     flag = 1;
-                     s += 6;
-                   }
-                 else if (!strncasecmp (s, ",b", 2))
-                   {
-                     flag = 0;
-                     s += 2;
-                   }
-                 else
-                   break;
+       /* If this is an absolute branch (ie no link) with an out of
+          range target, then we want to complain.  */
+       if (fixP->fx_r_type == (int) R_HPPA_PCREL_CALL
+           && (insn & 0xffe00000) == 0xe8000000)
+         CHECK_FIELD_WHERE (distance - 8, 8388607, -8388608,
+                            fixP->fx_file, fixP->fx_line);
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+       CHECK_FIELD_WHERE (new_val - 8, 8388607, -8388608,
+                          fixP->fx_file, fixP->fx_line);
+       val = new_val - 8;
 
 
-               /* Handle trap condition completer for UADDCM.  */
-               case 'T':
-                 flag = 0;
-                 if (!strncasecmp (s, ",tc", 3))
-                   {
-                     flag = 1;
-                     s += 3;
-                   }
+       insn = (insn & ~ 0x3ff1ffd) | re_assemble_22 (val >> 2);
+       break;
+      }
 
 
-                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 6);
+    case -10:
+      val = new_val;
+      insn = (insn & ~ 0xfff1) | re_assemble_16 (val & -8);
+      break;
 
 
-               /* Handle signed/unsigned at 21.  */
-               case 'S':
-                 {
-                   int sign = 1;
-                   if (strncasecmp (s, ",s", 2) == 0)
-                     {
-                       sign = 1;
-                       s += 2;
-                     }
-                   else if (strncasecmp (s, ",u", 2) == 0)
-                     {
-                       sign = 0;
-                       s += 2;
-                     }
+    case -16:
+      val = new_val;
+      insn = (insn & ~ 0xfff9) | re_assemble_16 (val & -4);
+      break;
 
 
-                   INSERT_FIELD_AND_CONTINUE (opcode, sign, 10);
-                 }
+    case 16:
+      val = new_val;
+      insn = (insn & ~ 0xffff) | re_assemble_16 (val);
+      break;
 
 
-               /* Handle left/right combination at 17:18.  */
-               case 'h':
-                 if (*s++ == ',')
-                   {
-                     int lr = 0;
-                     if (*s == 'r')
-                       lr = 2;
-                     else if (*s == 'l')
-                       lr = 0;
-                     else
-                       as_bad(_("Invalid left/right combination completer"));
+    case 32:
+      insn = new_val;
+      break;
 
 
-                     s++;
-                     INSERT_FIELD_AND_CONTINUE (opcode, lr, 13);
-                   }
-                 else
-                   as_bad(_("Invalid left/right combination completer"));
-                 break;
+    default:
+      as_bad_where (fixP->fx_file, fixP->fx_line,
+                   _("Unknown relocation encountered in md_apply_fix."));
+      return;
+    }
 
 
-               /* Handle saturation at 24:25.  */
-               case 'H':
-                 {
-                   int sat = 3;
-                   if (strncasecmp (s, ",ss", 3) == 0)
-                     {
-                       sat = 1;
-                       s += 3;
-                     }
-                   else if (strncasecmp (s, ",us", 3) == 0)
-                     {
-                       sat = 0;
-                       s += 3;
-                     }
+#ifdef OBJ_ELF
+  switch (fixP->fx_r_type)
+    {
+      case R_PARISC_TLS_GD21L:
+      case R_PARISC_TLS_GD14R:
+      case R_PARISC_TLS_LDM21L:
+      case R_PARISC_TLS_LDM14R:
+      case R_PARISC_TLS_LE21L:
+      case R_PARISC_TLS_LE14R:
+      case R_PARISC_TLS_IE21L:
+      case R_PARISC_TLS_IE14R:
+       if (fixP->fx_addsy)
+         S_SET_THREAD_LOCAL (fixP->fx_addsy);
+       break;
+      default:
+       break;
+    }
+#endif
 
 
-                   INSERT_FIELD_AND_CONTINUE (opcode, sat, 6);
-                 }
+  /* Insert the relocation.  */
+  bfd_put_32 (stdoutput, insn, fixpos);
+}
 
 
-               /* Handle permutation completer.  */
-               case '*':
-                 if (*s++ == ',')
-                   {
-                     int permloc[4];
-                     int perm = 0;
-                     int i = 0;
-                     permloc[0] = 13;
-                     permloc[1] = 10;
-                     permloc[2] = 8;
-                     permloc[3] = 6;
-                     for (; i < 4; i++)
-                       {
-                         switch (*s++)
-                           {
-                           case '0':
-                             perm = 0;
-                             break;
-                           case '1':
-                             perm = 1;
-                             break;
-                           case '2':
-                             perm = 2;
-                             break;
-                           case '3':
-                             perm = 3;
-                             break;
-                           default:
-                             as_bad(_("Invalid permutation completer"));
-                           }
-                         opcode |= perm << permloc[i];
-                       }
-                     continue;
-                   }
-                 else
-                   as_bad(_("Invalid permutation completer"));
-                 break;
+/* Exactly what point is a PC-relative offset relative TO?
+   On the PA, they're relative to the address of the offset.  */
 
 
-               default:
-                 abort ();
-               }
-             break;
+long
+md_pcrel_from (fixS *fixP)
+{
+  return fixP->fx_where + fixP->fx_frag->fr_address;
+}
 
 
-           /* Handle all conditions.  */
-           case '?':
-             {
-               args++;
-               switch (*args)
-                 {
-                 /* Handle FP compare conditions.  */
-                 case 'f':
-                   cond = pa_parse_fp_cmp_cond (&s);
-                   INSERT_FIELD_AND_CONTINUE (opcode, cond, 0);
+/* Return nonzero if the input line pointer is at the end of
+   a statement.  */
 
 
-                 /* Handle an add condition.  */
-                 case 'A':
-                 case 'a':
-                   cmpltr = 0;
-                   flag = 0;
-                   if (*s == ',')
-                     {
-                       s++;
+static int
+is_end_of_statement (void)
+{
+  return ((*input_line_pointer == '\n')
+         || (*input_line_pointer == ';')
+         || (*input_line_pointer == '!'));
+}
 
 
-                       /* 64 bit conditions.  */
-                       if (*args == 'A')
-                         {
-                           if (*s == '*')
-                             s++;
-                           else
-                             break;
-                         }
-                       else if (*s == '*')
-                         break;
-                       name = s;
+#define REG_NAME_CNT   (sizeof (pre_defined_registers) / sizeof (struct pd_reg))
 
 
-                       name = s;
-                       while (*s != ',' && *s != ' ' && *s != '\t')
-                         s += 1;
-                       c = *s;
-                       *s = 0x00;
-                       if (strcmp (name, "=") == 0)
-                         cmpltr = 1;
-                       else if (strcmp (name, "<") == 0)
-                         cmpltr = 2;
-                       else if (strcmp (name, "<=") == 0)
-                         cmpltr = 3;
-                       else if (strcasecmp (name, "nuv") == 0)
-                         cmpltr = 4;
-                       else if (strcasecmp (name, "znv") == 0)
-                         cmpltr = 5;
-                       else if (strcasecmp (name, "sv") == 0)
-                         cmpltr = 6;
-                       else if (strcasecmp (name, "od") == 0)
-                         cmpltr = 7;
-                       else if (strcasecmp (name, "tr") == 0)
-                         {
-                           cmpltr = 0;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, "<>") == 0)
-                         {
-                           cmpltr = 1;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, ">=") == 0)
-                         {
-                           cmpltr = 2;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, ">") == 0)
-                         {
-                           cmpltr = 3;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "uv") == 0)
-                         {
-                           cmpltr = 4;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "vnz") == 0)
-                         {
-                           cmpltr = 5;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "nsv") == 0)
-                         {
-                           cmpltr = 6;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "ev") == 0)
-                         {
-                           cmpltr = 7;
-                           flag = 1;
-                         }
-                       /* ",*" is a valid condition.  */
-                       else if (*args == 'a')
-                         as_bad (_("Invalid Add Condition: %s"), name);
-                       *s = c;
-                     }
-                   opcode |= cmpltr << 13;
-                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
+/* Given NAME, find the register number associated with that name, return
+   the integer value associated with the given name or -1 on failure.  */
 
 
-                 /* Handle non-negated add and branch condition.  */
-                 case 'd':
-                   cmpltr = pa_parse_nonneg_add_cmpltr (&s, 1);
-                   if (cmpltr < 0)
-                     {
-                       as_bad (_("Invalid Add and Branch Condition: %c"), *s);
-                       cmpltr = 0;
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
+static int
+reg_name_search (char *name)
+{
+  int middle, low, high;
+  int cmp;
 
 
-                 /* Handle 64 bit wide-mode add and branch condition.  */
-                 case 'W':
-                   cmpltr = pa_parse_addb_64_cmpltr (&s);
-                   if (cmpltr < 0)
-                     {
-                       as_bad (_("Invalid Add and Branch Condition: %c"), *s);
-                       cmpltr = 0;
-                     }
-                   else
-                     {
-                       /* Negated condition requires an opcode change.  */
-                       opcode |= (cmpltr & 8) << 24;
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr & 7, 13);
+  low = 0;
+  high = REG_NAME_CNT - 1;
 
 
-                 /* Handle a negated or non-negated add and branch
-                    condition.  */
-                 case '@':
-                   save_s = s;
-                   cmpltr = pa_parse_nonneg_add_cmpltr (&s, 1);
-                   if (cmpltr < 0)
-                     {
-                       s = save_s;
-                       cmpltr = pa_parse_neg_add_cmpltr (&s, 1);
-                       if (cmpltr < 0)
-                         {
-                           as_bad (_("Invalid Compare/Subtract Condition"));
-                           cmpltr = 0;
-                         }
-                       else
-                         {
-                           /* Negated condition requires an opcode change.  */
-                           opcode |= 1 << 27;
-                         }
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
+  do
+    {
+      middle = (low + high) / 2;
+      cmp = strcasecmp (name, pre_defined_registers[middle].name);
+      if (cmp < 0)
+       high = middle - 1;
+      else if (cmp > 0)
+       low = middle + 1;
+      else
+       return pre_defined_registers[middle].value;
+    }
+  while (low <= high);
 
 
-                 /* Handle branch on bit conditions.  */
-                 case 'B':
-                 case 'b':
-                   cmpltr = 0;
-                   if (*s == ',')
-                     {
-                       s++;
-
-                       if (*args == 'B')
-                         {
-                           if (*s == '*')
-                             s++;
-                           else
-                             break;
-                         }
-                       else if (*s == '*')
-                         break;
-
-                       if (strncmp (s, "<", 1) == 0)
-                         {
-                           cmpltr = 0;
-                           s++;
-                         }
-                       else if (strncmp (s, ">=", 2) == 0)
-                         {
-                           cmpltr = 1;
-                           s += 2;
-                         }
-                       else
-                         as_bad (_("Invalid Bit Branch Condition: %c"), *s);
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 15);
-
-                 /* Handle a compare/subtract condition.  */
-                 case 'S':
-                 case 's':
-                   cmpltr = 0;
-                   flag = 0;
-                   if (*s == ',')
-                     {
-                       s++;
-
-                       /* 64 bit conditions.  */
-                       if (*args == 'S')
-                         {
-                           if (*s == '*')
-                             s++;
-                           else
-                             break;
-                         }
-                       else if (*s == '*')
-                         break;
-                       name = s;
-
-                       name = s;
-                       while (*s != ',' && *s != ' ' && *s != '\t')
-                         s += 1;
-                       c = *s;
-                       *s = 0x00;
-                       if (strcmp (name, "=") == 0)
-                         cmpltr = 1;
-                       else if (strcmp (name, "<") == 0)
-                         cmpltr = 2;
-                       else if (strcmp (name, "<=") == 0)
-                         cmpltr = 3;
-                       else if (strcasecmp (name, "<<") == 0)
-                         cmpltr = 4;
-                       else if (strcasecmp (name, "<<=") == 0)
-                         cmpltr = 5;
-                       else if (strcasecmp (name, "sv") == 0)
-                         cmpltr = 6;
-                       else if (strcasecmp (name, "od") == 0)
-                         cmpltr = 7;
-                       else if (strcasecmp (name, "tr") == 0)
-                         {
-                           cmpltr = 0;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, "<>") == 0)
-                         {
-                           cmpltr = 1;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, ">=") == 0)
-                         {
-                           cmpltr = 2;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, ">") == 0)
-                         {
-                           cmpltr = 3;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, ">>=") == 0)
-                         {
-                           cmpltr = 4;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, ">>") == 0)
-                         {
-                           cmpltr = 5;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "nsv") == 0)
-                         {
-                           cmpltr = 6;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "ev") == 0)
-                         {
-                           cmpltr = 7;
-                           flag = 1;
-                         }
-                       /* ",*" is a valid condition.  */
-                       else if (*args != 'S')
-                         as_bad (_("Invalid Compare/Subtract Condition: %s"),
-                                 name);
-                       *s = c;
-                     }
-                   opcode |= cmpltr << 13;
-                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
-
-                 /* Handle a non-negated compare condition.  */
-                 case 't':
-                   cmpltr = pa_parse_nonneg_cmpsub_cmpltr (&s, 1);
-                   if (cmpltr < 0)
-                     {
-                       as_bad (_("Invalid Compare/Subtract Condition: %c"), *s);
-                       cmpltr = 0;
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
+  return -1;
+}
 
 
-                 /* Handle a 32 bit compare and branch condition.  */
-                 case 'n':
-                   save_s = s;
-                   cmpltr = pa_parse_nonneg_cmpsub_cmpltr (&s, 1);
-                   if (cmpltr < 0)
-                     {
-                       s = save_s;
-                       cmpltr = pa_parse_neg_cmpsub_cmpltr (&s, 1);
-                       if (cmpltr < 0)
-                         {
-                           as_bad (_("Invalid Compare and Branch Condition."));
-                           cmpltr = 0;
-                         }
-                       else
-                         {
-                           /* Negated condition requires an opcode change.  */
-                           opcode |= 1 << 27;
-                         }
-                     }
+/* Read a number from S.  The number might come in one of many forms,
+   the most common will be a hex or decimal constant, but it could be
+   a pre-defined register (Yuk!), or an absolute symbol.
 
 
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
+   Return 1 on success or 0 on failure.  If STRICT, then a missing
+   register prefix will cause a failure.  The number itself is
+   returned in `pa_number'.
 
 
-                 /* Handle a 64 bit compare and branch condition.  */
-                 case 'N':
-                   cmpltr = pa_parse_cmpb_64_cmpltr (&s);
-                   if (cmpltr >= 0)
-                     {
-                       /* Negated condition requires an opcode change.  */
-                       opcode |= (cmpltr & 8) << 26;
-                     }
-                   else
-                     /* Not a 64 bit cond.  Give 32 bit a chance.  */
-                     break;
+   IS_FLOAT indicates that a PA-89 FP register number should be
+   parsed;  A `l' or `r' suffix is checked for if but 2 of IS_FLOAT is
+   not set.
 
 
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr & 7, 13);
+   pa_parse_number can not handle negative constants and will fail
+   horribly if it is passed such a constant.  */
 
 
-                 /* Handle a 64 bit cmpib condition.  */
-                 case 'Q':
-                   cmpltr = pa_parse_cmpib_64_cmpltr (&s);
-                   if (cmpltr < 0)
-                     /* Not a 64 bit cond.  Give 32 bit a chance.  */
-                     break;
+static int
+pa_parse_number (char **s, int is_float)
+{
+  int num;
+  char *name;
+  char c;
+  symbolS *sym;
+  int status;
+  char *p = *s;
+  bfd_boolean have_prefix;
 
 
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
+  /* Skip whitespace before the number.  */
+  while (*p == ' ' || *p == '\t')
+    p = p + 1;
 
 
-                   /* Handle a logical instruction condition.  */
-                 case 'L':
-                 case 'l':
-                   cmpltr = 0;
-                   flag = 0;
-                   if (*s == ',')
-                     {
-                       s++;
+  pa_number = -1;
+  have_prefix = 0;
+  num = 0;
+  if (!strict && ISDIGIT (*p))
+    {
+      /* Looks like a number.  */
 
 
-                       /* 64 bit conditions.  */
-                       if (*args == 'L')
-                         {
-                           if (*s == '*')
-                             s++;
-                           else
-                             break;
-                         }
-                       else if (*s == '*')
-                         break;
+      if (*p == '0' && (*(p + 1) == 'x' || *(p + 1) == 'X'))
+       {
+         /* The number is specified in hex.  */
+         p += 2;
+         while (ISDIGIT (*p) || ((*p >= 'a') && (*p <= 'f'))
+                || ((*p >= 'A') && (*p <= 'F')))
+           {
+             if (ISDIGIT (*p))
+               num = num * 16 + *p - '0';
+             else if (*p >= 'a' && *p <= 'f')
+               num = num * 16 + *p - 'a' + 10;
+             else
+               num = num * 16 + *p - 'A' + 10;
+             ++p;
+           }
+       }
+      else
+       {
+         /* The number is specified in decimal.  */
+         while (ISDIGIT (*p))
+           {
+             num = num * 10 + *p - '0';
+             ++p;
+           }
+       }
 
 
-                       name = s;
-                       while (*s != ',' && *s != ' ' && *s != '\t')
-                         s += 1;
-                       c = *s;
-                       *s = 0x00;
+      pa_number = num;
 
 
-                       if (strcmp (name, "=") == 0)
-                         cmpltr = 1;
-                       else if (strcmp (name, "<") == 0)
-                         cmpltr = 2;
-                       else if (strcmp (name, "<=") == 0)
-                         cmpltr = 3;
-                       else if (strcasecmp (name, "od") == 0)
-                         cmpltr = 7;
-                       else if (strcasecmp (name, "tr") == 0)
-                         {
-                           cmpltr = 0;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, "<>") == 0)
-                         {
-                           cmpltr = 1;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, ">=") == 0)
-                         {
-                           cmpltr = 2;
-                           flag = 1;
-                         }
-                       else if (strcmp (name, ">") == 0)
-                         {
-                           cmpltr = 3;
-                           flag = 1;
-                         }
-                       else if (strcasecmp (name, "ev") == 0)
-                         {
-                           cmpltr = 7;
-                           flag = 1;
-                         }
-                       /* ",*" is a valid condition.  */
-                       else if (*args != 'L')
-                         as_bad (_("Invalid Logical Instruction Condition."));
-                       *s = c;
-                     }
-                   opcode |= cmpltr << 13;
-                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
-
-                 /* Handle a shift/extract/deposit condition.  */
-                 case 'X':
-                 case 'x':
-                 case 'y':
-                   cmpltr = 0;
-                   if (*s == ',')
-                     {
-                       save_s = s++;
-
-                       /* 64 bit conditions.  */
-                       if (*args == 'X')
-                         {
-                           if (*s == '*')
-                             s++;
-                           else
-                             break;
-                         }
-                       else if (*s == '*')
-                         break;
-
-                       name = s;
-                       while (*s != ',' && *s != ' ' && *s != '\t')
-                         s += 1;
-                       c = *s;
-                       *s = 0x00;
-                       if (strcmp (name, "=") == 0)
-                         cmpltr = 1;
-                       else if (strcmp (name, "<") == 0)
-                         cmpltr = 2;
-                       else if (strcasecmp (name, "od") == 0)
-                         cmpltr = 3;
-                       else if (strcasecmp (name, "tr") == 0)
-                         cmpltr = 4;
-                       else if (strcmp (name, "<>") == 0)
-                         cmpltr = 5;
-                       else if (strcmp (name, ">=") == 0)
-                         cmpltr = 6;
-                       else if (strcasecmp (name, "ev") == 0)
-                         cmpltr = 7;
-                       /* Handle movb,n.  Put things back the way they were.
-                          This includes moving s back to where it started.  */
-                       else if (strcasecmp (name, "n") == 0 && *args == 'y')
-                         {
-                           *s = c;
-                           s = save_s;
-                           continue;
-                         }
-                       /* ",*" is a valid condition.  */
-                       else if (*args != 'X')
-                         as_bad (_("Invalid Shift/Extract/Deposit Condition."));
-                       *s = c;
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
-
-                 /* Handle a unit instruction condition.  */
-                 case 'U':
-                 case 'u':
-                   cmpltr = 0;
-                   flag = 0;
-                   if (*s == ',')
-                     {
-                       s++;
-
-                       /* 64 bit conditions.  */
-                       if (*args == 'U')
-                         {
-                           if (*s == '*')
-                             s++;
-                           else
-                             break;
-                         }
-                       else if (*s == '*')
-                         break;
-
-                       if (strncasecmp (s, "sbz", 3) == 0)
-                         {
-                           cmpltr = 2;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "shz", 3) == 0)
-                         {
-                           cmpltr = 3;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "sdc", 3) == 0)
-                         {
-                           cmpltr = 4;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "sbc", 3) == 0)
-                         {
-                           cmpltr = 6;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "shc", 3) == 0)
-                         {
-                           cmpltr = 7;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "tr", 2) == 0)
-                         {
-                           cmpltr = 0;
-                           flag = 1;
-                           s += 2;
-                         }
-                       else if (strncasecmp (s, "nbz", 3) == 0)
-                         {
-                           cmpltr = 2;
-                           flag = 1;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "nhz", 3) == 0)
-                         {
-                           cmpltr = 3;
-                           flag = 1;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "ndc", 3) == 0)
-                         {
-                           cmpltr = 4;
-                           flag = 1;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "nbc", 3) == 0)
-                         {
-                           cmpltr = 6;
-                           flag = 1;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "nhc", 3) == 0)
-                         {
-                           cmpltr = 7;
-                           flag = 1;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "swz", 3) == 0)
-                         {
-                           cmpltr = 1;
-                           flag = 0;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "swc", 3) == 0)
-                         {
-                           cmpltr = 5;
-                           flag = 0;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "nwz", 3) == 0)
-                         {
-                           cmpltr = 1;
-                           flag = 1;
-                           s += 3;
-                         }
-                       else if (strncasecmp (s, "nwc", 3) == 0)
-                         {
-                           cmpltr = 5;
-                           flag = 1;
-                           s += 3;
-                         }
-                       /* ",*" is a valid condition.  */
-                       else if (*args != 'U')
-                         as_bad (_("Invalid Unit Instruction Condition."));
-                     }
-                   opcode |= cmpltr << 13;
-                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
-
-                 default:
-                   abort ();
-                 }
-               break;
-             }
-
-           /* Handle a nullification completer for branch instructions.  */
-           case 'n':
-             nullif = pa_parse_nullif (&s);
-             INSERT_FIELD_AND_CONTINUE (opcode, nullif, 1);
-
-           /* Handle a nullification completer for copr and spop insns.  */
-           case 'N':
-             nullif = pa_parse_nullif (&s);
-             INSERT_FIELD_AND_CONTINUE (opcode, nullif, 5);
-
-           /* Handle ,%r2 completer for new syntax branches.  */
-           case 'L':
-             if (*s == ',' && strncasecmp (s + 1, "%r2", 3) == 0)
-               s += 4;
-             else if (*s == ',' && strncasecmp (s + 1, "%rp", 3) == 0)
-               s += 4;
-             else
-               break;
-             continue;
-
-           /* Handle 3 bit entry into the fp compare array.   Valid values
-              are 0..6 inclusive.  */
-           case 'h':
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 6, 0, 0);
-                 num++;
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 13);
-               }
-             else
-               break;
-
-           /* Handle 3 bit entry into the fp compare array.   Valid values
-              are 0..6 inclusive.  */
-           case 'm':
-             get_expression (s);
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 s = expr_end;
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 6, 0, 0);
-                 num = (num + 1) ^ 1;
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 13);
-               }
-             else
-               break;
-
-           /* Handle graphics test completers for ftest */
-           case '=':
-             {
-               num = pa_parse_ftest_gfx_completer (&s);
-               INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-             }
-
-           /* Handle a 11 bit immediate at 31.  */
-           case 'i':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 1023, -1024, 0);
-                 num = low_sign_unext (num, 11);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-               }
-             else
+      /* Check for a `l' or `r' suffix.  */
+      if (is_float)
+       {
+         pa_number += FP_REG_BASE;
+         if (! (is_float & 2))
+           {
+             if (IS_R_SELECT (p))
                {
                {
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 11;
-                 continue;
+                 pa_number += FP_REG_RSEL;
+                 ++p;
                }
                }
-
-           /* Handle a 14 bit immediate at 31.  */
-           case 'J':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 int mb;
-
-                 /* XXX the completer stored away tidbits of information
-                    for us to extract.  We need a cleaner way to do this.
-                    Now that we have lots of letters again, it would be
-                    good to rethink this.  */
-                 mb = opcode & 1;
-                 opcode -= mb;
-                 num = evaluate_absolute (&the_insn);
-                 if (mb != (num < 0))
-                   break;
-                 CHECK_FIELD (num, 8191, -8192, 0);
-                 num = low_sign_unext (num, 14);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+             else if (IS_L_SELECT (p))
+               {
+                 ++p;
                }
                }
-             break;
+           }
+       }
+    }
+  else if (*p == '%')
+    {
+      /* The number might be a predefined register.  */
+      have_prefix = 1;
+      name = p;
+      p++;
+      c = *p;
+      /* Tege hack: Special case for general registers as the general
+        code makes a binary search with case translation, and is VERY
+        slow.  */
+      if (c == 'r')
+       {
+         p++;
+         if (*p == 'e' && *(p + 1) == 't'
+             && (*(p + 2) == '0' || *(p + 2) == '1'))
+           {
+             p += 2;
+             num = *p - '0' + 28;
+             p++;
+           }
+         else if (*p == 'p')
+           {
+             num = 2;
+             p++;
+           }
+         else if (!ISDIGIT (*p))
+           {
+             if (print_errors)
+               as_bad (_("Undefined register: '%s'."), name);
+             num = -1;
+           }
+         else
+           {
+             do
+               num = num * 10 + *p++ - '0';
+             while (ISDIGIT (*p));
+           }
+       }
+      else
+       {
+         /* Do a normal register search.  */
+         while (is_part_of_name (c))
+           {
+             p = p + 1;
+             c = *p;
+           }
+         *p = 0;
+         status = reg_name_search (name);
+         if (status >= 0)
+           num = status;
+         else
+           {
+             if (print_errors)
+               as_bad (_("Undefined register: '%s'."), name);
+             num = -1;
+           }
+         *p = c;
+       }
 
 
-           /* Handle a 14 bit immediate at 31.  */
-           case 'K':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 int mb;
+      pa_number = num;
+    }
+  else
+    {
+      /* And finally, it could be a symbol in the absolute section which
+        is effectively a constant, or a register alias symbol.  */
+      name = p;
+      c = *p;
+      while (is_part_of_name (c))
+       {
+         p = p + 1;
+         c = *p;
+       }
+      *p = 0;
+      if ((sym = symbol_find (name)) != NULL)
+       {
+         if (S_GET_SEGMENT (sym) == reg_section)
+           {
+             num = S_GET_VALUE (sym);
+             /* Well, we don't really have one, but we do have a
+                register, so...  */
+             have_prefix = TRUE;
+           }
+         else if (S_GET_SEGMENT (sym) == bfd_abs_section_ptr)
+           num = S_GET_VALUE (sym);
+         else if (!strict)
+           {
+             if (print_errors)
+               as_bad (_("Non-absolute symbol: '%s'."), name);
+             num = -1;
+           }
+       }
+      else if (!strict)
+       {
+         /* There is where we'd come for an undefined symbol
+            or for an empty string.  For an empty string we
+            will return zero.  That's a concession made for
+            compatibility with the braindamaged HP assemblers.  */
+         if (*name == 0)
+           num = 0;
+         else
+           {
+             if (print_errors)
+               as_bad (_("Undefined absolute constant: '%s'."), name);
+             num = -1;
+           }
+       }
+      *p = c;
 
 
-                 mb = opcode & 1;
-                 opcode -= mb;
-                 num = evaluate_absolute (&the_insn);
-                 if (mb == (num < 0))
-                   break;
-                 if (num % 4)
-                   break;
-                 CHECK_FIELD (num, 8191, -8192, 0);
-                 num = low_sign_unext (num, 14);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-               }
-             break;
+      pa_number = num;
+    }
 
 
-           /* Handle a 16 bit immediate at 31.  */
-           case '<':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 int mb;
+  if (!strict || have_prefix)
+    {
+      *s = p;
+      return 1;
+    }
+  return 0;
+}
 
 
-                 mb = opcode & 1;
-                 opcode -= mb;
-                 num = evaluate_absolute (&the_insn);
-                 if (mb != (num < 0))
-                   break;
-                 CHECK_FIELD (num, 32767, -32768, 0);
-                 num = re_assemble_16 (num);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-               }
-             break;
+/* Return nonzero if the given INSN and L/R information will require
+   a new PA-1.1 opcode.  */
 
 
-           /* Handle a 16 bit immediate at 31.  */
-           case '>':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 int mb;
+static int
+need_pa11_opcode (void)
+{
+  if ((pa_number & FP_REG_RSEL) != 0
+      && !(the_insn.fpof1 == DBL && the_insn.fpof2 == DBL))
+    {
+      /* If this instruction is specific to a particular architecture,
+        then set a new architecture.  */
+      if (bfd_get_mach (stdoutput) < pa11)
+       {
+         if (!bfd_set_arch_mach (stdoutput, bfd_arch_hppa, pa11))
+           as_warn (_("could not update architecture and machine"));
+       }
+      return TRUE;
+    }
+  else
+    return FALSE;
+}
 
 
-                 mb = opcode & 1;
-                 opcode -= mb;
-                 num = evaluate_absolute (&the_insn);
-                 if (mb == (num < 0))
-                   break;
-                 if (num % 4)
-                   break;
-                 CHECK_FIELD (num, 32767, -32768, 0);
-                 num = re_assemble_16 (num);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-               }
-             break;
+/* Parse a condition for a fcmp instruction.  Return the numerical
+   code associated with the condition.  */
 
 
-           /* Handle 14 bit immediate, shifted left three times.  */
-           case '#':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 if (num & 0x7)
-                   break;
-                 CHECK_FIELD (num, 8191, -8192, 0);
-                 if (num < 0)
-                   opcode |= 1;
-                 num &= 0x1fff;
-                 num >>= 3;
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 4);
-               }
-             else
-               {
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 14;
-                 continue;
-               }
-             break;
+static int
+pa_parse_fp_cmp_cond (char **s)
+{
+  int cond, i;
 
 
-           /* Handle 14 bit immediate, shifted left twice.  */
-           case 'd':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 if (num & 0x3)
-                   break;
-                 CHECK_FIELD (num, 8191, -8192, 0);
-                 if (num < 0)
-                   opcode |= 1;
-                 num &= 0x1fff;
-                 num >>= 2;
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 3);
-               }
-             else
-               {
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 14;
-                 continue;
-               }
+  cond = 0;
+
+  for (i = 0; i < 32; i++)
+    {
+      if (strncasecmp (*s, fp_cond_map[i].string,
+                      strlen (fp_cond_map[i].string)) == 0)
+       {
+         cond = fp_cond_map[i].cond;
+         *s += strlen (fp_cond_map[i].string);
+         /* If not a complete match, back up the input string and
+            report an error.  */
+         if (**s != ' ' && **s != '\t')
+           {
+             *s -= strlen (fp_cond_map[i].string);
+             break;
+           }
+         while (**s == ' ' || **s == '\t')
+           *s = *s + 1;
+         return cond;
+       }
+    }
 
 
-           /* Handle a 14 bit immediate at 31.  */
-           case 'j':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 8191, -8192, 0);
-                 num = low_sign_unext (num, 14);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-               }
-             else
-               {
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 14;
-                 continue;
-               }
+  as_bad (_("Invalid FP Compare Condition: %s"), *s);
 
 
-           /* Handle a 21 bit immediate at 31.  */
-           case 'k':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num >> 11, 1048575, -1048576, 0);
-                 opcode |= re_assemble_21 (num);
-                 continue;
-               }
-             else
-               {
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 21;
-                 continue;
-               }
+  /* Advance over the bogus completer.  */
+  while (**s != ',' && **s != ' ' && **s != '\t')
+    *s += 1;
 
 
-           /* Handle a 16 bit immediate at 31 (PA 2.0 wide mode only).  */
-           case 'l':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 32767, -32768, 0);
-                 opcode |= re_assemble_16 (num);
-                 continue;
-               }
-             else
-               {
-                 /* ??? Is this valid for wide mode?  */
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 14;
-                 continue;
-               }
+  return 0;
+}
 
 
-           /* Handle a word-aligned 16-bit imm. at 31 (PA2.0 wide).  */
-           case 'y':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 32767, -32768, 0);
-                 CHECK_ALIGN (num, 4, 0);
-                 opcode |= re_assemble_16 (num);
-                 continue;
-               }
-             else
-               {
-                 /* ??? Is this valid for wide mode?  */
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 14;
-                 continue;
-               }
+/* Parse a graphics test complete for ftest.  */
 
 
-           /* Handle a dword-aligned 16-bit imm. at 31 (PA2.0 wide).  */
-           case '&':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             if (the_insn.exp.X_op == O_constant)
-               {
-                 num = evaluate_absolute (&the_insn);
-                 CHECK_FIELD (num, 32767, -32768, 0);
-                 CHECK_ALIGN (num, 8, 0);
-                 opcode |= re_assemble_16 (num);
-                 continue;
-               }
-             else
-               {
-                 /* ??? Is this valid for wide mode?  */
-                 if (is_DP_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_GOTOFF;
-                 else if (is_PC_relative (the_insn.exp))
-                   the_insn.reloc = R_HPPA_PCREL_CALL;
-                 else
-                   the_insn.reloc = R_HPPA;
-                 the_insn.format = 14;
-                 continue;
-               }
+static int
+pa_parse_ftest_gfx_completer (char **s)
+{
+  int value;
 
 
-           /* Handle a 12 bit branch displacement.  */
-           case 'w':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             the_insn.pcrel = 1;
-             if (!strcmp (S_GET_NAME (the_insn.exp.X_add_symbol), "L$0\001"))
-               {
-                 num = evaluate_absolute (&the_insn);
-                 if (num % 4)
-                   {
-                     as_bad (_("Branch to unaligned address"));
-                     break;
-                   }
-                 CHECK_FIELD (num, 8199, -8184, 0);
+  value = 0;
+  if (strncasecmp (*s, "acc8", 4) == 0)
+    {
+      value = 5;
+      *s += 4;
+    }
+  else if (strncasecmp (*s, "acc6", 4) == 0)
+    {
+      value = 9;
+      *s += 4;
+    }
+  else if (strncasecmp (*s, "acc4", 4) == 0)
+    {
+      value = 13;
+      *s += 4;
+    }
+  else if (strncasecmp (*s, "acc2", 4) == 0)
+    {
+      value = 17;
+      *s += 4;
+    }
+  else if (strncasecmp (*s, "acc", 3) == 0)
+    {
+      value = 1;
+      *s += 3;
+    }
+  else if (strncasecmp (*s, "rej8", 4) == 0)
+    {
+      value = 6;
+      *s += 4;
+    }
+  else if (strncasecmp (*s, "rej", 3) == 0)
+    {
+      value = 2;
+      *s += 3;
+    }
+  else
+    {
+      value = 0;
+      as_bad (_("Invalid FTEST completer: %s"), *s);
+    }
 
 
-                 opcode |= re_assemble_12 ((num - 8) >> 2);
-                 continue;
-               }
-             else
-               {
-                 the_insn.reloc = R_HPPA_PCREL_CALL;
-                 the_insn.format = 12;
-                 the_insn.arg_reloc = last_call_desc.arg_reloc;
-                 memset (&last_call_desc, 0, sizeof (struct call_desc));
-                 s = expr_end;
-                 continue;
-               }
+  return value;
+}
 
 
-           /* Handle a 17 bit branch displacement.  */
-           case 'W':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             the_insn.pcrel = 1;
-             if (!the_insn.exp.X_add_symbol
-                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
-                             "L$0\001"))
-               {
-                 num = evaluate_absolute (&the_insn);
-                 if (num % 4)
-                   {
-                     as_bad (_("Branch to unaligned address"));
-                     break;
-                   }
-                 CHECK_FIELD (num, 262143, -262144, 0);
+/* Parse an FP operand format completer returning the completer
+   type.  */
 
 
-                 if (the_insn.exp.X_add_symbol)
-                   num -= 8;
+static fp_operand_format
+pa_parse_fp_cnv_format (char **s)
+{
+  int format;
+
+  format = SGL;
+  if (**s == ',')
+    {
+      *s += 1;
+      if (strncasecmp (*s, "sgl", 3) == 0)
+       {
+         format = SGL;
+         *s += 4;
+       }
+      else if (strncasecmp (*s, "dbl", 3) == 0)
+       {
+         format = DBL;
+         *s += 4;
+       }
+      else if (strncasecmp (*s, "quad", 4) == 0)
+       {
+         format = QUAD;
+         *s += 5;
+       }
+      else if (strncasecmp (*s, "w", 1) == 0)
+       {
+         format = W;
+         *s += 2;
+       }
+      else if (strncasecmp (*s, "uw", 2) == 0)
+       {
+         format = UW;
+         *s += 3;
+       }
+      else if (strncasecmp (*s, "dw", 2) == 0)
+       {
+         format = DW;
+         *s += 3;
+       }
+      else if (strncasecmp (*s, "udw", 3) == 0)
+       {
+         format = UDW;
+         *s += 4;
+       }
+      else if (strncasecmp (*s, "qw", 2) == 0)
+       {
+         format = QW;
+         *s += 3;
+       }
+      else if (strncasecmp (*s, "uqw", 3) == 0)
+       {
+         format = UQW;
+         *s += 4;
+       }
+      else
+       {
+         format = ILLEGAL_FMT;
+         as_bad (_("Invalid FP Operand Format: %3s"), *s);
+       }
+    }
 
 
-                 opcode |= re_assemble_17 (num >> 2);
-                 continue;
-               }
-             else
-               {
-                 the_insn.reloc = R_HPPA_PCREL_CALL;
-                 the_insn.format = 17;
-                 the_insn.arg_reloc = last_call_desc.arg_reloc;
-                 memset (&last_call_desc, 0, sizeof (struct call_desc));
-                 continue;
-               }
+  return format;
+}
 
 
-           /* Handle a 22 bit branch displacement.  */
-           case 'X':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             the_insn.pcrel = 1;
-             if (!the_insn.exp.X_add_symbol
-                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
-                             "L$0\001"))
-               {
-                 num = evaluate_absolute (&the_insn);
-                 if (num % 4)
-                   {
-                     as_bad (_("Branch to unaligned address"));
-                     break;
-                   }
-                 CHECK_FIELD (num, 8388607, -8388608, 0);
+/* Parse an FP operand format completer returning the completer
+   type.  */
 
 
-                 if (the_insn.exp.X_add_symbol)
-                   num -= 8;
+static fp_operand_format
+pa_parse_fp_format (char **s)
+{
+  int format;
 
 
-                 opcode |= re_assemble_22 (num >> 2);
-               }
-             else
-               {
-                 the_insn.reloc = R_HPPA_PCREL_CALL;
-                 the_insn.format = 22;
-                 the_insn.arg_reloc = last_call_desc.arg_reloc;
-                 memset (&last_call_desc, 0, sizeof (struct call_desc));
-                 continue;
-               }
+  format = SGL;
+  if (**s == ',')
+    {
+      *s += 1;
+      if (strncasecmp (*s, "sgl", 3) == 0)
+       {
+         format = SGL;
+         *s += 4;
+       }
+      else if (strncasecmp (*s, "dbl", 3) == 0)
+       {
+         format = DBL;
+         *s += 4;
+       }
+      else if (strncasecmp (*s, "quad", 4) == 0)
+       {
+         format = QUAD;
+         *s += 5;
+       }
+      else
+       {
+         format = ILLEGAL_FMT;
+         as_bad (_("Invalid FP Operand Format: %3s"), *s);
+       }
+    }
 
 
-           /* Handle an absolute 17 bit branch target.  */
-           case 'z':
-             the_insn.field_selector = pa_chk_field_selector (&s);
-             get_expression (s);
-             s = expr_end;
-             the_insn.pcrel = 0;
-             if (!the_insn.exp.X_add_symbol
-                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
-                             "L$0\001"))
-               {
-                 num = evaluate_absolute (&the_insn);
-                 if (num % 4)
-                   {
-                     as_bad (_("Branch to unaligned address"));
-                     break;
-                   }
-                 CHECK_FIELD (num, 262143, -262144, 0);
+  return format;
+}
 
 
-                 if (the_insn.exp.X_add_symbol)
-                   num -= 8;
+/* Convert from a selector string into a selector type.  */
 
 
-                 opcode |= re_assemble_17 (num >> 2);
-                 continue;
-               }
-             else
-               {
-                 the_insn.reloc = R_HPPA_ABS_CALL;
-                 the_insn.format = 17;
-                 the_insn.arg_reloc = last_call_desc.arg_reloc;
-                 memset (&last_call_desc, 0, sizeof (struct call_desc));
-                 continue;
-               }
+static int
+pa_chk_field_selector (char **str)
+{
+  int middle, low, high;
+  int cmp;
+  char name[4];
 
 
-           /* Handle '%r1' implicit operand of addil instruction.  */
-           case 'Z':
-             if (*s == ',' && *(s + 1) == '%' && *(s + 3) == '1'
-                 && (*(s + 2) == 'r' || *(s + 2) == 'R'))
-               {
-                 s += 4;
-                 continue;
-               }
-             else
-               break;
+  /* Read past any whitespace.  */
+  /* FIXME: should we read past newlines and formfeeds??? */
+  while (**str == ' ' || **str == '\t' || **str == '\n' || **str == '\f')
+    *str = *str + 1;
 
 
-           /* Handle '%sr0,%r31' implicit operand of be,l instruction.  */
-           case 'Y':
-             if (strncasecmp (s, "%sr0,%r31", 9) != 0)
-               break;
-             s += 9;
-             continue;
+  if ((*str)[1] == '\'' || (*str)[1] == '%')
+    name[0] = TOLOWER ((*str)[0]),
+    name[1] = 0;
+  else if ((*str)[2] == '\'' || (*str)[2] == '%')
+    name[0] = TOLOWER ((*str)[0]),
+    name[1] = TOLOWER ((*str)[1]),
+    name[2] = 0;
+  else if ((*str)[3] == '\'' || (*str)[3] == '%')
+    name[0] = TOLOWER ((*str)[0]),
+    name[1] = TOLOWER ((*str)[1]),
+    name[2] = TOLOWER ((*str)[2]),
+    name[3] = 0;
+  else
+    return e_fsel;
 
 
-           /* Handle immediate value of 0 for ordered load/store instructions.  */
-           case '@':
-             if (*s != '0')
-               break;
-             s++;
-             continue;
+  low = 0;
+  high = sizeof (selector_table) / sizeof (struct selector_entry) - 1;
 
 
-           /* Handle a 2 bit shift count at 25.  */
-           case '.':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 3, 1, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
+  do
+    {
+      middle = (low + high) / 2;
+      cmp = strcmp (name, selector_table[middle].prefix);
+      if (cmp < 0)
+       high = middle - 1;
+      else if (cmp > 0)
+       low = middle + 1;
+      else
+       {
+         *str += strlen (name) + 1;
+#ifndef OBJ_SOM
+         if (selector_table[middle].field_selector == e_nsel)
+           return e_fsel;
+#endif
+         return selector_table[middle].field_selector;
+       }
+    }
+  while (low <= high);
 
 
-           /* Handle a 4 bit shift count at 25.  */
-           case '*':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 15, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
+  return e_fsel;
+}
 
 
-           /* Handle a 5 bit shift count at 26.  */
-           case 'p':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 31, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, 31 - num, 5);
+/* Parse a .byte, .word, .long expression for the HPPA.  Called by
+   cons via the TC_PARSE_CONS_EXPRESSION macro.  */
 
 
-           /* Handle a 6 bit shift count at 20,22:26.  */
-           case '~':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 63, 0, strict);
-             num = 63 - num;
-             opcode |= (num & 0x20) << 6;
-             INSERT_FIELD_AND_CONTINUE (opcode, num & 0x1f, 5);
+int
+parse_cons_expression_hppa (expressionS *exp)
+{
+  int hppa_field_selector = pa_chk_field_selector (&input_line_pointer);
+  expression (exp);
+  return hppa_field_selector;
+}
 
 
-           /* Handle a 6 bit field length at 23,27:31.  */
-           case '%':
-             flag = 0;
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 64, 1, strict);
-             num--;
-             opcode |= (num & 0x20) << 3;
-             num = 31 - (num & 0x1f);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+/* Evaluate an absolute expression EXP which may be modified by
+   the selector FIELD_SELECTOR.  Return the value of the expression.  */
+static int
+evaluate_absolute (struct pa_it *insn)
+{
+  offsetT value;
+  expressionS exp;
+  int field_selector = insn->field_selector;
 
 
-           /* Handle a 6 bit field length at 19,27:31.  */
-           case '|':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 64, 1, strict);
-             num--;
-             opcode |= (num & 0x20) << 7;
-             num = 31 - (num & 0x1f);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+  exp = insn->exp;
+  value = exp.X_add_number;
 
 
-           /* Handle a 5 bit bit position at 26.  */
-           case 'P':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 31, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 5);
+  return hppa_field_adjust (0, value, field_selector);
+}
 
 
-           /* Handle a 6 bit bit position at 20,22:26.  */
-           case 'q':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 63, 0, strict);
-             opcode |= (num & 0x20) << 6;
-             INSERT_FIELD_AND_CONTINUE (opcode, num & 0x1f, 5);
+/* Mark (via expr_end) the end of an absolute expression.  FIXME.  */
 
 
-           /* Handle a 5 bit immediate at 10 with 'd' as the complement
-              of the high bit of the immediate.  */
-           case 'B':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 63, 0, strict);
-             if (num & 0x20)
-               ;
-             else
-               opcode |= (1 << 13);
-             INSERT_FIELD_AND_CONTINUE (opcode, num & 0x1f, 21);
+static int
+pa_get_absolute_expression (struct pa_it *insn, char **strp)
+{
+  char *save_in;
 
 
-           /* Handle a 5 bit immediate at 10.  */
-           case 'Q':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 31, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
+  insn->field_selector = pa_chk_field_selector (strp);
+  save_in = input_line_pointer;
+  input_line_pointer = *strp;
+  expression (&insn->exp);
+  expr_end = input_line_pointer;
+  input_line_pointer = save_in;
+  if (insn->exp.X_op != O_constant)
+    {
+      /* We have a non-match in strict mode.  */
+      if (!strict)
+       as_bad (_("Bad segment (should be absolute)."));
+      return 0;
+    }
+  return evaluate_absolute (insn);
+}
 
 
-           /* Handle a 9 bit immediate at 28.  */
-           case '$':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 511, 1, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 3);
+/* Get an absolute number.  The input string is terminated at the
+   first whitespace character.  */
 
 
-           /* Handle a 13 bit immediate at 18.  */
-           case 'A':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 8191, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 13);
+static int
+pa_get_number (struct pa_it *insn, char **strp)
+{
+  char *save_in;
+  char *s, c;
+  int result;
 
 
-           /* Handle a 26 bit immediate at 31.  */
-           case 'D':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 67108863, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+  save_in = input_line_pointer;
+  input_line_pointer = *strp;
 
 
-           /* Handle a 3 bit SFU identifier at 25.  */
-           case 'v':
-             if (*s++ != ',')
-               as_bad (_("Invalid SFU identifier"));
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 7, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
+  /* The PA assembly syntax is ambiguous in a variety of ways.  Consider
+     this string "4 %r5"  Is that the number 4 followed by the register
+     r5, or is that 4 MOD r5?  This situation occurs for example in the
+     coprocessor load and store instructions.  Previously, calling
+     pa_get_absolute_expression directly results in r5 being entered
+     in the symbol table.
 
 
-           /* Handle a 20 bit SOP field for spop0.  */
-           case 'O':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 1048575, 0, strict);
-             num = (num & 0x1f) | ((num & 0x000fffe0) << 6);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+     So, when looking for an absolute number, we cut off the input string
+     at the first whitespace character.  Thus, expressions should generally
+     contain no whitespace.  */
 
 
-           /* Handle a 15bit SOP field for spop1.  */
-           case 'o':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 32767, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 11);
+  s = *strp;
+  while (*s != ',' && *s != ' ' && *s != '\t')
+    s++;
 
 
-           /* Handle a 10bit SOP field for spop3.  */
-           case '0':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 1023, 0, strict);
-             num = (num & 0x1f) | ((num & 0x000003e0) << 6);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+  c = *s;
+  *s = 0;
 
 
-           /* Handle a 15 bit SOP field for spop2.  */
-           case '1':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 32767, 0, strict);
-             num = (num & 0x1f) | ((num & 0x00007fe0) << 6);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+  result = pa_get_absolute_expression (insn, strp);
 
 
-           /* Handle a 3-bit co-processor ID field.  */
-           case 'u':
-             if (*s++ != ',')
-               as_bad (_("Invalid COPR identifier"));
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 7, 0, strict);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
+  input_line_pointer = save_in;
+  *s = c;
+  return result;
+}
 
 
-           /* Handle a 22bit SOP field for copr.  */
-           case '2':
-             num = pa_get_absolute_expression (&the_insn, &s);
-             if (strict && the_insn.exp.X_op != O_constant)
-               break;
-             s = expr_end;
-             CHECK_FIELD (num, 4194303, 0, strict);
-             num = (num & 0x1f) | ((num & 0x003fffe0) << 4);
-             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+/* Given an argument location specification return the associated
+   argument location number.  */
 
 
-           /* Handle a source FP operand format completer.  */
-           case '{':
-             if (*s == ',' && *(s+1) == 't')
-               {
-                 the_insn.trunc = 1;
-                 s += 2;
-               }
-             else
-               the_insn.trunc = 0;
-             flag = pa_parse_fp_cnv_format (&s);
-             the_insn.fpof1 = flag;
-             if (flag == W || flag == UW)
-               flag = SGL;
-             if (flag == DW || flag == UDW)
-               flag = DBL;
-             if (flag == QW || flag == UQW)
-               flag = QUAD;
-             INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+static unsigned int
+pa_build_arg_reloc (char *type_name)
+{
 
 
-           /* Handle a destination FP operand format completer.  */
-           case '_':
-             /* pa_parse_format needs the ',' prefix.  */
-             s--;
-             flag = pa_parse_fp_cnv_format (&s);
-             the_insn.fpof2 = flag;
-             if (flag == W || flag == UW)
-               flag = SGL;
-             if (flag == DW || flag == UDW)
-               flag = DBL;
-             if (flag == QW || flag == UQW)
-               flag = QUAD;
-             opcode |= flag << 13;
-             if (the_insn.fpof1 == SGL
-                 || the_insn.fpof1 == DBL
-                 || the_insn.fpof1 == QUAD)
-               {
-                 if (the_insn.fpof2 == SGL
-                     || the_insn.fpof2 == DBL
-                     || the_insn.fpof2 == QUAD)
-                   flag = 0;
-                 else if (the_insn.fpof2 == W
-                     || the_insn.fpof2 == DW
-                     || the_insn.fpof2 == QW)
-                   flag = 2;
-                 else if (the_insn.fpof2 == UW
-                     || the_insn.fpof2 == UDW
-                     || the_insn.fpof2 == UQW)
-                   flag = 6;
-                 else
-                   abort ();
-               }
-             else if (the_insn.fpof1 == W
-                      || the_insn.fpof1 == DW
-                      || the_insn.fpof1 == QW)
-               {
-                 if (the_insn.fpof2 == SGL
-                     || the_insn.fpof2 == DBL
-                     || the_insn.fpof2 == QUAD)
-                   flag = 1;
-                 else
-                   abort ();
-               }
-             else if (the_insn.fpof1 == UW
-                      || the_insn.fpof1 == UDW
-                      || the_insn.fpof1 == UQW)
-               {
-                 if (the_insn.fpof2 == SGL
-                     || the_insn.fpof2 == DBL
-                     || the_insn.fpof2 == QUAD)
-                   flag = 5;
-                 else
-                   abort ();
-               }
-             flag |= the_insn.trunc;
-             INSERT_FIELD_AND_CONTINUE (opcode, flag, 15);
+  if (strncasecmp (type_name, "no", 2) == 0)
+    return 0;
+  if (strncasecmp (type_name, "gr", 2) == 0)
+    return 1;
+  else if (strncasecmp (type_name, "fr", 2) == 0)
+    return 2;
+  else if (strncasecmp (type_name, "fu", 2) == 0)
+    return 3;
+  else
+    as_bad (_("Invalid argument location: %s\n"), type_name);
 
 
-           /* Handle a source FP operand format completer.  */
-           case 'F':
-             flag = pa_parse_fp_format (&s);
-             the_insn.fpof1 = flag;
-             INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+  return 0;
+}
 
 
-           /* Handle a destination FP operand format completer.  */
-           case 'G':
-             /* pa_parse_format needs the ',' prefix.  */
-             s--;
-             flag = pa_parse_fp_format (&s);
-             the_insn.fpof2 = flag;
-             INSERT_FIELD_AND_CONTINUE (opcode, flag, 13);
+/* Encode and return an argument relocation specification for
+   the given register in the location specified by arg_reloc.  */
 
 
-           /* Handle a source FP operand format completer at 20.  */
-           case 'I':
-             flag = pa_parse_fp_format (&s);
-             the_insn.fpof1 = flag;
-             INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
+static unsigned int
+pa_align_arg_reloc (unsigned int reg, unsigned int arg_reloc)
+{
+  unsigned int new_reloc;
 
 
-           /* Handle a floating point operand format at 26.
-              Only allows single and double precision.  */
-           case 'H':
-             flag = pa_parse_fp_format (&s);
-             switch (flag)
-               {
-               case SGL:
-                 opcode |= 0x20;
-               case DBL:
-                 the_insn.fpof1 = flag;
-                 continue;
+  new_reloc = arg_reloc;
+  switch (reg)
+    {
+    case 0:
+      new_reloc <<= 8;
+      break;
+    case 1:
+      new_reloc <<= 6;
+      break;
+    case 2:
+      new_reloc <<= 4;
+      break;
+    case 3:
+      new_reloc <<= 2;
+      break;
+    default:
+      as_bad (_("Invalid argument description: %d"), reg);
+    }
 
 
-               case QUAD:
-               case ILLEGAL_FMT:
-               default:
-                 as_bad (_("Invalid Floating Point Operand Format."));
-               }
-             break;
+  return new_reloc;
+}
 
 
-           /* Handle all floating point registers.  */
-           case 'f':
-             switch (*++args)
-               {
-               /* Float target register.  */
-               case 't':
-                 if (!pa_parse_number (&s, 3))
-                   break;
-                 num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                 CHECK_FIELD (num, 31, 0, 0);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+/* Parse a non-negated compare/subtract completer returning the
+   number (for encoding in instructions) of the given completer.  */
 
 
-               /* Float target register with L/R selection.  */
-               case 'T':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= num;
+static int
+pa_parse_nonneg_cmpsub_cmpltr (char **s)
+{
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
+  char *save_s = *s;
+  int nullify = 0;
 
 
-                   /* 0x30 opcodes are FP arithmetic operation opcodes
-                      and need to be turned into 0x38 opcodes.  This
-                      is not necessary for loads/stores.  */
-                   if (need_pa11_opcode ()
-                       && ((opcode & 0xfc000000) == 0x30000000))
-                     opcode |= 1 << 27;
+  cmpltr = 0;
+  if (**s == ',')
+    {
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
 
 
-                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 6 : 0);
-                   continue;
-                 }
+      if (strcmp (name, "=") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, "<") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, "<=") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcmp (name, "<<") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcmp (name, "<<=") == 0)
+       {
+         cmpltr = 5;
+       }
+      else if (strcasecmp (name, "sv") == 0)
+       {
+         cmpltr = 6;
+       }
+      else if (strcasecmp (name, "od") == 0)
+       {
+         cmpltr = 7;
+       }
+      /* If we have something like addb,n then there is no condition
+        completer.  */
+      else if (strcasecmp (name, "n") == 0)
+       {
+         cmpltr = 0;
+         nullify = 1;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
+    }
 
 
-               /* Float operand 1.  */
-               case 'a':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= num << 21;
-                   if (need_pa11_opcode ())
-                     {
-                       opcode |= (pa_number & FP_REG_RSEL ? 1 << 7 : 0);
-                       opcode |= 1 << 27;
-                     }
-                   continue;
-                 }
+  /* Reset pointers if this was really a ,n for a branch instruction.  */
+  if (nullify)
+    *s = save_s;
 
 
-               /* Float operand 1 with L/R selection.  */
-               case 'X':
-               case 'A':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= num << 21;
-                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 7 : 0);
-                   continue;
-                 }
+  return cmpltr;
+}
 
 
-               /* Float operand 2.  */
-               case 'b':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= num << 16;
-                   if (need_pa11_opcode ())
-                     {
-                       opcode |= (pa_number & FP_REG_RSEL ? 1 << 12 : 0);
-                       opcode |= 1 << 27;
-                     }
-                   continue;
-                 }
+/* Parse a negated compare/subtract completer returning the
+   number (for encoding in instructions) of the given completer.  */
+
+static int
+pa_parse_neg_cmpsub_cmpltr (char **s)
+{
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
+  char *save_s = *s;
+  int nullify = 0;
+
+  cmpltr = 0;
+  if (**s == ',')
+    {
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
+
+      if (strcasecmp (name, "tr") == 0)
+       {
+         cmpltr = 0;
+       }
+      else if (strcmp (name, "<>") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, ">=") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, ">") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcmp (name, ">>=") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcmp (name, ">>") == 0)
+       {
+         cmpltr = 5;
+       }
+      else if (strcasecmp (name, "nsv") == 0)
+       {
+         cmpltr = 6;
+       }
+      else if (strcasecmp (name, "ev") == 0)
+       {
+         cmpltr = 7;
+       }
+      /* If we have something like addb,n then there is no condition
+        completer.  */
+      else if (strcasecmp (name, "n") == 0)
+       {
+         cmpltr = 0;
+         nullify = 1;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
+    }
 
 
-               /* Float operand 2 with L/R selection.  */
-               case 'B':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= num << 16;
-                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 12 : 0);
-                   continue;
-                 }
+  /* Reset pointers if this was really a ,n for a branch instruction.  */
+  if (nullify)
+    *s = save_s;
 
 
-               /* Float operand 3 for fmpyfadd, fmpynfadd.  */
-               case 'C':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= (num & 0x1c) << 11;
-                   opcode |= (num & 0x03) << 9;
-                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 8 : 0);
-                   continue;
-                 }
+  return cmpltr;
+}
 
 
-               /* Float mult operand 1 for fmpyadd, fmpysub */
-               case 'i':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   if (the_insn.fpof1 == SGL)
-                     {
-                       if (num < 16)
-                         {
-                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
-                           break;
-                         }
-                       num &= 0xF;
-                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
-                 }
+/* Parse a 64 bit compare and branch completer returning the number (for
+   encoding in instructions) of the given completer.
 
 
-               /* Float mult operand 2 for fmpyadd, fmpysub */
-               case 'j':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   if (the_insn.fpof1 == SGL)
-                     {
-                       if (num < 16)
-                         {
-                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
-                           break;
-                         }
-                       num &= 0xF;
-                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
-                 }
+   Nonnegated comparisons are returned as 0-7, negated comparisons are
+   returned as 8-15.  */
 
 
-               /* Float mult target for fmpyadd, fmpysub */
-               case 'k':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   if (the_insn.fpof1 == SGL)
-                     {
-                       if (num < 16)
-                         {
-                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
-                           break;
-                         }
-                       num &= 0xF;
-                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
-                 }
+static int
+pa_parse_cmpb_64_cmpltr (char **s)
+{
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
 
 
-               /* Float add operand 1 for fmpyadd, fmpysub */
-               case 'l':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   if (the_insn.fpof1 == SGL)
-                     {
-                       if (num < 16)
-                         {
-                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
-                           break;
-                         }
-                       num &= 0xF;
-                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
-                 }
+  cmpltr = -1;
+  if (**s == ',')
+    {
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
 
 
-               /* Float add target for fmpyadd, fmpysub */
-               case 'm':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   if (the_insn.fpof1 == SGL)
-                     {
-                       if (num < 16)
-                         {
-                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
-                           break;
-                         }
-                       num &= 0xF;
-                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
-                     }
-                   INSERT_FIELD_AND_CONTINUE (opcode, num, 11);
-                 }
+      if (strcmp (name, "*") == 0)
+       {
+         cmpltr = 0;
+       }
+      else if (strcmp (name, "*=") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, "*<") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, "*<=") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcmp (name, "*<<") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcmp (name, "*<<=") == 0)
+       {
+         cmpltr = 5;
+       }
+      else if (strcasecmp (name, "*sv") == 0)
+       {
+         cmpltr = 6;
+       }
+      else if (strcasecmp (name, "*od") == 0)
+       {
+         cmpltr = 7;
+       }
+      else if (strcasecmp (name, "*tr") == 0)
+       {
+         cmpltr = 8;
+       }
+      else if (strcmp (name, "*<>") == 0)
+       {
+         cmpltr = 9;
+       }
+      else if (strcmp (name, "*>=") == 0)
+       {
+         cmpltr = 10;
+       }
+      else if (strcmp (name, "*>") == 0)
+       {
+         cmpltr = 11;
+       }
+      else if (strcmp (name, "*>>=") == 0)
+       {
+         cmpltr = 12;
+       }
+      else if (strcmp (name, "*>>") == 0)
+       {
+         cmpltr = 13;
+       }
+      else if (strcasecmp (name, "*nsv") == 0)
+       {
+         cmpltr = 14;
+       }
+      else if (strcasecmp (name, "*ev") == 0)
+       {
+         cmpltr = 15;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
+    }
 
 
-               /* Handle L/R register halves like 'x'.  */
-               case 'E':
-               case 'e':
-                 {
-                   if (!pa_parse_number (&s, 1))
-                     break;
-                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                   CHECK_FIELD (num, 31, 0, 0);
-                   opcode |= num << 16;
-                   if (need_pa11_opcode ())
-                     {
-                       opcode |= (pa_number & FP_REG_RSEL ? 1 << 1 : 0);
-                     }
-                   continue;
-                 }
+  return cmpltr;
+}
 
 
-               /* Float target register (PA 2.0 wide).  */
-               case 'x':
-                 if (!pa_parse_number (&s, 3))
-                   break;
-                 num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
-                 CHECK_FIELD (num, 31, 0, 0);
-                 INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
+/* Parse a 64 bit compare immediate and branch completer returning the number
+   (for encoding in instructions) of the given completer.  */
 
 
-               default:
-                 abort ();
-               }
-             break;
+static int
+pa_parse_cmpib_64_cmpltr (char **s)
+{
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
 
 
-           default:
-             abort ();
-           }
-         break;
+  cmpltr = -1;
+  if (**s == ',')
+    {
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
+
+      if (strcmp (name, "*<<") == 0)
+       {
+         cmpltr = 0;
+       }
+      else if (strcmp (name, "*=") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, "*<") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, "*<=") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcmp (name, "*>>=") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcmp (name, "*<>") == 0)
+       {
+         cmpltr = 5;
        }
        }
-
- failed:
-      /* Check if the args matched.  */
-      if (match == FALSE)
+      else if (strcasecmp (name, "*>=") == 0)
        {
        {
-         if (&insn[1] - pa_opcodes < (int) NUMOPCODES
-             && !strcmp (insn->name, insn[1].name))
-           {
-             ++insn;
-             s = argstart;
-             continue;
-           }
-         else
-           {
-             as_bad (_("Invalid operands %s"), error_message);
-             return;
-           }
+         cmpltr = 6;
        }
        }
-      break;
+      else if (strcasecmp (name, "*>") == 0)
+       {
+         cmpltr = 7;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
     }
 
     }
 
-  the_insn.opcode = opcode;
+  return cmpltr;
 }
 
 }
 
-/* Turn a string in input_line_pointer into a floating point constant of type
-   type, and store the appropriate bytes in *litP.  The number of LITTLENUMS
-   emitted is stored in *sizeP .  An error message or NULL is returned.  */
-
-#define MAX_LITTLENUMS 6
+/* Parse a non-negated addition completer returning the number
+   (for encoding in instructions) of the given completer.  */
 
 
-char *
-md_atof (type, litP, sizeP)
-     char type;
-     char *litP;
-     int *sizeP;
+static int
+pa_parse_nonneg_add_cmpltr (char **s)
 {
 {
-  int prec;
-  LITTLENUM_TYPE words[MAX_LITTLENUMS];
-  LITTLENUM_TYPE *wordP;
-  char *t;
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
+  char *save_s = *s;
+  int nullify = 0;
 
 
-  switch (type)
+  cmpltr = 0;
+  if (**s == ',')
     {
     {
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
+      if (strcmp (name, "=") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, "<") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, "<=") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcasecmp (name, "nuv") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcasecmp (name, "znv") == 0)
+       {
+         cmpltr = 5;
+       }
+      else if (strcasecmp (name, "sv") == 0)
+       {
+         cmpltr = 6;
+       }
+      else if (strcasecmp (name, "od") == 0)
+       {
+         cmpltr = 7;
+       }
+      /* If we have something like addb,n then there is no condition
+        completer.  */
+      else if (strcasecmp (name, "n") == 0)
+       {
+         cmpltr = 0;
+         nullify = 1;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
+    }
 
 
-    case 'f':
-    case 'F':
-    case 's':
-    case 'S':
-      prec = 2;
-      break;
+  /* Reset pointers if this was really a ,n for a branch instruction.  */
+  if (nullify)
+    *s = save_s;
 
 
-    case 'd':
-    case 'D':
-    case 'r':
-    case 'R':
-      prec = 4;
-      break;
+  return cmpltr;
+}
 
 
-    case 'x':
-    case 'X':
-      prec = 6;
-      break;
+/* Parse a negated addition completer returning the number
+   (for encoding in instructions) of the given completer.  */
 
 
-    case 'p':
-    case 'P':
-      prec = 6;
-      break;
+static int
+pa_parse_neg_add_cmpltr (char **s)
+{
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
+  char *save_s = *s;
+  int nullify = 0;
 
 
-    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--;)
+  cmpltr = 0;
+  if (**s == ',')
     {
     {
-      md_number_to_chars (litP, (valueT) (*wordP++), sizeof (LITTLENUM_TYPE));
-      litP += sizeof (LITTLENUM_TYPE);
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
+      if (strcasecmp (name, "tr") == 0)
+       {
+         cmpltr = 0;
+       }
+      else if (strcmp (name, "<>") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, ">=") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, ">") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcasecmp (name, "uv") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcasecmp (name, "vnz") == 0)
+       {
+         cmpltr = 5;
+       }
+      else if (strcasecmp (name, "nsv") == 0)
+       {
+         cmpltr = 6;
+       }
+      else if (strcasecmp (name, "ev") == 0)
+       {
+         cmpltr = 7;
+       }
+      /* If we have something like addb,n then there is no condition
+        completer.  */
+      else if (strcasecmp (name, "n") == 0)
+       {
+         cmpltr = 0;
+         nullify = 1;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
     }
     }
-  return NULL;
-}
 
 
-/* Write out big-endian.  */
+  /* Reset pointers if this was really a ,n for a branch instruction.  */
+  if (nullify)
+    *s = save_s;
 
 
-void
-md_number_to_chars (buf, val, n)
-     char *buf;
-     valueT val;
-     int n;
-{
-  number_to_chars_bigendian (buf, val, n);
+  return cmpltr;
 }
 
 }
 
-/* Translate internal representation of relocation info to BFD target
-   format.  */
+/* Parse a 64 bit wide mode add and branch completer returning the number (for
+   encoding in instructions) of the given completer.  */
 
 
-arelent **
-tc_gen_reloc (section, fixp)
-     asection *section;
-     fixS *fixp;
+static int
+pa_parse_addb_64_cmpltr (char **s)
 {
 {
-  arelent *reloc;
-  struct hppa_fix_struct *hppa_fixp;
-  static arelent *no_relocs = NULL;
-  arelent **relocs;
-  reloc_type **codes;
-  reloc_type code;
-  int n_relocs;
-  int i;
-
-  hppa_fixp = (struct hppa_fix_struct *) fixp->tc_fix_data;
-  if (fixp->fx_addsy == 0)
-    return &no_relocs;
-
-  assert (hppa_fixp != 0);
-  assert (section != 0);
-
-  reloc = (arelent *) xmalloc (sizeof (arelent));
-
-  reloc->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-  *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
-  codes = hppa_gen_reloc_type (stdoutput,
-                              fixp->fx_r_type,
-                              hppa_fixp->fx_r_format,
-                              hppa_fixp->fx_r_field,
-                              fixp->fx_subsy != NULL,
-                              symbol_get_bfdsym (fixp->fx_addsy));
+  int cmpltr;
+  char *name = *s + 1;
+  char c;
+  char *save_s = *s;
+  int nullify = 0;
 
 
-  if (codes == NULL)
+  cmpltr = 0;
+  if (**s == ',')
     {
     {
-      as_bad (_("Cannot handle fixup at %s:%d"), fixp->fx_file, fixp->fx_line);
-      abort ();
+      *s += 1;
+      while (**s != ',' && **s != ' ' && **s != '\t')
+       *s += 1;
+      c = **s;
+      **s = 0x00;
+      if (strcmp (name, "=") == 0)
+       {
+         cmpltr = 1;
+       }
+      else if (strcmp (name, "<") == 0)
+       {
+         cmpltr = 2;
+       }
+      else if (strcmp (name, "<=") == 0)
+       {
+         cmpltr = 3;
+       }
+      else if (strcasecmp (name, "nuv") == 0)
+       {
+         cmpltr = 4;
+       }
+      else if (strcasecmp (name, "*=") == 0)
+       {
+         cmpltr = 5;
+       }
+      else if (strcasecmp (name, "*<") == 0)
+       {
+         cmpltr = 6;
+       }
+      else if (strcasecmp (name, "*<=") == 0)
+       {
+         cmpltr = 7;
+       }
+      else if (strcmp (name, "tr") == 0)
+       {
+         cmpltr = 8;
+       }
+      else if (strcmp (name, "<>") == 0)
+       {
+         cmpltr = 9;
+       }
+      else if (strcmp (name, ">=") == 0)
+       {
+         cmpltr = 10;
+       }
+      else if (strcmp (name, ">") == 0)
+       {
+         cmpltr = 11;
+       }
+      else if (strcasecmp (name, "uv") == 0)
+       {
+         cmpltr = 12;
+       }
+      else if (strcasecmp (name, "*<>") == 0)
+       {
+         cmpltr = 13;
+       }
+      else if (strcasecmp (name, "*>=") == 0)
+       {
+         cmpltr = 14;
+       }
+      else if (strcasecmp (name, "*>") == 0)
+       {
+         cmpltr = 15;
+       }
+      /* If we have something like addb,n then there is no condition
+        completer.  */
+      else if (strcasecmp (name, "n") == 0)
+       {
+         cmpltr = 0;
+         nullify = 1;
+       }
+      else
+       {
+         cmpltr = -1;
+       }
+      **s = c;
     }
 
     }
 
-  for (n_relocs = 0; codes[n_relocs]; n_relocs++)
-    ;
+  /* Reset pointers if this was really a ,n for a branch instruction.  */
+  if (nullify)
+    *s = save_s;
 
 
-  relocs = (arelent **) xmalloc (sizeof (arelent *) * n_relocs + 1);
-  reloc = (arelent *) xmalloc (sizeof (arelent) * n_relocs);
-  for (i = 0; i < n_relocs; i++)
-    relocs[i] = &reloc[i];
+  return cmpltr;
+}
 
 
-  relocs[n_relocs] = NULL;
+/* Do the real work for assembling a single instruction.  Store results
+   into the global "the_insn" variable.  */
 
 
-#ifdef OBJ_ELF
-  switch (fixp->fx_r_type)
-    {
-    default:
-      assert (n_relocs == 1);
+static void
+pa_ip (char *str)
+{
+  const char *error_message = "";
+  char *s, c, *argstart, *name, *save_s;
+  const char *args;
+  int match = FALSE;
+  int comma = 0;
+  int cmpltr, nullif, flag, cond, need_cond, num;
+  int immediate_check = 0, pos = -1, len = -1;
+  unsigned long opcode;
+  struct pa_opcode *insn;
 
 
-      code = *codes[0];
+#ifdef OBJ_SOM
+  /* We must have a valid space and subspace.  */
+  pa_check_current_space_and_subspace ();
+#endif
 
 
-      /* Now, do any processing that is dependent on the relocation type.  */
-      switch (code)
-       {
-       case R_PARISC_DLTREL21L:
-       case R_PARISC_DLTREL14R:
-       case R_PARISC_DLTREL14F:
-       case R_PARISC_PLABEL32:
-       case R_PARISC_PLABEL21L:
-       case R_PARISC_PLABEL14R:
-         /* For plabel relocations, the addend of the
-            relocation should be either 0 (no static link) or 2
-            (static link required).  This adjustment is done in
-            bfd/elf32-hppa.c:elf32_hppa_relocate_section.
+  /* Convert everything up to the first whitespace character into lower
+     case.  */
+  for (s = str; *s != ' ' && *s != '\t' && *s != '\n' && *s != '\0'; s++)
+    *s = TOLOWER (*s);
 
 
-            We also slam a zero addend into the DLT relative relocs;
-            it doesn't make a lot of sense to use any addend since
-            it gets you a different (eg unknown) DLT entry.  */
-         reloc->addend = 0;
-         break;
+  /* Skip to something interesting.  */
+  for (s = str;
+       ISUPPER (*s) || ISLOWER (*s) || (*s >= '0' && *s <= '3');
+       ++s)
+    ;
 
 
-#ifdef ELF_ARG_RELOC
-       case R_PARISC_PCREL17R:
-       case R_PARISC_PCREL17F:
-       case R_PARISC_PCREL17C:
-       case R_PARISC_DIR17R:
-       case R_PARISC_DIR17F:
-       case R_PARISC_PCREL21L:
-       case R_PARISC_DIR21L:
-         reloc->addend = HPPA_R_ADDEND (hppa_fixp->fx_arg_reloc,
-                                        fixp->fx_offset);
-         break;
-#endif
+  switch (*s)
+    {
 
 
-       case R_PARISC_DIR32:
-         /* Facilitate hand-crafted unwind info.  */
-         if (strcmp (section->name, UNWIND_SECTION_NAME) == 0)
-           code = R_PARISC_SEGREL32;
-         /* Fall thru */
+    case '\0':
+      break;
 
 
-       default:
-         reloc->addend = fixp->fx_offset;
-         break;
-       }
+    case ',':
+      comma = 1;
 
 
-      reloc->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-      *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
-      reloc->howto = bfd_reloc_type_lookup (stdoutput,
-                                           (bfd_reloc_code_real_type) code);
-      reloc->address = fixp->fx_frag->fr_address + fixp->fx_where;
+      /*FALLTHROUGH */
 
 
-      assert (reloc->howto && (unsigned int) code == reloc->howto->type);
+    case ' ':
+      *s++ = '\0';
       break;
       break;
+
+    default:
+      as_bad (_("Unknown opcode: `%s'"), str);
+      return;
     }
     }
-#else /* OBJ_SOM */
 
 
-  /* Walk over reach relocation returned by the BFD backend.  */
-  for (i = 0; i < n_relocs; i++)
+  /* Look up the opcode in the hash table.  */
+  if ((insn = (struct pa_opcode *) hash_find (op_hash, str)) == NULL)
     {
     {
-      code = *codes[i];
+      as_bad (_("Unknown opcode: `%s'"), str);
+      return;
+    }
 
 
-      relocs[i]->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-      *relocs[i]->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
-      relocs[i]->howto =
-       bfd_reloc_type_lookup (stdoutput,
-                              (bfd_reloc_code_real_type) code);
-      relocs[i]->address = fixp->fx_frag->fr_address + fixp->fx_where;
+  if (comma)
+    *--s = ',';
 
 
-      switch (code)
-       {
-       case R_COMP2:
-         /* The only time we ever use a R_COMP2 fixup is for the difference
-            of two symbols.  With that in mind we fill in all four
-            relocs now and break out of the loop.  */
-         assert (i == 1);
-         relocs[0]->sym_ptr_ptr = (asymbol **) &(bfd_abs_symbol);
-         relocs[0]->howto =
-           bfd_reloc_type_lookup (stdoutput,
-                                  (bfd_reloc_code_real_type) *codes[0]);
-         relocs[0]->address = fixp->fx_frag->fr_address + fixp->fx_where;
-         relocs[0]->addend = 0;
-         relocs[1]->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-         *relocs[1]->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
-         relocs[1]->howto =
-           bfd_reloc_type_lookup (stdoutput,
-                                  (bfd_reloc_code_real_type) *codes[1]);
-         relocs[1]->address = fixp->fx_frag->fr_address + fixp->fx_where;
-         relocs[1]->addend = 0;
-         relocs[2]->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-         *relocs[2]->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_subsy);
-         relocs[2]->howto =
-           bfd_reloc_type_lookup (stdoutput,
-                                  (bfd_reloc_code_real_type) *codes[2]);
-         relocs[2]->address = fixp->fx_frag->fr_address + fixp->fx_where;
-         relocs[2]->addend = 0;
-         relocs[3]->sym_ptr_ptr = (asymbol **) &(bfd_abs_symbol);
-         relocs[3]->howto =
-           bfd_reloc_type_lookup (stdoutput,
-                                  (bfd_reloc_code_real_type) *codes[3]);
-         relocs[3]->address = fixp->fx_frag->fr_address + fixp->fx_where;
-         relocs[3]->addend = 0;
-         relocs[4]->sym_ptr_ptr = (asymbol **) &(bfd_abs_symbol);
-         relocs[4]->howto =
-           bfd_reloc_type_lookup (stdoutput,
-                                  (bfd_reloc_code_real_type) *codes[4]);
-         relocs[4]->address = fixp->fx_frag->fr_address + fixp->fx_where;
-         relocs[4]->addend = 0;
-         goto done;
-       case R_PCREL_CALL:
-       case R_ABS_CALL:
-         relocs[i]->addend = HPPA_R_ADDEND (hppa_fixp->fx_arg_reloc, 0);
-         break;
+  /* Mark the location where arguments for the instruction start, then
+     start processing them.  */
+  argstart = s;
+  for (;;)
+    {
+      /* Do some initialization.  */
+      opcode = insn->match;
+      strict = (insn->flags & FLAG_STRICT);
+      memset (&the_insn, 0, sizeof (the_insn));
+      need_cond = 1;
 
 
-       case R_DLT_REL:
-       case R_DATA_PLABEL:
-       case R_CODE_PLABEL:
-         /* For plabel relocations, the addend of the
-            relocation should be either 0 (no static link) or 2
-            (static link required).
+      the_insn.reloc = R_HPPA_NONE;
 
 
-            FIXME: We always assume no static link!
+      if (insn->arch >= pa20
+         && bfd_get_mach (stdoutput) < insn->arch)
+       goto failed;
 
 
-            We also slam a zero addend into the DLT relative relocs;
-            it doesn't make a lot of sense to use any addend since
-            it gets you a different (eg unknown) DLT entry.  */
-         relocs[i]->addend = 0;
-         break;
+      /* Build the opcode, checking as we go to make
+        sure that the operands match.  */
+      for (args = insn->args;; ++args)
+       {
+         /* Absorb white space in instruction.  */
+         while (*s == ' ' || *s == '\t')
+           s++;
 
 
-       case R_N_MODE:
-       case R_S_MODE:
-       case R_D_MODE:
-       case R_R_MODE:
-       case R_FSEL:
-       case R_LSEL:
-       case R_RSEL:
-       case R_BEGIN_BRTAB:
-       case R_END_BRTAB:
-       case R_BEGIN_TRY:
-       case R_N0SEL:
-       case R_N1SEL:
-         /* There is no symbol or addend associated with these fixups.  */
-          relocs[i]->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-          *relocs[i]->sym_ptr_ptr = symbol_get_bfdsym (dummy_symbol);
-         relocs[i]->addend = 0;
-         break;
+         switch (*args)
+           {
+           /* End of arguments.  */
+           case '\0':
+             if (*s == '\0')
+               match = TRUE;
+             break;
 
 
-       case R_END_TRY:
-       case R_ENTRY:
-       case R_EXIT:
-         /* There is no symbol associated with these fixups.  */
-          relocs[i]->sym_ptr_ptr = (asymbol **) xmalloc (sizeof (asymbol *));
-          *relocs[i]->sym_ptr_ptr = symbol_get_bfdsym (dummy_symbol);
-         relocs[i]->addend = fixp->fx_offset;
-         break;
+           case '+':
+             if (*s == '+')
+               {
+                 ++s;
+                 continue;
+               }
+             if (*s == '-')
+               continue;
+             break;
 
 
-       default:
-         relocs[i]->addend = fixp->fx_offset;
-       }
-    }
+           /* These must match exactly.  */
+           case '(':
+           case ')':
+           case ',':
+           case ' ':
+             if (*s++ == *args)
+               continue;
+             break;
 
 
- done:
-#endif
+           /* Handle a 5 bit register or control register field at 10.  */
+           case 'b':
+           case '^':
+             if (!pa_parse_number (&s, 0))
+               break;
+             num = pa_number;
+             CHECK_FIELD (num, 31, 0, 0);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
 
 
-  return relocs;
-}
+           /* Handle %sar or %cr11.  No bits get set, we just verify that it
+              is there.  */
+           case '!':
+             /* Skip whitespace before register.  */
+             while (*s == ' ' || *s == '\t')
+               s = s + 1;
 
 
-/* Process any machine dependent frag types.  */
+             if (!strncasecmp (s, "%sar", 4))
+               {
+                 s += 4;
+                 continue;
+               }
+             else if (!strncasecmp (s, "%cr11", 5))
+               {
+                 s += 5;
+                 continue;
+               }
+             break;
 
 
-void
-md_convert_frag (abfd, sec, fragP)
-     register bfd *abfd ATTRIBUTE_UNUSED;
-     register asection *sec ATTRIBUTE_UNUSED;
-     register fragS *fragP;
-{
-  unsigned int address;
+           /* Handle a 5 bit register field at 15.  */
+           case 'x':
+             if (!pa_parse_number (&s, 0))
+               break;
+             num = pa_number;
+             CHECK_FIELD (num, 31, 0, 0);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
 
 
-  if (fragP->fr_type == rs_machine_dependent)
-    {
-      switch ((int) fragP->fr_subtype)
-       {
-       case 0:
-         fragP->fr_type = rs_fill;
-         know (fragP->fr_var == 1);
-         know (fragP->fr_next);
-         address = fragP->fr_address + fragP->fr_fix;
-         if (address % fragP->fr_offset)
-           {
-             fragP->fr_offset =
-               fragP->fr_next->fr_address
-               - fragP->fr_address
-               - fragP->fr_fix;
-           }
-         else
-           fragP->fr_offset = 0;
-         break;
-       }
-    }
-}
+           /* Handle a 5 bit register field at 31.  */
+           case 't':
+             if (!pa_parse_number (&s, 0))
+               break;
+             num = pa_number;
+             CHECK_FIELD (num, 31, 0, 0);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-/* Round up a section size to the appropriate boundary.  */
+           /* Handle a 5 bit register field at 10 and 15.  */
+           case 'a':
+             if (!pa_parse_number (&s, 0))
+               break;
+             num = pa_number;
+             CHECK_FIELD (num, 31, 0, 0);
+             opcode |= num << 16;
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
 
 
-valueT
-md_section_align (segment, size)
-     asection *segment;
-     valueT size;
-{
-  int align = bfd_get_section_alignment (stdoutput, segment);
-  int align2 = (1 << align) - 1;
+           /* Handle a 5 bit field length at 31.  */
+           case 'T':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 32, 1, 0);
+             SAVE_IMMEDIATE(num);
+             INSERT_FIELD_AND_CONTINUE (opcode, 32 - num, 0);
 
 
-  return (size + align2) & ~align2;
-}
+           /* Handle a 5 bit immediate at 15.  */
+           case '5':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             /* When in strict mode, we want to just reject this
+                match instead of giving an out of range error.  */
+             CHECK_FIELD (num, 15, -16, strict);
+             num = low_sign_unext (num, 5);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
 
 
-/* Return the approximate size of a frag before relaxation has occurred.  */
-int
-md_estimate_size_before_relax (fragP, segment)
-     register fragS *fragP;
-     asection *segment ATTRIBUTE_UNUSED;
-{
-  int size;
+           /* Handle a 5 bit immediate at 31.  */
+           case 'V':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             /* When in strict mode, we want to just reject this
+                match instead of giving an out of range error.  */
+             CHECK_FIELD (num, 15, -16, strict);
+             num = low_sign_unext (num, 5);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-  size = 0;
+           /* Handle an unsigned 5 bit immediate at 31.  */
+           case 'r':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 31, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-  while ((fragP->fr_fix + size) % fragP->fr_offset)
-    size++;
+           /* Handle an unsigned 5 bit immediate at 15.  */
+           case 'R':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 31, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
 
 
-  return size;
-}
-\f
-#ifdef OBJ_ELF
-# ifdef WARN_COMMENTS
-const char *md_shortopts = "Vc";
-# else
-const char *md_shortopts = "V";
-# endif
-#else
-# ifdef WARN_COMMENTS
-const char *md_shortopts = "c";
-# else
-const char *md_shortopts = "";
-# endif
-#endif
+           /* Handle an unsigned 10 bit immediate at 15.  */
+           case 'U':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 1023, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
 
 
-struct option md_longopts[] = {
-#ifdef WARN_COMMENTS
-  {"warn-comment", no_argument, NULL, 'c'},
-#endif
-  {NULL, no_argument, NULL, 0}
-};
-size_t md_longopts_size = sizeof (md_longopts);
+           /* Handle a 2 bit space identifier at 17.  */
+           case 's':
+             if (!pa_parse_number (&s, 0))
+               break;
+             num = pa_number;
+             CHECK_FIELD (num, 3, 0, 1);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 14);
 
 
-int
-md_parse_option (c, arg)
-     int c ATTRIBUTE_UNUSED;
-     char *arg ATTRIBUTE_UNUSED;
-{
-  switch (c)
-    {
-    default:
-      return 0;
+           /* Handle a 3 bit space identifier at 18.  */
+           case 'S':
+             if (!pa_parse_number (&s, 0))
+               break;
+             num = pa_number;
+             CHECK_FIELD (num, 7, 0, 1);
+             opcode |= re_assemble_3 (num);
+             continue;
 
 
-#ifdef OBJ_ELF
-    case 'V':
-      print_version_id ();
-      break;
-#endif
-#ifdef WARN_COMMENTS
-    case 'c':
-      warn_comment = 1;
-      break;
-#endif
-    }
+           /* Handle all completers.  */
+           case 'c':
+             switch (*++args)
+               {
 
 
-  return 1;
-}
+               /* Handle a completer for an indexing load or store.  */
+               case 'X':
+               case 'x':
+                 {
+                   int uu = 0;
+                   int m = 0;
+                   int i = 0;
+                   while (*s == ',' && i < 2)
+                     {
+                       s++;
+                       if (strncasecmp (s, "sm", 2) == 0)
+                         {
+                           uu = 1;
+                           m = 1;
+                           s++;
+                           i++;
+                         }
+                       else if (strncasecmp (s, "m", 1) == 0)
+                         m = 1;
+                       else if ((strncasecmp (s, "s ", 2) == 0)
+                                || (strncasecmp (s, "s,", 2) == 0))
+                         uu = 1;
+                       else if (strict)
+                         {
+                           /* This is a match failure.  */
+                           s--;
+                           break;
+                         }
+                       else
+                         as_bad (_("Invalid Indexed Load Completer."));
+                       s++;
+                       i++;
+                     }
+                   if (i > 2)
+                     as_bad (_("Invalid Indexed Load Completer Syntax."));
+                   opcode |= m << 5;
+                   INSERT_FIELD_AND_CONTINUE (opcode, uu, 13);
+                 }
 
 
-void
-md_show_usage (stream)
-     FILE *stream ATTRIBUTE_UNUSED;
-{
-#ifdef OBJ_ELF
-  fprintf (stream, _("\
-  -Q                      ignored\n"));
-#endif
-#ifdef WARN_COMMENTS
-  fprintf (stream, _("\
-  -c                      print a warning if a comment is found\n"));
-#endif
-}
-\f
-/* We have no need to default values of symbols.  */
+               /* Handle a short load/store completer.  */
+               case 'M':
+               case 'm':
+               case 'q':
+               case 'J':
+               case 'e':
+                 {
+                   int a = 0;
+                   int m = 0;
+                   if (*s == ',')
+                     {
+                       s++;
+                       if (strncasecmp (s, "ma", 2) == 0)
+                         {
+                           a = 0;
+                           m = 1;
+                           s += 2;
+                         }
+                       else if (strncasecmp (s, "mb", 2) == 0)
+                         {
+                           a = 1;
+                           m = 1;
+                           s += 2;
+                         }
+                       else if (strict)
+                         /* This is a match failure.  */
+                         s--;
+                       else
+                         {
+                           as_bad (_("Invalid Short Load/Store Completer."));
+                           s += 2;
+                         }
+                     }
+                   /* If we did not get a ma/mb completer, then we do not
+                      consider this a positive match for 'ce'.  */
+                   else if (*args == 'e')
+                     break;
 
 
-symbolS *
-md_undefined_symbol (name)
-     char *name ATTRIBUTE_UNUSED;
-{
-  return 0;
-}
+                  /* 'J', 'm', 'M' and 'q' are the same, except for where they
+                      encode the before/after field.  */
+                  if (*args == 'm' || *args == 'M')
+                     {
+                       opcode |= m << 5;
+                       INSERT_FIELD_AND_CONTINUE (opcode, a, 13);
+                     }
+                   else if (*args == 'q')
+                     {
+                       opcode |= m << 3;
+                       INSERT_FIELD_AND_CONTINUE (opcode, a, 2);
+                     }
+                   else if (*args == 'J')
+                     {
+                       /* M bit is explicit in the major opcode.  */
+                       INSERT_FIELD_AND_CONTINUE (opcode, a, 2);
+                     }
+                   else if (*args == 'e')
+                     {
+                       /* Stash the ma/mb flag temporarily in the
+                          instruction.  We will use (and remove it)
+                          later when handling 'J', 'K', '<' & '>'.  */
+                       opcode |= a;
+                       continue;
+                     }
+                 }
 
 
-#if defined (OBJ_SOM) || defined (ELF_ARG_RELOC)
-#define nonzero_dibits(x) \
-  ((x) | (((x) & 0x55555555) << 1) | (((x) & 0xAAAAAAAA) >> 1))
-#define arg_reloc_stub_needed(CALLER, CALLEE) \
-  (((CALLER) ^ (CALLEE)) & nonzero_dibits (CALLER) & nonzero_dibits (CALLEE))
-#else
-#define arg_reloc_stub_needed(CALLER, CALLEE) 0
-#endif
+               /* Handle a stbys completer.  */
+               case 'A':
+               case 's':
+                 {
+                   int a = 0;
+                   int m = 0;
+                   int i = 0;
+                   while (*s == ',' && i < 2)
+                     {
+                       s++;
+                       if (strncasecmp (s, "m", 1) == 0)
+                         m = 1;
+                       else if ((strncasecmp (s, "b ", 2) == 0)
+                                || (strncasecmp (s, "b,", 2) == 0))
+                         a = 0;
+                       else if (strncasecmp (s, "e", 1) == 0)
+                         a = 1;
+                       /* In strict mode, this is a match failure.  */
+                       else if (strict)
+                         {
+                           s--;
+                           break;
+                         }
+                       else
+                         as_bad (_("Invalid Store Bytes Short Completer"));
+                       s++;
+                       i++;
+                     }
+                   if (i > 2)
+                     as_bad (_("Invalid Store Bytes Short Completer"));
+                   opcode |= m << 5;
+                   INSERT_FIELD_AND_CONTINUE (opcode, a, 13);
+                 }
 
 
-/* Apply a fixup to an instruction.  */
+               /* Handle load cache hint completer.  */
+               case 'c':
+                 cmpltr = 0;
+                 if (!strncmp (s, ",sl", 3))
+                   {
+                     s += 3;
+                     cmpltr = 2;
+                   }
+                 INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 10);
 
 
-int
-md_apply_fix (fixP, valp)
-     fixS *fixP;
-     valueT *valp;
-{
-  unsigned char *buf;
-  struct hppa_fix_struct *hppa_fixP;
-  offsetT new_val;
-  int insn, val, fmt;
+               /* Handle store cache hint completer.  */
+               case 'C':
+                 cmpltr = 0;
+                 if (!strncmp (s, ",sl", 3))
+                   {
+                     s += 3;
+                     cmpltr = 2;
+                   }
+                 else if (!strncmp (s, ",bc", 3))
+                   {
+                     s += 3;
+                     cmpltr = 1;
+                   }
+                 INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 10);
 
 
-  /* SOM uses R_HPPA_ENTRY and R_HPPA_EXIT relocations which can
-     never be "applied" (they are just markers).  Likewise for
-     R_HPPA_BEGIN_BRTAB and R_HPPA_END_BRTAB.  */
-#ifdef OBJ_SOM
-  if (fixP->fx_r_type == R_HPPA_ENTRY
-      || fixP->fx_r_type == R_HPPA_EXIT
-      || fixP->fx_r_type == R_HPPA_BEGIN_BRTAB
-      || fixP->fx_r_type == R_HPPA_END_BRTAB
-      || fixP->fx_r_type == R_HPPA_BEGIN_TRY)
-    return 1;
+               /* Handle load and clear cache hint completer.  */
+               case 'd':
+                 cmpltr = 0;
+                 if (!strncmp (s, ",co", 3))
+                   {
+                     s += 3;
+                     cmpltr = 1;
+                   }
+                 INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 10);
 
 
-  /* Disgusting.  We must set fx_offset ourselves -- R_HPPA_END_TRY
-     fixups are considered not adjustable, which in turn causes
-     adjust_reloc_syms to not set fx_offset.  Ugh.  */
-  if (fixP->fx_r_type == R_HPPA_END_TRY)
-    {
-      fixP->fx_offset = *valp;
-      return 1;
-    }
-#endif
-#ifdef OBJ_ELF
-  if (fixP->fx_r_type == (int) R_PARISC_GNU_VTENTRY
-      || fixP->fx_r_type == (int) R_PARISC_GNU_VTINHERIT)
-    return 1;
-#endif
+               /* Handle load ordering completer.  */
+               case 'o':
+                 if (strncmp (s, ",o", 2) != 0)
+                   break;
+                 s += 2;
+                 continue;
 
 
-  /* There should have been an HPPA specific fixup associated
-     with the GAS fixup.  */
-  hppa_fixP = (struct hppa_fix_struct *) fixP->tc_fix_data;
-  if (hppa_fixP == NULL)
-    {
-      printf (_("no hppa_fixup entry for fixup type 0x%x at %s:%d"),
-             fixP->fx_r_type, fixP->fx_file, fixP->fx_line);
-      return 0;
-    }
+               /* Handle a branch gate completer.  */
+               case 'g':
+                 if (strncasecmp (s, ",gate", 5) != 0)
+                   break;
+                 s += 5;
+                 continue;
 
 
-  buf = fixP->fx_frag->fr_literal + fixP->fx_where;
-  insn = bfd_get_32 (stdoutput, buf);
-  fmt = bfd_hppa_insn2fmt (stdoutput, insn);
+               /* Handle a branch link and push completer.  */
+               case 'p':
+                 if (strncasecmp (s, ",l,push", 7) != 0)
+                   break;
+                 s += 7;
+                 continue;
 
 
-  /* If there is a symbol associated with this fixup, then it's something
-     which will need a SOM relocation (except for some PC-relative relocs).
-     In such cases we should treat the "val" or "addend" as zero since it
-     will be added in as needed from fx_offset in tc_gen_reloc.  */
-  if ((fixP->fx_addsy != NULL
-       || fixP->fx_r_type == (int) R_HPPA_NONE)
-#ifdef OBJ_SOM
-      && fmt != 32
-#endif
-      )
-    new_val = ((fmt == 12 || fmt == 17 || fmt == 22) ? 8 : 0);
-#ifdef OBJ_SOM
-  /* These field selectors imply that we do not want an addend.  */
-  else if (hppa_fixP->fx_r_field == e_psel
-          || hppa_fixP->fx_r_field == e_rpsel
-          || hppa_fixP->fx_r_field == e_lpsel
-          || hppa_fixP->fx_r_field == e_tsel
-          || hppa_fixP->fx_r_field == e_rtsel
-          || hppa_fixP->fx_r_field == e_ltsel)
-    new_val = ((fmt == 12 || fmt == 17 || fmt == 22) ? 8 : 0);
-  /* This is truly disgusting.  The machine independent code blindly
-     adds in the value of the symbol being relocated against.  Damn!  */
-  else if (fmt == 32
-          && fixP->fx_addsy != NULL
-          && S_GET_SEGMENT (fixP->fx_addsy) != bfd_com_section_ptr)
-    new_val = hppa_field_adjust (*valp - S_GET_VALUE (fixP->fx_addsy),
-                                0, hppa_fixP->fx_r_field);
-#endif
-  else
-    new_val = hppa_field_adjust (*valp, 0, hppa_fixP->fx_r_field);
+               /* Handle a branch link completer.  */
+               case 'l':
+                 if (strncasecmp (s, ",l", 2) != 0)
+                   break;
+                 s += 2;
+                 continue;
 
 
-  /* Handle pc-relative exceptions from above.  */
-  if ((fmt == 12 || fmt == 17 || fmt == 22)
-      && fixP->fx_addsy
-      && fixP->fx_pcrel
-      && !arg_reloc_stub_needed (symbol_arg_reloc_info (fixP->fx_addsy),
-                                hppa_fixP->fx_arg_reloc)
-#ifdef OBJ_ELF
-      && (*valp - 8 + 8192 < 16384
-         || (fmt == 17 && *valp - 8 + 262144 < 524288)
-         || (fmt == 22 && *valp - 8 + 8388608 < 16777216))
-#endif
-#ifdef OBJ_SOM
-      && (*valp - 8 + 262144 < 524288
-         || (fmt == 22 && *valp - 8 + 8388608 < 16777216))
-#endif
-      && !S_IS_EXTERNAL (fixP->fx_addsy)
-      && !S_IS_WEAK (fixP->fx_addsy)
-      && S_GET_SEGMENT (fixP->fx_addsy) == hppa_fixP->segment
-      && !(fixP->fx_subsy
-          && S_GET_SEGMENT (fixP->fx_subsy) != hppa_fixP->segment))
-    {
-      new_val = hppa_field_adjust (*valp, 0, hppa_fixP->fx_r_field);
-    }
+               /* Handle a branch pop completer.  */
+               case 'P':
+                 if (strncasecmp (s, ",pop", 4) != 0)
+                   break;
+                 s += 4;
+                 continue;
 
 
-  switch (fmt)
-    {
-    case 10:
-      CHECK_FIELD (new_val, 8191, -8192, 0);
-      val = new_val;
+               /* Handle a local processor completer.  */
+               case 'L':
+                 if (strncasecmp (s, ",l", 2) != 0)
+                   break;
+                 s += 2;
+                 continue;
 
 
-      insn = (insn & ~ 0x3ff1) | (((val & 0x1ff8) << 1)
-                                 | ((val & 0x2000) >> 13));
-      break;
-    case -11:
-      CHECK_FIELD (new_val, 8191, -8192, 0);
-      val = new_val;
+               /* Handle a PROBE read/write completer.  */
+               case 'w':
+                 flag = 0;
+                 if (!strncasecmp (s, ",w", 2))
+                   {
+                     flag = 1;
+                     s += 2;
+                   }
+                 else if (!strncasecmp (s, ",r", 2))
+                   {
+                     flag = 0;
+                     s += 2;
+                   }
 
 
-      insn = (insn & ~ 0x3ff9) | (((val & 0x1ffc) << 1)
-                                 | ((val & 0x2000) >> 13));
-      break;
-      /* Handle all opcodes with the 'j' operand type.  */
-    case 14:
-      CHECK_FIELD (new_val, 8191, -8192, 0);
-      val = new_val;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 6);
 
 
-      insn = ((insn & ~ 0x3fff) | low_sign_unext (val, 14));
-      break;
+               /* Handle MFCTL wide completer.  */
+               case 'W':
+                 if (strncasecmp (s, ",w", 2) != 0)
+                   break;
+                 s += 2;
+                 continue;
 
 
-      /* Handle all opcodes with the 'k' operand type.  */
-    case 21:
-      CHECK_FIELD (new_val, 1048575, -1048576, 0);
-      val = new_val;
+               /* Handle an RFI restore completer.  */
+               case 'r':
+                 flag = 0;
+                 if (!strncasecmp (s, ",r", 2))
+                   {
+                     flag = 5;
+                     s += 2;
+                   }
 
 
-      insn = (insn & ~ 0x1fffff) | re_assemble_21 (val);
-      break;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 5);
+
+               /* Handle a system control completer.  */
+               case 'Z':
+                 if (*s == ',' && (*(s + 1) == 'm' || *(s + 1) == 'M'))
+                   {
+                     flag = 1;
+                     s += 2;
+                   }
+                 else
+                   flag = 0;
 
 
-      /* Handle all the opcodes with the 'i' operand type.  */
-    case 11:
-      CHECK_FIELD (new_val, 1023, -1024, 0);
-      val = new_val;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 5);
 
 
-      insn = (insn & ~ 0x7ff) | low_sign_unext (val, 11);
-      break;
+               /* Handle intermediate/final completer for DCOR.  */
+               case 'i':
+                 flag = 0;
+                 if (!strncasecmp (s, ",i", 2))
+                   {
+                     flag = 1;
+                     s += 2;
+                   }
 
 
-      /* Handle all the opcodes with the 'w' operand type.  */
-    case 12:
-      CHECK_FIELD (new_val - 8, 8191, -8192, 0);
-      val = new_val - 8;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 6);
 
 
-      insn = (insn & ~ 0x1ffd) | re_assemble_12 (val >> 2);
-      break;
+               /* Handle zero/sign extension completer.  */
+               case 'z':
+                 flag = 1;
+                 if (!strncasecmp (s, ",z", 2))
+                   {
+                     flag = 0;
+                     s += 2;
+                   }
 
 
-      /* Handle some of the opcodes with the 'W' operand type.  */
-    case 17:
-      {
-       offsetT distance = *valp;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 10);
 
 
-       /* If this is an absolute branch (ie no link) with an out of
-          range target, then we want to complain.  */
-       if (fixP->fx_r_type == (int) R_HPPA_PCREL_CALL
-           && (insn & 0xffe00000) == 0xe8000000)
-         CHECK_FIELD (distance - 8, 262143, -262144, 0);
+               /* Handle add completer.  */
+               case 'a':
+                 flag = 1;
+                 if (!strncasecmp (s, ",l", 2))
+                   {
+                     flag = 2;
+                     s += 2;
+                   }
+                 else if (!strncasecmp (s, ",tsv", 4))
+                   {
+                     flag = 3;
+                     s += 4;
+                   }
 
 
-       CHECK_FIELD (new_val - 8, 262143, -262144, 0);
-       val = new_val - 8;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 10);
 
 
-       insn = (insn & ~ 0x1f1ffd) | re_assemble_17 (val >> 2);
-       break;
-      }
+               /* Handle 64 bit carry for ADD.  */
+               case 'Y':
+                 flag = 0;
+                 if (!strncasecmp (s, ",dc,tsv", 7) ||
+                     !strncasecmp (s, ",tsv,dc", 7))
+                   {
+                     flag = 1;
+                     s += 7;
+                   }
+                 else if (!strncasecmp (s, ",dc", 3))
+                   {
+                     flag = 0;
+                     s += 3;
+                   }
+                 else
+                   break;
 
 
-    case 22:
-      {
-       offsetT distance = *valp;
+                 /* Condition is not required with "dc".  */
+                 need_cond = 0;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-       /* If this is an absolute branch (ie no link) with an out of
-          range target, then we want to complain.  */
-       if (fixP->fx_r_type == (int) R_HPPA_PCREL_CALL
-           && (insn & 0xffe00000) == 0xe8000000)
-         CHECK_FIELD (distance - 8, 8388607, -8388608, 0);
+               /* Handle 32 bit carry for ADD.  */
+               case 'y':
+                 flag = 0;
+                 if (!strncasecmp (s, ",c,tsv", 6) ||
+                     !strncasecmp (s, ",tsv,c", 6))
+                   {
+                     flag = 1;
+                     s += 6;
+                   }
+                 else if (!strncasecmp (s, ",c", 2))
+                   {
+                     flag = 0;
+                     s += 2;
+                   }
+                 else
+                   break;
 
 
-       CHECK_FIELD (new_val - 8, 8388607, -8388608, 0);
-       val = new_val - 8;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-       insn = (insn & ~ 0x3ff1ffd) | re_assemble_22 (val >> 2);
-       break;
-      }
+               /* Handle trap on signed overflow.  */
+               case 'v':
+                 flag = 0;
+                 if (!strncasecmp (s, ",tsv", 4))
+                   {
+                     flag = 1;
+                     s += 4;
+                   }
 
 
-    case -10:
-      val = new_val;
-      insn = (insn & ~ 0xfff1) | re_assemble_16 (val & -8);
-      break;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-    case -16:
-      val = new_val;
-      insn = (insn & ~ 0xfff9) | re_assemble_16 (val & -4);
-      break;
+               /* Handle trap on condition and overflow.  */
+               case 't':
+                 flag = 0;
+                 if (!strncasecmp (s, ",tc,tsv", 7) ||
+                     !strncasecmp (s, ",tsv,tc", 7))
+                   {
+                     flag = 1;
+                     s += 7;
+                   }
+                 else if (!strncasecmp (s, ",tc", 3))
+                   {
+                     flag = 0;
+                     s += 3;
+                   }
+                 else
+                   break;
 
 
-    case 16:
-      val = new_val;
-      insn = (insn & ~ 0xffff) | re_assemble_16 (val);
-      break;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-    case 32:
-      insn = new_val;
-      break;
+               /* Handle 64 bit borrow for SUB.  */
+               case 'B':
+                 flag = 0;
+                 if (!strncasecmp (s, ",db,tsv", 7) ||
+                     !strncasecmp (s, ",tsv,db", 7))
+                   {
+                     flag = 1;
+                     s += 7;
+                   }
+                 else if (!strncasecmp (s, ",db", 3))
+                   {
+                     flag = 0;
+                     s += 3;
+                   }
+                 else
+                   break;
 
 
-    default:
-      as_bad (_("Unknown relocation encountered in md_apply_fix."));
-      return 0;
-    }
+                 /* Condition is not required with "db".  */
+                 need_cond = 0;
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-  /* Insert the relocation.  */
-  bfd_put_32 (stdoutput, insn, buf);
-  return 1;
-}
+               /* Handle 32 bit borrow for SUB.  */
+               case 'b':
+                 flag = 0;
+                 if (!strncasecmp (s, ",b,tsv", 6) ||
+                     !strncasecmp (s, ",tsv,b", 6))
+                   {
+                     flag = 1;
+                     s += 6;
+                   }
+                 else if (!strncasecmp (s, ",b", 2))
+                   {
+                     flag = 0;
+                     s += 2;
+                   }
+                 else
+                   break;
 
 
-/* Exactly what point is a PC-relative offset relative TO?
-   On the PA, they're relative to the address of the offset.  */
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-long
-md_pcrel_from (fixP)
-     fixS *fixP;
-{
-  return fixP->fx_where + fixP->fx_frag->fr_address;
-}
+               /* Handle trap condition completer for UADDCM.  */
+               case 'T':
+                 flag = 0;
+                 if (!strncasecmp (s, ",tc", 3))
+                   {
+                     flag = 1;
+                     s += 3;
+                   }
 
 
-/* Return nonzero if the input line pointer is at the end of
-   a statement.  */
+                 INSERT_FIELD_AND_CONTINUE (opcode, flag, 6);
 
 
-static int
-is_end_of_statement ()
-{
-  return ((*input_line_pointer == '\n')
-         || (*input_line_pointer == ';')
-         || (*input_line_pointer == '!'));
-}
+               /* Handle signed/unsigned at 21.  */
+               case 'S':
+                 {
+                   int sign = 1;
+                   if (strncasecmp (s, ",s", 2) == 0)
+                     {
+                       sign = 1;
+                       s += 2;
+                     }
+                   else if (strncasecmp (s, ",u", 2) == 0)
+                     {
+                       sign = 0;
+                       s += 2;
+                     }
 
 
-/* Read a number from S.  The number might come in one of many forms,
-   the most common will be a hex or decimal constant, but it could be
-   a pre-defined register (Yuk!), or an absolute symbol.
+                   INSERT_FIELD_AND_CONTINUE (opcode, sign, 10);
+                 }
 
 
-   Return 1 on success or 0 on failure.  If STRICT, then a missing
-   register prefix will cause a failure.  The number itself is
-   returned in `pa_number'.
+               /* Handle left/right combination at 17:18.  */
+               case 'h':
+                 if (*s++ == ',')
+                   {
+                     int lr = 0;
+                     if (*s == 'r')
+                       lr = 2;
+                     else if (*s == 'l')
+                       lr = 0;
+                     else
+                       as_bad (_("Invalid left/right combination completer"));
 
 
-   IS_FLOAT indicates that a PA-89 FP register number should be
-   parsed;  A `l' or `r' suffix is checked for if but 2 of IS_FLOAT is
-   not set.
+                     s++;
+                     INSERT_FIELD_AND_CONTINUE (opcode, lr, 13);
+                   }
+                 else
+                   as_bad (_("Invalid left/right combination completer"));
+                 break;
 
 
-   pa_parse_number can not handle negative constants and will fail
-   horribly if it is passed such a constant.  */
+               /* Handle saturation at 24:25.  */
+               case 'H':
+                 {
+                   int sat = 3;
+                   if (strncasecmp (s, ",ss", 3) == 0)
+                     {
+                       sat = 1;
+                       s += 3;
+                     }
+                   else if (strncasecmp (s, ",us", 3) == 0)
+                     {
+                       sat = 0;
+                       s += 3;
+                     }
 
 
-static int
-pa_parse_number (s, is_float)
-     char **s;
-     int is_float;
-{
-  int num;
-  char *name;
-  char c;
-  symbolS *sym;
-  int status;
-  char *p = *s;
-  boolean have_prefix;
+                   INSERT_FIELD_AND_CONTINUE (opcode, sat, 6);
+                 }
 
 
-  /* Skip whitespace before the number.  */
-  while (*p == ' ' || *p == '\t')
-    p = p + 1;
+               /* Handle permutation completer.  */
+               case '*':
+                 if (*s++ == ',')
+                   {
+                     int permloc[4];
+                     int perm = 0;
+                     int i = 0;
+                     permloc[0] = 13;
+                     permloc[1] = 10;
+                     permloc[2] = 8;
+                     permloc[3] = 6;
+                     for (; i < 4; i++)
+                       {
+                         switch (*s++)
+                           {
+                           case '0':
+                             perm = 0;
+                             break;
+                           case '1':
+                             perm = 1;
+                             break;
+                           case '2':
+                             perm = 2;
+                             break;
+                           case '3':
+                             perm = 3;
+                             break;
+                           default:
+                             as_bad (_("Invalid permutation completer"));
+                           }
+                         opcode |= perm << permloc[i];
+                       }
+                     continue;
+                   }
+                 else
+                   as_bad (_("Invalid permutation completer"));
+                 break;
 
 
-  pa_number = -1;
-  have_prefix = 0;
-  num = 0;
-  if (!strict && isdigit (*p))
-    {
-      /* Looks like a number.  */
+               default:
+                 abort ();
+               }
+             break;
 
 
-      if (*p == '0' && (*(p + 1) == 'x' || *(p + 1) == 'X'))
-       {
-         /* The number is specified in hex.  */
-         p += 2;
-         while (isdigit (*p) || ((*p >= 'a') && (*p <= 'f'))
-                || ((*p >= 'A') && (*p <= 'F')))
-           {
-             if (isdigit (*p))
-               num = num * 16 + *p - '0';
-             else if (*p >= 'a' && *p <= 'f')
-               num = num * 16 + *p - 'a' + 10;
-             else
-               num = num * 16 + *p - 'A' + 10;
-             ++p;
-           }
-       }
-      else
-       {
-         /* The number is specified in decimal.  */
-         while (isdigit (*p))
-           {
-             num = num * 10 + *p - '0';
-             ++p;
-           }
-       }
+           /* Handle all conditions.  */
+           case '?':
+             {
+               args++;
+               switch (*args)
+                 {
+                 /* Handle FP compare conditions.  */
+                 case 'f':
+                   cond = pa_parse_fp_cmp_cond (&s);
+                   INSERT_FIELD_AND_CONTINUE (opcode, cond, 0);
 
 
-      pa_number = num;
+                 /* Handle an add condition.  */
+                 case 'A':
+                 case 'a':
+                   cmpltr = 0;
+                   flag = 0;
+                   if (*s == ',')
+                     {
+                       s++;
 
 
-      /* Check for a `l' or `r' suffix.  */
-      if (is_float)
-       {
-         pa_number += FP_REG_BASE;
-         if (! (is_float & 2))
-           {
-             if (IS_R_SELECT (p))
-               {
-                 pa_number += FP_REG_RSEL;
-                 ++p;
-               }
-             else if (IS_L_SELECT (p))
-               {
-                 ++p;
-               }
-           }
-       }
-    }
-  else if (*p == '%')
-    {
-      /* The number might be a predefined register.  */
-      have_prefix = 1;
-      name = p;
-      p++;
-      c = *p;
-      /* Tege hack: Special case for general registers as the general
-         code makes a binary search with case translation, and is VERY
-         slow.  */
-      if (c == 'r')
-       {
-         p++;
-         if (*p == 'e' && *(p + 1) == 't'
-             && (*(p + 2) == '0' || *(p + 2) == '1'))
-           {
-             p += 2;
-             num = *p - '0' + 28;
-             p++;
-           }
-         else if (*p == 'p')
-           {
-             num = 2;
-             p++;
-           }
-         else if (!isdigit (*p))
-           {
-             if (print_errors)
-               as_bad (_("Undefined register: '%s'."), name);
-             num = -1;
-           }
-         else
-           {
-             do
-               num = num * 10 + *p++ - '0';
-             while (isdigit (*p));
-           }
-       }
-      else
-       {
-         /* Do a normal register search.  */
-         while (is_part_of_name (c))
-           {
-             p = p + 1;
-             c = *p;
-           }
-         *p = 0;
-         status = reg_name_search (name);
-         if (status >= 0)
-           num = status;
-         else
-           {
-             if (print_errors)
-               as_bad (_("Undefined register: '%s'."), name);
-             num = -1;
-           }
-         *p = c;
-       }
+                       /* 64 bit conditions.  */
+                       if (*args == 'A')
+                         {
+                           if (*s == '*')
+                             s++;
+                           else
+                             break;
+                         }
+                       else if (*s == '*')
+                         break;
 
 
-      pa_number = num;
-    }
-  else
-    {
-      /* And finally, it could be a symbol in the absolute section which
-         is effectively a constant, or a register alias symbol.  */
-      name = p;
-      c = *p;
-      while (is_part_of_name (c))
-       {
-         p = p + 1;
-         c = *p;
-       }
-      *p = 0;
-      if ((sym = symbol_find (name)) != NULL)
-       {
-         if (S_GET_SEGMENT (sym) == reg_section)
-           {
-             num = S_GET_VALUE (sym);
-             /* Well, we don't really have one, but we do have a
-                register, so...  */
-             have_prefix = true;
-           }
-         else if (S_GET_SEGMENT (sym) == &bfd_abs_section)
-           num = S_GET_VALUE (sym);
-         else if (!strict)
-           {
-             if (print_errors)
-               as_bad (_("Non-absolute symbol: '%s'."), name);
-             num = -1;
-           }
-       }
-      else if (!strict)
-       {
-         /* There is where we'd come for an undefined symbol
-            or for an empty string.  For an empty string we
-            will return zero.  That's a concession made for
-            compatability with the braindamaged HP assemblers.  */
-         if (*name == 0)
-           num = 0;
-         else
-           {
-             if (print_errors)
-               as_bad (_("Undefined absolute constant: '%s'."), name);
-             num = -1;
-           }
-       }
-      *p = c;
+                       name = s;
+                       while (*s != ',' && *s != ' ' && *s != '\t')
+                         s += 1;
+                       c = *s;
+                       *s = 0x00;
+                       if (strcmp (name, "=") == 0)
+                         cmpltr = 1;
+                       else if (strcmp (name, "<") == 0)
+                         cmpltr = 2;
+                       else if (strcmp (name, "<=") == 0)
+                         cmpltr = 3;
+                       else if (strcasecmp (name, "nuv") == 0)
+                         cmpltr = 4;
+                       else if (strcasecmp (name, "znv") == 0)
+                         cmpltr = 5;
+                       else if (strcasecmp (name, "sv") == 0)
+                         cmpltr = 6;
+                       else if (strcasecmp (name, "od") == 0)
+                         cmpltr = 7;
+                       else if (strcasecmp (name, "tr") == 0)
+                         {
+                           cmpltr = 0;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, "<>") == 0)
+                         {
+                           cmpltr = 1;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, ">=") == 0)
+                         {
+                           cmpltr = 2;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, ">") == 0)
+                         {
+                           cmpltr = 3;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "uv") == 0)
+                         {
+                           cmpltr = 4;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "vnz") == 0)
+                         {
+                           cmpltr = 5;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "nsv") == 0)
+                         {
+                           cmpltr = 6;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "ev") == 0)
+                         {
+                           cmpltr = 7;
+                           flag = 1;
+                         }
+                       /* ",*" is a valid condition.  */
+                       else if (*args == 'a' || *name)
+                         as_bad (_("Invalid Add Condition: %s"), name);
+                       *s = c;
+                     }
+                   /* Except with "dc", we have a match failure with
+                      'A' if we don't have a doubleword condition.  */
+                   else if (*args == 'A' && need_cond)
+                     break;
 
 
-      pa_number = num;
-    }
+                   opcode |= cmpltr << 13;
+                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
 
 
-  if (!strict || have_prefix)
-    {
-      *s = p;
-      return 1;
-    }
-  return 0;
-}
+                 /* Handle non-negated add and branch condition.  */
+                 case 'd':
+                   cmpltr = pa_parse_nonneg_add_cmpltr (&s);
+                   if (cmpltr < 0)
+                     {
+                       as_bad (_("Invalid Add and Branch Condition"));
+                       cmpltr = 0;
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
 
 
-#define REG_NAME_CNT   (sizeof (pre_defined_registers) / sizeof (struct pd_reg))
+                 /* Handle 64 bit wide-mode add and branch condition.  */
+                 case 'W':
+                   cmpltr = pa_parse_addb_64_cmpltr (&s);
+                   if (cmpltr < 0)
+                     {
+                       as_bad (_("Invalid Add and Branch Condition"));
+                       cmpltr = 0;
+                     }
+                   else
+                     {
+                       /* Negated condition requires an opcode change.  */
+                       opcode |= (cmpltr & 8) << 24;
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr & 7, 13);
 
 
-/* Given NAME, find the register number associated with that name, return
-   the integer value associated with the given name or -1 on failure.  */
+                 /* Handle a negated or non-negated add and branch
+                    condition.  */
+                 case '@':
+                   save_s = s;
+                   cmpltr = pa_parse_nonneg_add_cmpltr (&s);
+                   if (cmpltr < 0)
+                     {
+                       s = save_s;
+                       cmpltr = pa_parse_neg_add_cmpltr (&s);
+                       if (cmpltr < 0)
+                         {
+                           as_bad (_("Invalid Compare/Subtract Condition"));
+                           cmpltr = 0;
+                         }
+                       else
+                         {
+                           /* Negated condition requires an opcode change.  */
+                           opcode |= 1 << 27;
+                         }
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
 
 
-static int
-reg_name_search (name)
-     char *name;
-{
-  int middle, low, high;
-  int cmp;
+                 /* Handle branch on bit conditions.  */
+                 case 'B':
+                 case 'b':
+                   cmpltr = 0;
+                   if (*s == ',')
+                     {
+                       s++;
 
 
-  low = 0;
-  high = REG_NAME_CNT - 1;
+                       if (*args == 'B')
+                         {
+                           if (*s == '*')
+                             s++;
+                           else
+                             break;
+                         }
+                       else if (*s == '*')
+                         break;
 
 
-  do
-    {
-      middle = (low + high) / 2;
-      cmp = strcasecmp (name, pre_defined_registers[middle].name);
-      if (cmp < 0)
-       high = middle - 1;
-      else if (cmp > 0)
-       low = middle + 1;
-      else
-       return pre_defined_registers[middle].value;
-    }
-  while (low <= high);
+                       if (strncmp (s, "<", 1) == 0)
+                         {
+                           cmpltr = 0;
+                           s++;
+                         }
+                       else if (strncmp (s, ">=", 2) == 0)
+                         {
+                           cmpltr = 1;
+                           s += 2;
+                         }
+                       else
+                         as_bad (_("Invalid Branch On Bit Condition: %c"), *s);
+                     }
+                   else
+                     as_bad (_("Missing Branch On Bit Condition"));
 
 
-  return -1;
-}
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 15);
 
 
-/* Return nonzero if the given INSN and L/R information will require
-   a new PA-1.1 opcode.  */
+                 /* Handle a compare/subtract condition.  */
+                 case 'S':
+                 case 's':
+                   cmpltr = 0;
+                   flag = 0;
+                   if (*s == ',')
+                     {
+                       s++;
 
 
-static int
-need_pa11_opcode ()
-{
-  if ((pa_number & FP_REG_RSEL) != 0
-      && !(the_insn.fpof1 == DBL && the_insn.fpof2 == DBL))
-    {
-      /* If this instruction is specific to a particular architecture,
-        then set a new architecture.  */
-      if (bfd_get_mach (stdoutput) < pa11)
-       {
-         if (!bfd_set_arch_mach (stdoutput, bfd_arch_hppa, pa11))
-           as_warn (_("could not update architecture and machine"));
-       }
-      return TRUE;
-    }
-  else
-    return FALSE;
-}
+                       /* 64 bit conditions.  */
+                       if (*args == 'S')
+                         {
+                           if (*s == '*')
+                             s++;
+                           else
+                             break;
+                         }
+                       else if (*s == '*')
+                         break;
 
 
-/* Parse a condition for a fcmp instruction.  Return the numerical
-   code associated with the condition.  */
+                       name = s;
+                       while (*s != ',' && *s != ' ' && *s != '\t')
+                         s += 1;
+                       c = *s;
+                       *s = 0x00;
+                       if (strcmp (name, "=") == 0)
+                         cmpltr = 1;
+                       else if (strcmp (name, "<") == 0)
+                         cmpltr = 2;
+                       else if (strcmp (name, "<=") == 0)
+                         cmpltr = 3;
+                       else if (strcasecmp (name, "<<") == 0)
+                         cmpltr = 4;
+                       else if (strcasecmp (name, "<<=") == 0)
+                         cmpltr = 5;
+                       else if (strcasecmp (name, "sv") == 0)
+                         cmpltr = 6;
+                       else if (strcasecmp (name, "od") == 0)
+                         cmpltr = 7;
+                       else if (strcasecmp (name, "tr") == 0)
+                         {
+                           cmpltr = 0;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, "<>") == 0)
+                         {
+                           cmpltr = 1;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, ">=") == 0)
+                         {
+                           cmpltr = 2;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, ">") == 0)
+                         {
+                           cmpltr = 3;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, ">>=") == 0)
+                         {
+                           cmpltr = 4;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, ">>") == 0)
+                         {
+                           cmpltr = 5;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "nsv") == 0)
+                         {
+                           cmpltr = 6;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "ev") == 0)
+                         {
+                           cmpltr = 7;
+                           flag = 1;
+                         }
+                       /* ",*" is a valid condition.  */
+                       else if (*args != 'S' || *name)
+                         as_bad (_("Invalid Compare/Subtract Condition: %s"),
+                                 name);
+                       *s = c;
+                     }
+                   /* Except with "db", we have a match failure with
+                      'S' if we don't have a doubleword condition.  */
+                   else if (*args == 'S' && need_cond)
+                     break;
+
+                   opcode |= cmpltr << 13;
+                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
+
+                 /* Handle a non-negated compare condition.  */
+                 case 't':
+                   cmpltr = pa_parse_nonneg_cmpsub_cmpltr (&s);
+                   if (cmpltr < 0)
+                     {
+                       as_bad (_("Invalid Compare/Subtract Condition"));
+                       cmpltr = 0;
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
 
 
-static int
-pa_parse_fp_cmp_cond (s)
-     char **s;
-{
-  int cond, i;
+                 /* Handle a 32 bit compare and branch condition.  */
+                 case 'n':
+                   save_s = s;
+                   cmpltr = pa_parse_nonneg_cmpsub_cmpltr (&s);
+                   if (cmpltr < 0)
+                     {
+                       s = save_s;
+                       cmpltr = pa_parse_neg_cmpsub_cmpltr (&s);
+                       if (cmpltr < 0)
+                         {
+                           as_bad (_("Invalid Compare and Branch Condition"));
+                           cmpltr = 0;
+                         }
+                       else
+                         {
+                           /* Negated condition requires an opcode change.  */
+                           opcode |= 1 << 27;
+                         }
+                     }
 
 
-  cond = 0;
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
 
 
-  for (i = 0; i < 32; i++)
-    {
-      if (strncasecmp (*s, fp_cond_map[i].string,
-                      strlen (fp_cond_map[i].string)) == 0)
-       {
-         cond = fp_cond_map[i].cond;
-         *s += strlen (fp_cond_map[i].string);
-         /* If not a complete match, back up the input string and
-            report an error.  */
-         if (**s != ' ' && **s != '\t')
-           {
-             *s -= strlen (fp_cond_map[i].string);
-             break;
-           }
-         while (**s == ' ' || **s == '\t')
-           *s = *s + 1;
-         return cond;
-       }
-    }
+                 /* Handle a 64 bit compare and branch condition.  */
+                 case 'N':
+                   cmpltr = pa_parse_cmpb_64_cmpltr (&s);
+                   if (cmpltr >= 0)
+                     {
+                       /* Negated condition requires an opcode change.  */
+                       opcode |= (cmpltr & 8) << 26;
+                     }
+                   else
+                     /* Not a 64 bit cond.  Give 32 bit a chance.  */
+                     break;
 
 
-  as_bad (_("Invalid FP Compare Condition: %s"), *s);
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr & 7, 13);
 
 
-  /* Advance over the bogus completer.  */
-  while (**s != ',' && **s != ' ' && **s != '\t')
-    *s += 1;
+                 /* Handle a 64 bit cmpib condition.  */
+                 case 'Q':
+                   cmpltr = pa_parse_cmpib_64_cmpltr (&s);
+                   if (cmpltr < 0)
+                     /* Not a 64 bit cond.  Give 32 bit a chance.  */
+                     break;
 
 
-  return 0;
-}
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
 
 
-/* Parse a graphics test complete for ftest.  */
+                 /* Handle a logical instruction condition.  */
+                 case 'L':
+                 case 'l':
+                   cmpltr = 0;
+                   flag = 0;
+                   if (*s == ',')
+                     {
+                       s++;
 
 
-static int
-pa_parse_ftest_gfx_completer (s)
-     char **s;
-{
-  int value;
+                       /* 64 bit conditions.  */
+                       if (*args == 'L')
+                         {
+                           if (*s == '*')
+                             s++;
+                           else
+                             break;
+                         }
+                       else if (*s == '*')
+                         break;
 
 
-  value = 0;
-  if (strncasecmp (*s, "acc8", 4) == 0)
-    {
-      value = 5;
-      *s += 4;
-    }
-  else if (strncasecmp (*s, "acc6", 4) == 0)
-    {
-      value = 9;
-      *s += 4;
-    }
-  else if (strncasecmp (*s, "acc4", 4) == 0)
-    {
-      value = 13;
-      *s += 4;
-    }
-  else if (strncasecmp (*s, "acc2", 4) == 0)
-    {
-      value = 17;
-      *s += 4;
-    }
-  else if (strncasecmp (*s, "acc", 3) == 0)
-    {
-      value = 1;
-      *s += 3;
-    }
-  else if (strncasecmp (*s, "rej8", 4) == 0)
-    {
-      value = 6;
-      *s += 4;
-    }
-  else if (strncasecmp (*s, "rej", 3) == 0)
-    {
-      value = 2;
-      *s += 3;
-    }
-  else
-    {
-      value = 0;
-      as_bad (_("Invalid FTEST completer: %s"), *s);
-    }
+                       name = s;
+                       while (*s != ',' && *s != ' ' && *s != '\t')
+                         s += 1;
+                       c = *s;
+                       *s = 0x00;
 
 
-  return value;
-}
+                       if (strcmp (name, "=") == 0)
+                         cmpltr = 1;
+                       else if (strcmp (name, "<") == 0)
+                         cmpltr = 2;
+                       else if (strcmp (name, "<=") == 0)
+                         cmpltr = 3;
+                       else if (strcasecmp (name, "od") == 0)
+                         cmpltr = 7;
+                       else if (strcasecmp (name, "tr") == 0)
+                         {
+                           cmpltr = 0;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, "<>") == 0)
+                         {
+                           cmpltr = 1;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, ">=") == 0)
+                         {
+                           cmpltr = 2;
+                           flag = 1;
+                         }
+                       else if (strcmp (name, ">") == 0)
+                         {
+                           cmpltr = 3;
+                           flag = 1;
+                         }
+                       else if (strcasecmp (name, "ev") == 0)
+                         {
+                           cmpltr = 7;
+                           flag = 1;
+                         }
+                       /* ",*" is a valid condition.  */
+                       else if (*args != 'L' || *name)
+                         as_bad (_("Invalid Logical Instruction Condition."));
+                       *s = c;
+                     }
+                   /* 32-bit is default for no condition.  */
+                   else if (*args == 'L')
+                     break;
 
 
-/* Parse an FP operand format completer returning the completer
-   type.  */
+                   opcode |= cmpltr << 13;
+                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
 
 
-static fp_operand_format
-pa_parse_fp_cnv_format (s)
-     char **s;
-{
-  int format;
+                 /* Handle a shift/extract/deposit condition.  */
+                 case 'X':
+                 case 'x':
+                 case 'y':
+                   cmpltr = 0;
+                   /* Check immediate values in shift/extract/deposit
+                    * instructions if they will give undefined behaviour.  */
+                   immediate_check = 1;
+                   if (*s == ',')
+                     {
+                       save_s = s++;
 
 
-  format = SGL;
-  if (**s == ',')
-    {
-      *s += 1;
-      if (strncasecmp (*s, "sgl", 3) == 0)
-       {
-         format = SGL;
-         *s += 4;
-       }
-      else if (strncasecmp (*s, "dbl", 3) == 0)
-       {
-         format = DBL;
-         *s += 4;
-       }
-      else if (strncasecmp (*s, "quad", 4) == 0)
-       {
-         format = QUAD;
-         *s += 5;
-       }
-      else if (strncasecmp (*s, "w", 1) == 0)
-       {
-         format = W;
-         *s += 2;
-       }
-      else if (strncasecmp (*s, "uw", 2) == 0)
-       {
-         format = UW;
-         *s += 3;
-       }
-      else if (strncasecmp (*s, "dw", 2) == 0)
-       {
-         format = DW;
-         *s += 3;
-       }
-      else if (strncasecmp (*s, "udw", 3) == 0)
-       {
-         format = UDW;
-         *s += 4;
-       }
-      else if (strncasecmp (*s, "qw", 2) == 0)
-       {
-         format = QW;
-         *s += 3;
-       }
-      else if (strncasecmp (*s, "uqw", 3) == 0)
-       {
-         format = UQW;
-         *s += 4;
-       }
-      else
-       {
-         format = ILLEGAL_FMT;
-         as_bad (_("Invalid FP Operand Format: %3s"), *s);
-       }
-    }
+                       /* 64 bit conditions.  */
+                       if (*args == 'X')
+                         {
+                           if (*s == '*')
+                             s++;
+                           else
+                             break;
+                         }
+                       else if (*s == '*')
+                         break;
 
 
-  return format;
-}
+                       name = s;
+                       while (*s != ',' && *s != ' ' && *s != '\t')
+                         s += 1;
+                       c = *s;
+                       *s = 0x00;
+                       if (strcmp (name, "=") == 0)
+                         cmpltr = 1;
+                       else if (strcmp (name, "<") == 0)
+                         cmpltr = 2;
+                       else if (strcasecmp (name, "od") == 0)
+                         cmpltr = 3;
+                       else if (strcasecmp (name, "tr") == 0)
+                         cmpltr = 4;
+                       else if (strcmp (name, "<>") == 0)
+                         cmpltr = 5;
+                       else if (strcmp (name, ">=") == 0)
+                         cmpltr = 6;
+                       else if (strcasecmp (name, "ev") == 0)
+                         cmpltr = 7;
+                       /* Handle movb,n.  Put things back the way they were.
+                          This includes moving s back to where it started.  */
+                       else if (strcasecmp (name, "n") == 0 && *args == 'y')
+                         {
+                           *s = c;
+                           s = save_s;
+                           continue;
+                         }
+                       /* ",*" is a valid condition.  */
+                       else if (*args != 'X' || *name)
+                         as_bad (_("Invalid Shift/Extract/Deposit Condition."));
+                       *s = c;
+                     }
 
 
-/* Parse an FP operand format completer returning the completer
-   type.  */
+                   INSERT_FIELD_AND_CONTINUE (opcode, cmpltr, 13);
 
 
-static fp_operand_format
-pa_parse_fp_format (s)
-     char **s;
-{
-  int format;
+                 /* Handle a unit instruction condition.  */
+                 case 'U':
+                 case 'u':
+                   cmpltr = 0;
+                   flag = 0;
+                   if (*s == ',')
+                     {
+                       int uxor;
+                       s++;
 
 
-  format = SGL;
-  if (**s == ',')
-    {
-      *s += 1;
-      if (strncasecmp (*s, "sgl", 3) == 0)
-       {
-         format = SGL;
-         *s += 4;
-       }
-      else if (strncasecmp (*s, "dbl", 3) == 0)
-       {
-         format = DBL;
-         *s += 4;
-       }
-      else if (strncasecmp (*s, "quad", 4) == 0)
-       {
-         format = QUAD;
-         *s += 5;
-       }
-      else
-       {
-         format = ILLEGAL_FMT;
-         as_bad (_("Invalid FP Operand Format: %3s"), *s);
-       }
-    }
+                       /* 64 bit conditions.  */
+                       if (*args == 'U')
+                         {
+                           if (*s == '*')
+                             s++;
+                           else
+                             break;
+                         }
+                       else if (*s == '*')
+                         break;
 
 
-  return format;
-}
+                       /* The uxor instruction only supports unit conditions
+                          not involving carries.  */
+                       uxor = (opcode & 0xfc000fc0) == 0x08000380;
+                       if (strncasecmp (s, "sbz", 3) == 0)
+                         {
+                           cmpltr = 2;
+                           s += 3;
+                         }
+                       else if (strncasecmp (s, "shz", 3) == 0)
+                         {
+                           cmpltr = 3;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "sdc", 3) == 0)
+                         {
+                           cmpltr = 4;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "sbc", 3) == 0)
+                         {
+                           cmpltr = 6;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "shc", 3) == 0)
+                         {
+                           cmpltr = 7;
+                           s += 3;
+                         }
+                       else if (strncasecmp (s, "tr", 2) == 0)
+                         {
+                           cmpltr = 0;
+                           flag = 1;
+                           s += 2;
+                         }
+                       else if (strncasecmp (s, "nbz", 3) == 0)
+                         {
+                           cmpltr = 2;
+                           flag = 1;
+                           s += 3;
+                         }
+                       else if (strncasecmp (s, "nhz", 3) == 0)
+                         {
+                           cmpltr = 3;
+                           flag = 1;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "ndc", 3) == 0)
+                         {
+                           cmpltr = 4;
+                           flag = 1;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "nbc", 3) == 0)
+                         {
+                           cmpltr = 6;
+                           flag = 1;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "nhc", 3) == 0)
+                         {
+                           cmpltr = 7;
+                           flag = 1;
+                           s += 3;
+                         }
+                       else if (strncasecmp (s, "swz", 3) == 0)
+                         {
+                           cmpltr = 1;
+                           flag = 0;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "swc", 3) == 0)
+                         {
+                           cmpltr = 5;
+                           flag = 0;
+                           s += 3;
+                         }
+                       else if (strncasecmp (s, "nwz", 3) == 0)
+                         {
+                           cmpltr = 1;
+                           flag = 1;
+                           s += 3;
+                         }
+                       else if (!uxor && strncasecmp (s, "nwc", 3) == 0)
+                         {
+                           cmpltr = 5;
+                           flag = 1;
+                           s += 3;
+                         }
+                       /* ",*" is a valid condition.  */
+                       else if (*args != 'U' || (*s != ' ' && *s != '\t'))
+                         as_bad (_("Invalid Unit Instruction Condition."));
+                     }
+                   /* 32-bit is default for no condition.  */
+                   else if (*args == 'U')
+                     break;
 
 
-/* Convert from a selector string into a selector type.  */
+                   opcode |= cmpltr << 13;
+                   INSERT_FIELD_AND_CONTINUE (opcode, flag, 12);
 
 
-static int
-pa_chk_field_selector (str)
-     char **str;
-{
-  int middle, low, high;
-  int cmp;
-  char name[4];
+                 default:
+                   abort ();
+                 }
+               break;
+             }
 
 
-  /* Read past any whitespace.  */
-  /* FIXME: should we read past newlines and formfeeds??? */
-  while (**str == ' ' || **str == '\t' || **str == '\n' || **str == '\f')
-    *str = *str + 1;
+           /* Handle a nullification completer for branch instructions.  */
+           case 'n':
+             nullif = pa_parse_nullif (&s);
+             INSERT_FIELD_AND_CONTINUE (opcode, nullif, 1);
 
 
-  if ((*str)[1] == '\'' || (*str)[1] == '%')
-    name[0] = tolower ((*str)[0]),
-    name[1] = 0;
-  else if ((*str)[2] == '\'' || (*str)[2] == '%')
-    name[0] = tolower ((*str)[0]),
-    name[1] = tolower ((*str)[1]),
-    name[2] = 0;
-  else if ((*str)[3] == '\'' || (*str)[3] == '%')
-    name[0] = tolower ((*str)[0]),
-    name[1] = tolower ((*str)[1]),
-    name[2] = tolower ((*str)[2]),
-    name[3] = 0;
-  else
-    return e_fsel;
+           /* Handle a nullification completer for copr and spop insns.  */
+           case 'N':
+             nullif = pa_parse_nullif (&s);
+             INSERT_FIELD_AND_CONTINUE (opcode, nullif, 5);
 
 
-  low = 0;
-  high = sizeof (selector_table) / sizeof (struct selector_entry) - 1;
+           /* Handle ,%r2 completer for new syntax branches.  */
+           case 'L':
+             if (*s == ',' && strncasecmp (s + 1, "%r2", 3) == 0)
+               s += 4;
+             else if (*s == ',' && strncasecmp (s + 1, "%rp", 3) == 0)
+               s += 4;
+             else
+               break;
+             continue;
 
 
-  do
-    {
-      middle = (low + high) / 2;
-      cmp = strcmp (name, selector_table[middle].prefix);
-      if (cmp < 0)
-       high = middle - 1;
-      else if (cmp > 0)
-       low = middle + 1;
-      else
-       {
-         *str += strlen (name) + 1;
-#ifndef OBJ_SOM
-         if (selector_table[middle].field_selector == e_nsel)
-           return e_fsel;
-#endif
-         return selector_table[middle].field_selector;
-       }
-    }
-  while (low <= high);
+           /* Handle 3 bit entry into the fp compare array.   Valid values
+              are 0..6 inclusive.  */
+           case 'h':
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 6, 0, 0);
+                 num++;
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 13);
+               }
+             else
+               break;
 
 
-  return e_fsel;
-}
+           /* Handle 3 bit entry into the fp compare array.   Valid values
+              are 0..6 inclusive.  */
+           case 'm':
+             get_expression (s);
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 s = expr_end;
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 6, 0, 0);
+                 num = (num + 1) ^ 1;
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 13);
+               }
+             else
+               break;
 
 
-/* Mark (via expr_end) the end of an expression (I think).  FIXME.  */
+           /* Handle graphics test completers for ftest */
+           case '=':
+             {
+               num = pa_parse_ftest_gfx_completer (&s);
+               INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+             }
 
 
-static int
-get_expression (str)
-     char *str;
-{
-  char *save_in;
-  asection *seg;
+           /* Handle a 11 bit immediate at 31.  */
+           case 'i':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 1023, -1024, 0);
+                 num = low_sign_unext (num, 11);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+               }
+             else
+               {
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 11;
+                 continue;
+               }
 
 
-  save_in = input_line_pointer;
-  input_line_pointer = str;
-  seg = expression (&the_insn.exp);
-  if (!(seg == absolute_section
-       || seg == undefined_section
-       || SEG_NORMAL (seg)))
-    {
-      as_warn (_("Bad segment in expression."));
-      expr_end = input_line_pointer;
-      input_line_pointer = save_in;
-      return 1;
-    }
-  expr_end = input_line_pointer;
-  input_line_pointer = save_in;
-  return 0;
-}
+           /* Handle a 14 bit immediate at 31.  */
+           case 'J':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 int mb;
 
 
-/* Mark (via expr_end) the end of an absolute expression.  FIXME.  */
-static int
-pa_get_absolute_expression (insn, strp)
-     struct pa_it *insn;
-     char **strp;
-{
-  char *save_in;
+                 /* XXX the completer stored away tidbits of information
+                    for us to extract.  We need a cleaner way to do this.
+                    Now that we have lots of letters again, it would be
+                    good to rethink this.  */
+                 mb = opcode & 1;
+                 opcode -= mb;
+                 num = evaluate_absolute (&the_insn);
+                 if (mb != (num < 0))
+                   break;
+                 CHECK_FIELD (num, 8191, -8192, 0);
+                 num = low_sign_unext (num, 14);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+               }
+             break;
 
 
-  insn->field_selector = pa_chk_field_selector (strp);
-  save_in = input_line_pointer;
-  input_line_pointer = *strp;
-  expression (&insn->exp);
-  /* This is not perfect, but is a huge improvement over doing nothing.
+           /* Handle a 14 bit immediate at 31.  */
+           case 'K':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 int mb;
 
 
-     The PA assembly syntax is ambigious in a variety of ways.  Consider
-     this string "4 %r5"  Is that the number 4 followed by the register
-     r5, or is that 4 MOD r5?
+                 mb = opcode & 1;
+                 opcode -= mb;
+                 num = evaluate_absolute (&the_insn);
+                 if (mb == (num < 0))
+                   break;
+                 if (num % 4)
+                   break;
+                 CHECK_FIELD (num, 8191, -8192, 0);
+                 num = low_sign_unext (num, 14);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+               }
+             break;
 
 
-     If we get a modulo expresion When looking for an absolute, we try
-     again cutting off the input string at the first whitespace character.  */
-  if (insn->exp.X_op == O_modulus)
-    {
-      char *s, c;
-      int retval;
+           /* Handle a 16 bit immediate at 31.  */
+           case '<':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 int mb;
 
 
-      input_line_pointer = *strp;
-      s = *strp;
-      while (*s != ',' && *s != ' ' && *s != '\t')
-        s++;
+                 mb = opcode & 1;
+                 opcode -= mb;
+                 num = evaluate_absolute (&the_insn);
+                 if (mb != (num < 0))
+                   break;
+                 CHECK_FIELD (num, 32767, -32768, 0);
+                 num = re_assemble_16 (num);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+               }
+             break;
 
 
-      c = *s;
-      *s = 0;
+           /* Handle a 16 bit immediate at 31.  */
+           case '>':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 int mb;
 
 
-      retval = pa_get_absolute_expression (insn, strp);
+                 mb = opcode & 1;
+                 opcode -= mb;
+                 num = evaluate_absolute (&the_insn);
+                 if (mb == (num < 0))
+                   break;
+                 if (num % 4)
+                   break;
+                 CHECK_FIELD (num, 32767, -32768, 0);
+                 num = re_assemble_16 (num);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+               }
+             break;
 
 
-      input_line_pointer = save_in;
-      *s = c;
-      return evaluate_absolute (insn);
-    }
-  /* When in strict mode we have a non-match, fix up the pointers
-     and return to our caller.  */
-  if (insn->exp.X_op != O_constant && strict)
-    {
-      expr_end = input_line_pointer;
-      input_line_pointer = save_in;
-      return 0;
-    }
-  if (insn->exp.X_op != O_constant)
-    {
-      as_bad (_("Bad segment (should be absolute)."));
-      expr_end = input_line_pointer;
-      input_line_pointer = save_in;
-      return 0;
-    }
-  expr_end = input_line_pointer;
-  input_line_pointer = save_in;
-  return evaluate_absolute (insn);
-}
+           /* Handle 14 bit immediate, shifted left three times.  */
+           case '#':
+             if (bfd_get_mach (stdoutput) != pa20)
+               break;
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 if (num & 0x7)
+                   break;
+                 CHECK_FIELD (num, 8191, -8192, 0);
+                 if (num < 0)
+                   opcode |= 1;
+                 num &= 0x1fff;
+                 num >>= 3;
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 4);
+               }
+             else
+               {
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 14;
+                 continue;
+               }
+             break;
 
 
-/* Evaluate an absolute expression EXP which may be modified by
-   the selector FIELD_SELECTOR.  Return the value of the expression.  */
-static int
-evaluate_absolute (insn)
-     struct pa_it *insn;
-{
-  offsetT value;
-  expressionS exp;
-  int field_selector = insn->field_selector;
+           /* Handle 14 bit immediate, shifted left twice.  */
+           case 'd':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 if (num & 0x3)
+                   break;
+                 CHECK_FIELD (num, 8191, -8192, 0);
+                 if (num < 0)
+                   opcode |= 1;
+                 num &= 0x1fff;
+                 num >>= 2;
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 3);
+               }
+             else
+               {
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 14;
+                 continue;
+               }
 
 
-  exp = insn->exp;
-  value = exp.X_add_number;
+           /* Handle a 14 bit immediate at 31.  */
+           case 'j':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 8191, -8192, 0);
+                 num = low_sign_unext (num, 14);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+               }
+             else
+               {
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 14;
+                 continue;
+               }
 
 
-  return hppa_field_adjust (0, value, field_selector);
-}
+           /* Handle a 21 bit immediate at 31.  */
+           case 'k':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num >> 11, 1048575, -1048576, 0);
+                 opcode |= re_assemble_21 (num);
+                 continue;
+               }
+             else
+               {
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 21;
+                 continue;
+               }
 
 
-/* Given an argument location specification return the associated
-   argument location number.  */
+           /* Handle a 16 bit immediate at 31 (PA 2.0 wide mode only).  */
+           case 'l':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 32767, -32768, 0);
+                 opcode |= re_assemble_16 (num);
+                 continue;
+               }
+             else
+               {
+                 /* ??? Is this valid for wide mode?  */
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 14;
+                 continue;
+               }
 
 
-static unsigned int
-pa_build_arg_reloc (type_name)
-     char *type_name;
-{
+           /* Handle a word-aligned 16-bit imm. at 31 (PA2.0 wide).  */
+           case 'y':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 32767, -32768, 0);
+                 CHECK_ALIGN (num, 4, 0);
+                 opcode |= re_assemble_16 (num);
+                 continue;
+               }
+             else
+               {
+                 /* ??? Is this valid for wide mode?  */
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 14;
+                 continue;
+               }
 
 
-  if (strncasecmp (type_name, "no", 2) == 0)
-    return 0;
-  if (strncasecmp (type_name, "gr", 2) == 0)
-    return 1;
-  else if (strncasecmp (type_name, "fr", 2) == 0)
-    return 2;
-  else if (strncasecmp (type_name, "fu", 2) == 0)
-    return 3;
-  else
-    as_bad (_("Invalid argument location: %s\n"), type_name);
+           /* Handle a dword-aligned 16-bit imm. at 31 (PA2.0 wide).  */
+           case '&':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             if (the_insn.exp.X_op == O_constant)
+               {
+                 num = evaluate_absolute (&the_insn);
+                 CHECK_FIELD (num, 32767, -32768, 0);
+                 CHECK_ALIGN (num, 8, 0);
+                 opcode |= re_assemble_16 (num);
+                 continue;
+               }
+             else
+               {
+                 /* ??? Is this valid for wide mode?  */
+                 if (is_DP_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_GOTOFF;
+                 else if (is_PC_relative (the_insn.exp))
+                   the_insn.reloc = R_HPPA_PCREL_CALL;
+#ifdef OBJ_ELF
+                 else if (is_tls_gdidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_GD21L;
+                 else if (is_tls_ldidx (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDM21L;
+                 else if (is_tls_dtpoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LDO21L;
+                 else if (is_tls_ieoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_IE21L;
+                 else if (is_tls_leoff (the_insn.exp))
+                   the_insn.reloc = R_PARISC_TLS_LE21L;
+#endif
+                 else
+                   the_insn.reloc = R_HPPA;
+                 the_insn.format = 14;
+                 continue;
+               }
 
 
-  return 0;
-}
+           /* Handle a 12 bit branch displacement.  */
+           case 'w':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             the_insn.pcrel = 1;
+             if (!the_insn.exp.X_add_symbol
+                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
+                             FAKE_LABEL_NAME))
+               {
+                 num = evaluate_absolute (&the_insn);
+                 if (num % 4)
+                   {
+                     as_bad (_("Branch to unaligned address"));
+                     break;
+                   }
+                 if (the_insn.exp.X_add_symbol)
+                   num -= 8;
+                 CHECK_FIELD (num, 8191, -8192, 0);
+                 opcode |= re_assemble_12 (num >> 2);
+                 continue;
+               }
+             else
+               {
+                 the_insn.reloc = R_HPPA_PCREL_CALL;
+                 the_insn.format = 12;
+                 the_insn.arg_reloc = last_call_desc.arg_reloc;
+                 memset (&last_call_desc, 0, sizeof (struct call_desc));
+                 s = expr_end;
+                 continue;
+               }
 
 
-/* Encode and return an argument relocation specification for
-   the given register in the location specified by arg_reloc.  */
+           /* Handle a 17 bit branch displacement.  */
+           case 'W':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             the_insn.pcrel = 1;
+             if (!the_insn.exp.X_add_symbol
+                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
+                             FAKE_LABEL_NAME))
+               {
+                 num = evaluate_absolute (&the_insn);
+                 if (num % 4)
+                   {
+                     as_bad (_("Branch to unaligned address"));
+                     break;
+                   }
+                 if (the_insn.exp.X_add_symbol)
+                   num -= 8;
+                 CHECK_FIELD (num, 262143, -262144, 0);
+                 opcode |= re_assemble_17 (num >> 2);
+                 continue;
+               }
+             else
+               {
+                 the_insn.reloc = R_HPPA_PCREL_CALL;
+                 the_insn.format = 17;
+                 the_insn.arg_reloc = last_call_desc.arg_reloc;
+                 memset (&last_call_desc, 0, sizeof (struct call_desc));
+                 continue;
+               }
 
 
-static unsigned int
-pa_align_arg_reloc (reg, arg_reloc)
-     unsigned int reg;
-     unsigned int arg_reloc;
-{
-  unsigned int new_reloc;
+           /* Handle a 22 bit branch displacement.  */
+           case 'X':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             the_insn.pcrel = 1;
+             if (!the_insn.exp.X_add_symbol
+                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
+                             FAKE_LABEL_NAME))
+               {
+                 num = evaluate_absolute (&the_insn);
+                 if (num % 4)
+                   {
+                     as_bad (_("Branch to unaligned address"));
+                     break;
+                   }
+                 if (the_insn.exp.X_add_symbol)
+                   num -= 8;
+                 CHECK_FIELD (num, 8388607, -8388608, 0);
+                 opcode |= re_assemble_22 (num >> 2);
+               }
+             else
+               {
+                 the_insn.reloc = R_HPPA_PCREL_CALL;
+                 the_insn.format = 22;
+                 the_insn.arg_reloc = last_call_desc.arg_reloc;
+                 memset (&last_call_desc, 0, sizeof (struct call_desc));
+                 continue;
+               }
 
 
-  new_reloc = arg_reloc;
-  switch (reg)
-    {
-    case 0:
-      new_reloc <<= 8;
-      break;
-    case 1:
-      new_reloc <<= 6;
-      break;
-    case 2:
-      new_reloc <<= 4;
-      break;
-    case 3:
-      new_reloc <<= 2;
-      break;
-    default:
-      as_bad (_("Invalid argument description: %d"), reg);
-    }
+           /* Handle an absolute 17 bit branch target.  */
+           case 'z':
+             the_insn.field_selector = pa_chk_field_selector (&s);
+             get_expression (s);
+             s = expr_end;
+             the_insn.pcrel = 0;
+             if (!the_insn.exp.X_add_symbol
+                 || !strcmp (S_GET_NAME (the_insn.exp.X_add_symbol),
+                             FAKE_LABEL_NAME))
+               {
+                 num = evaluate_absolute (&the_insn);
+                 if (num % 4)
+                   {
+                     as_bad (_("Branch to unaligned address"));
+                     break;
+                   }
+                 if (the_insn.exp.X_add_symbol)
+                   num -= 8;
+                 CHECK_FIELD (num, 262143, -262144, 0);
+                 opcode |= re_assemble_17 (num >> 2);
+                 continue;
+               }
+             else
+               {
+                 the_insn.reloc = R_HPPA_ABS_CALL;
+                 the_insn.format = 17;
+                 the_insn.arg_reloc = last_call_desc.arg_reloc;
+                 memset (&last_call_desc, 0, sizeof (struct call_desc));
+                 continue;
+               }
 
 
-  return new_reloc;
-}
+           /* Handle '%r1' implicit operand of addil instruction.  */
+           case 'Z':
+             if (*s == ',' && *(s + 1) == '%' && *(s + 3) == '1'
+                 && (*(s + 2) == 'r' || *(s + 2) == 'R'))
+               {
+                 s += 4;
+                 continue;
+               }
+             else
+               break;
 
 
-/* Parse a PA nullification completer (,n).  Return nonzero if the
-   completer was found; return zero if no completer was found.  */
+           /* Handle '%sr0,%r31' implicit operand of be,l instruction.  */
+           case 'Y':
+             if (strncasecmp (s, "%sr0,%r31", 9) != 0)
+               break;
+             s += 9;
+             continue;
 
 
-static int
-pa_parse_nullif (s)
-     char **s;
-{
-  int nullif;
+           /* Handle immediate value of 0 for ordered load/store instructions.  */
+           case '@':
+             if (*s != '0')
+               break;
+             s++;
+             continue;
 
 
-  nullif = 0;
-  if (**s == ',')
-    {
-      *s = *s + 1;
-      if (strncasecmp (*s, "n", 1) == 0)
-       nullif = 1;
-      else
-       {
-         as_bad (_("Invalid Nullification: (%c)"), **s);
-         nullif = 0;
-       }
-      *s = *s + 1;
-    }
+           /* Handle a 2 bit shift count at 25.  */
+           case '.':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 3, 1, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
 
 
-  return nullif;
-}
+           /* Handle a 4 bit shift count at 25.  */
+           case '*':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 15, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
 
 
-/* Parse a non-negated compare/subtract completer returning the
-   number (for encoding in instrutions) of the given completer.
+           /* Handle a 5 bit shift count at 26.  */
+           case 'p':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 31, 0, strict);
+             SAVE_IMMEDIATE(num);
+             INSERT_FIELD_AND_CONTINUE (opcode, 31 - num, 5);
 
 
-   ISBRANCH specifies whether or not this is parsing a condition
-   completer for a branch (vs a nullification completer for a
-   computational instruction.  */
+           /* Handle a 6 bit shift count at 20,22:26.  */
+           case '~':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 63, 0, strict);
+             SAVE_IMMEDIATE(num);
+             num = 63 - num;
+             opcode |= (num & 0x20) << 6;
+             INSERT_FIELD_AND_CONTINUE (opcode, num & 0x1f, 5);
 
 
-static int
-pa_parse_nonneg_cmpsub_cmpltr (s, isbranch)
-     char **s;
-     int isbranch;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
-  char *save_s = *s;
-  int nullify = 0;
+           /* Handle a 6 bit field length at 23,27:31.  */
+           case '%':
+             flag = 0;
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 64, 1, strict);
+             SAVE_IMMEDIATE(num);
+             num--;
+             opcode |= (num & 0x20) << 3;
+             num = 31 - (num & 0x1f);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-  cmpltr = 0;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
+           /* Handle a 6 bit field length at 19,27:31.  */
+           case '|':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 64, 1, strict);
+             SAVE_IMMEDIATE(num);
+             num--;
+             opcode |= (num & 0x20) << 7;
+             num = 31 - (num & 0x1f);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-      if (strcmp (name, "=") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, "<") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, "<=") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcmp (name, "<<") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcmp (name, "<<=") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "sv") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "od") == 0)
-       {
-         cmpltr = 7;
-       }
-      /* If we have something like addb,n then there is no condition
-         completer.  */
-      else if (strcasecmp (name, "n") == 0 && isbranch)
-       {
-         cmpltr = 0;
-         nullify = 1;
-       }
-      else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
-    }
+           /* Handle a 5 bit bit position at 26.  */
+           case 'P':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 31, 0, strict);
+             SAVE_IMMEDIATE(num);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 5);
 
 
-  /* Reset pointers if this was really a ,n for a branch instruction.  */
-  if (nullify)
-    *s = save_s;
+           /* Handle a 6 bit bit position at 20,22:26.  */
+           case 'q':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 63, 0, strict);
+             SAVE_IMMEDIATE(num);
+             opcode |= (num & 0x20) << 6;
+             INSERT_FIELD_AND_CONTINUE (opcode, num & 0x1f, 5);
 
 
-  return cmpltr;
-}
+           /* Handle a 5 bit immediate at 10 with 'd' as the complement
+              of the high bit of the immediate.  */
+           case 'B':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 63, 0, strict);
+             if (num & 0x20)
+               ;
+             else
+               opcode |= (1 << 13);
+             INSERT_FIELD_AND_CONTINUE (opcode, num & 0x1f, 21);
 
 
-/* Parse a negated compare/subtract completer returning the
-   number (for encoding in instrutions) of the given completer.
+           /* Handle a 5 bit immediate at 10.  */
+           case 'Q':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 31, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
 
 
-   ISBRANCH specifies whether or not this is parsing a condition
-   completer for a branch (vs a nullification completer for a
-   computational instruction.  */
+           /* Handle a 9 bit immediate at 28.  */
+           case '$':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 511, 1, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 3);
 
 
-static int
-pa_parse_neg_cmpsub_cmpltr (s, isbranch)
-     char **s;
-     int isbranch;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
-  char *save_s = *s;
-  int nullify = 0;
+           /* Handle a 13 bit immediate at 18.  */
+           case 'A':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 8191, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 13);
 
 
-  cmpltr = 0;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
+           /* Handle a 26 bit immediate at 31.  */
+           case 'D':
+             num = pa_get_absolute_expression (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 67108863, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+
+           /* Handle a 3 bit SFU identifier at 25.  */
+           case 'v':
+             if (*s++ != ',')
+               as_bad (_("Invalid SFU identifier"));
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 7, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
+
+           /* Handle a 20 bit SOP field for spop0.  */
+           case 'O':
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 1048575, 0, strict);
+             num = (num & 0x1f) | ((num & 0x000fffe0) << 6);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-      if (strcasecmp (name, "tr") == 0)
-       {
-         cmpltr = 0;
-       }
-      else if (strcmp (name, "<>") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, ">=") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, ">") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcmp (name, ">>=") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcmp (name, ">>") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "nsv") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "ev") == 0)
-       {
-         cmpltr = 7;
-       }
-      /* If we have something like addb,n then there is no condition
-         completer.  */
-      else if (strcasecmp (name, "n") == 0 && isbranch)
-       {
-         cmpltr = 0;
-         nullify = 1;
-       }
-      else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
-    }
+           /* Handle a 15bit SOP field for spop1.  */
+           case 'o':
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 32767, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 11);
 
 
-  /* Reset pointers if this was really a ,n for a branch instruction.  */
-  if (nullify)
-    *s = save_s;
+           /* Handle a 10bit SOP field for spop3.  */
+           case '0':
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 1023, 0, strict);
+             num = (num & 0x1f) | ((num & 0x000003e0) << 6);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-  return cmpltr;
-}
+           /* Handle a 15 bit SOP field for spop2.  */
+           case '1':
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 32767, 0, strict);
+             num = (num & 0x1f) | ((num & 0x00007fe0) << 6);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-/* Parse a 64 bit compare and branch completer returning the number (for
-   encoding in instrutions) of the given completer.
+           /* Handle a 3-bit co-processor ID field.  */
+           case 'u':
+             if (*s++ != ',')
+               as_bad (_("Invalid COPR identifier"));
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 7, 0, strict);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
 
 
-   Nonnegated comparisons are returned as 0-7, negated comparisons are
-   returned as 8-15.  */
+           /* Handle a 22bit SOP field for copr.  */
+           case '2':
+             num = pa_get_number (&the_insn, &s);
+             if (strict && the_insn.exp.X_op != O_constant)
+               break;
+             s = expr_end;
+             CHECK_FIELD (num, 4194303, 0, strict);
+             num = (num & 0x1f) | ((num & 0x003fffe0) << 4);
+             INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-static int
-pa_parse_cmpb_64_cmpltr (s)
-     char **s;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
+           /* Handle a source FP operand format completer.  */
+           case '{':
+             if (*s == ',' && *(s+1) == 't')
+               {
+                 the_insn.trunc = 1;
+                 s += 2;
+               }
+             else
+               the_insn.trunc = 0;
+             flag = pa_parse_fp_cnv_format (&s);
+             the_insn.fpof1 = flag;
+             if (flag == W || flag == UW)
+               flag = SGL;
+             if (flag == DW || flag == UDW)
+               flag = DBL;
+             if (flag == QW || flag == UQW)
+               flag = QUAD;
+             INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-  cmpltr = -1;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
+           /* Handle a destination FP operand format completer.  */
+           case '_':
+             /* pa_parse_format needs the ',' prefix.  */
+             s--;
+             flag = pa_parse_fp_cnv_format (&s);
+             the_insn.fpof2 = flag;
+             if (flag == W || flag == UW)
+               flag = SGL;
+             if (flag == DW || flag == UDW)
+               flag = DBL;
+             if (flag == QW || flag == UQW)
+               flag = QUAD;
+             opcode |= flag << 13;
+             if (the_insn.fpof1 == SGL
+                 || the_insn.fpof1 == DBL
+                 || the_insn.fpof1 == QUAD)
+               {
+                 if (the_insn.fpof2 == SGL
+                     || the_insn.fpof2 == DBL
+                     || the_insn.fpof2 == QUAD)
+                   flag = 0;
+                 else if (the_insn.fpof2 == W
+                     || the_insn.fpof2 == DW
+                     || the_insn.fpof2 == QW)
+                   flag = 2;
+                 else if (the_insn.fpof2 == UW
+                     || the_insn.fpof2 == UDW
+                     || the_insn.fpof2 == UQW)
+                   flag = 6;
+                 else
+                   abort ();
+               }
+             else if (the_insn.fpof1 == W
+                      || the_insn.fpof1 == DW
+                      || the_insn.fpof1 == QW)
+               {
+                 if (the_insn.fpof2 == SGL
+                     || the_insn.fpof2 == DBL
+                     || the_insn.fpof2 == QUAD)
+                   flag = 1;
+                 else
+                   abort ();
+               }
+             else if (the_insn.fpof1 == UW
+                      || the_insn.fpof1 == UDW
+                      || the_insn.fpof1 == UQW)
+               {
+                 if (the_insn.fpof2 == SGL
+                     || the_insn.fpof2 == DBL
+                     || the_insn.fpof2 == QUAD)
+                   flag = 5;
+                 else
+                   abort ();
+               }
+             flag |= the_insn.trunc;
+             INSERT_FIELD_AND_CONTINUE (opcode, flag, 15);
 
 
-      if (strcmp (name, "*") == 0)
-       {
-         cmpltr = 0;
-       }
-      else if (strcmp (name, "*=") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, "*<") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, "*<=") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcmp (name, "*<<") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcmp (name, "*<<=") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "*sv") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "*od") == 0)
-       {
-         cmpltr = 7;
-       }
-      else if (strcasecmp (name, "*tr") == 0)
-       {
-         cmpltr = 8;
-       }
-      else if (strcmp (name, "*<>") == 0)
-       {
-         cmpltr = 9;
-       }
-      else if (strcmp (name, "*>=") == 0)
-       {
-         cmpltr = 10;
-       }
-      else if (strcmp (name, "*>") == 0)
-       {
-         cmpltr = 11;
-       }
-      else if (strcmp (name, "*>>=") == 0)
-       {
-         cmpltr = 12;
-       }
-      else if (strcmp (name, "*>>") == 0)
-       {
-         cmpltr = 13;
-       }
-      else if (strcasecmp (name, "*nsv") == 0)
-       {
-         cmpltr = 14;
-       }
-      else if (strcasecmp (name, "*ev") == 0)
-       {
-         cmpltr = 15;
-       }
-      else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
-    }
+           /* Handle a source FP operand format completer.  */
+           case 'F':
+             flag = pa_parse_fp_format (&s);
+             the_insn.fpof1 = flag;
+             INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-  return cmpltr;
-}
+           /* Handle a destination FP operand format completer.  */
+           case 'G':
+             /* pa_parse_format needs the ',' prefix.  */
+             s--;
+             flag = pa_parse_fp_format (&s);
+             the_insn.fpof2 = flag;
+             INSERT_FIELD_AND_CONTINUE (opcode, flag, 13);
 
 
-/* Parse a 64 bit compare immediate and branch completer returning the number
-   (for encoding in instrutions) of the given completer.  */
+           /* Handle a source FP operand format completer at 20.  */
+           case 'I':
+             flag = pa_parse_fp_format (&s);
+             the_insn.fpof1 = flag;
+             INSERT_FIELD_AND_CONTINUE (opcode, flag, 11);
 
 
-static int
-pa_parse_cmpib_64_cmpltr (s)
-     char **s;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
+           /* Handle a floating point operand format at 26.
+              Only allows single and double precision.  */
+           case 'H':
+             flag = pa_parse_fp_format (&s);
+             switch (flag)
+               {
+               case SGL:
+                 opcode |= 0x20;
+               case DBL:
+                 the_insn.fpof1 = flag;
+                 continue;
 
 
-  cmpltr = -1;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
+               case QUAD:
+               case ILLEGAL_FMT:
+               default:
+                 as_bad (_("Invalid Floating Point Operand Format."));
+               }
+             break;
 
 
-      if (strcmp (name, "*<<") == 0)
-       {
-         cmpltr = 0;
-       }
-      else if (strcmp (name, "*=") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, "*<") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, "*<=") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcmp (name, "*>>=") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcmp (name, "*<>") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "*>=") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "*>") == 0)
-       {
-         cmpltr = 7;
-       }
-      else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
-    }
+           /* Handle all floating point registers.  */
+           case 'f':
+             switch (*++args)
+               {
+               /* Float target register.  */
+               case 't':
+                 if (!pa_parse_number (&s, 3))
+                   break;
+                 /* RSEL should not be set.  */
+                 if (pa_number & FP_REG_RSEL)
+                   break;
+                 num = pa_number - FP_REG_BASE;
+                 CHECK_FIELD (num, 31, 0, 0);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
 
 
-  return cmpltr;
-}
+               /* Float target register with L/R selection.  */
+               case 'T':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= num;
 
 
-/* Parse a non-negated addition completer returning the number
-   (for encoding in instrutions) of the given completer.
+                   /* 0x30 opcodes are FP arithmetic operation opcodes
+                      and need to be turned into 0x38 opcodes.  This
+                      is not necessary for loads/stores.  */
+                   if (need_pa11_opcode ()
+                       && ((opcode & 0xfc000000) == 0x30000000))
+                     opcode |= 1 << 27;
 
 
-   ISBRANCH specifies whether or not this is parsing a condition
-   completer for a branch (vs a nullification completer for a
-   computational instruction.  */
+                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 6 : 0);
+                   continue;
+                 }
 
 
-static int
-pa_parse_nonneg_add_cmpltr (s, isbranch)
-     char **s;
-     int isbranch;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
-  char *save_s = *s;
+               /* Float operand 1.  */
+               case 'a':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= num << 21;
+                   if (need_pa11_opcode ())
+                     {
+                       opcode |= (pa_number & FP_REG_RSEL ? 1 << 7 : 0);
+                       opcode |= 1 << 27;
+                     }
+                   continue;
+                 }
 
 
-  cmpltr = 0;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
-      if (strcmp (name, "=") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, "<") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, "<=") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcasecmp (name, "nuv") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcasecmp (name, "znv") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "sv") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "od") == 0)
-       {
-         cmpltr = 7;
-       }
-      /* If we have something like addb,n then there is no condition
-         completer.  */
-      else if (strcasecmp (name, "n") == 0 && isbranch)
-       {
-         cmpltr = 0;
-       }
-      else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
-    }
+               /* Float operand 1 with L/R selection.  */
+               case 'X':
+               case 'A':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= num << 21;
+                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 7 : 0);
+                   continue;
+                 }
 
 
-  /* Reset pointers if this was really a ,n for a branch instruction.  */
-  if (cmpltr == 0 && *name == 'n' && isbranch)
-    *s = save_s;
+               /* Float operand 2.  */
+               case 'b':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= num << 16;
+                   if (need_pa11_opcode ())
+                     {
+                       opcode |= (pa_number & FP_REG_RSEL ? 1 << 12 : 0);
+                       opcode |= 1 << 27;
+                     }
+                   continue;
+                 }
 
 
-  return cmpltr;
-}
+               /* Float operand 2 with L/R selection.  */
+               case 'B':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= num << 16;
+                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 12 : 0);
+                   continue;
+                 }
 
 
-/* Parse a negated addition completer returning the number
-   (for encoding in instrutions) of the given completer.
+               /* Float operand 3 for fmpyfadd, fmpynfadd.  */
+               case 'C':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= (num & 0x1c) << 11;
+                   opcode |= (num & 0x03) << 9;
+                   opcode |= (pa_number & FP_REG_RSEL ? 1 << 8 : 0);
+                   continue;
+                 }
+
+               /* Float mult operand 1 for fmpyadd, fmpysub */
+               case 'i':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   if (the_insn.fpof1 == SGL)
+                     {
+                       if (num < 16)
+                         {
+                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
+                           break;
+                         }
+                       num &= 0xF;
+                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, num, 21);
+                 }
 
 
-   ISBRANCH specifies whether or not this is parsing a condition
-   completer for a branch (vs a nullification completer for a
-   computational instruction).  */
+               /* Float mult operand 2 for fmpyadd, fmpysub */
+               case 'j':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   if (the_insn.fpof1 == SGL)
+                     {
+                       if (num < 16)
+                         {
+                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
+                           break;
+                         }
+                       num &= 0xF;
+                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
+                 }
 
 
-static int
-pa_parse_neg_add_cmpltr (s, isbranch)
-     char **s;
-     int isbranch;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
-  char *save_s = *s;
+               /* Float mult target for fmpyadd, fmpysub */
+               case 'k':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   if (the_insn.fpof1 == SGL)
+                     {
+                       if (num < 16)
+                         {
+                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
+                           break;
+                         }
+                       num &= 0xF;
+                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, num, 0);
+                 }
 
 
-  cmpltr = 0;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
-      if (strcasecmp (name, "tr") == 0)
-       {
-         cmpltr = 0;
-       }
-      else if (strcmp (name, "<>") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, ">=") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, ">") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcasecmp (name, "uv") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcasecmp (name, "vnz") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "nsv") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "ev") == 0)
-       {
-         cmpltr = 7;
-       }
-      /* If we have something like addb,n then there is no condition
-         completer.  */
-      else if (strcasecmp (name, "n") == 0 && isbranch)
-       {
-         cmpltr = 0;
-       }
-      else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
-    }
+               /* Float add operand 1 for fmpyadd, fmpysub */
+               case 'l':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   if (the_insn.fpof1 == SGL)
+                     {
+                       if (num < 16)
+                         {
+                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
+                           break;
+                         }
+                       num &= 0xF;
+                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, num, 6);
+                 }
 
 
-  /* Reset pointers if this was really a ,n for a branch instruction.  */
-  if (cmpltr == 0 && *name == 'n' && isbranch)
-    *s = save_s;
+               /* Float add target for fmpyadd, fmpysub */
+               case 'm':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   if (the_insn.fpof1 == SGL)
+                     {
+                       if (num < 16)
+                         {
+                           as_bad  (_("Invalid register for single precision fmpyadd or fmpysub"));
+                           break;
+                         }
+                       num &= 0xF;
+                       num |= (pa_number & FP_REG_RSEL ? 1 << 4 : 0);
+                     }
+                   INSERT_FIELD_AND_CONTINUE (opcode, num, 11);
+                 }
 
 
-  return cmpltr;
-}
+               /* Handle L/R register halves like 'x'.  */
+               case 'E':
+               case 'e':
+                 {
+                   if (!pa_parse_number (&s, 1))
+                     break;
+                   num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                   CHECK_FIELD (num, 31, 0, 0);
+                   opcode |= num << 16;
+                   if (need_pa11_opcode ())
+                     {
+                       opcode |= (pa_number & FP_REG_RSEL ? 1 << 1 : 0);
+                     }
+                   continue;
+                 }
 
 
-/* Parse a 64 bit wide mode add and branch completer returning the number (for
-   encoding in instrutions) of the given completer.  */
+               /* Float target register (PA 2.0 wide).  */
+               case 'x':
+                 if (!pa_parse_number (&s, 3))
+                   break;
+                 num = (pa_number & ~FP_REG_RSEL) - FP_REG_BASE;
+                 CHECK_FIELD (num, 31, 0, 0);
+                 INSERT_FIELD_AND_CONTINUE (opcode, num, 16);
 
 
-static int
-pa_parse_addb_64_cmpltr (s)
-     char **s;
-{
-  int cmpltr;
-  char *name = *s + 1;
-  char c;
-  char *save_s = *s;
-  int nullify = 0;
+               default:
+                 abort ();
+               }
+             break;
 
 
-  cmpltr = 0;
-  if (**s == ',')
-    {
-      *s += 1;
-      while (**s != ',' && **s != ' ' && **s != '\t')
-       *s += 1;
-      c = **s;
-      **s = 0x00;
-      if (strcmp (name, "=") == 0)
-       {
-         cmpltr = 1;
-       }
-      else if (strcmp (name, "<") == 0)
-       {
-         cmpltr = 2;
-       }
-      else if (strcmp (name, "<=") == 0)
-       {
-         cmpltr = 3;
-       }
-      else if (strcasecmp (name, "nuv") == 0)
-       {
-         cmpltr = 4;
-       }
-      else if (strcasecmp (name, "*=") == 0)
-       {
-         cmpltr = 5;
-       }
-      else if (strcasecmp (name, "*<") == 0)
-       {
-         cmpltr = 6;
-       }
-      else if (strcasecmp (name, "*<=") == 0)
-       {
-         cmpltr = 7;
-       }
-      else if (strcmp (name, "tr") == 0)
-       {
-         cmpltr = 8;
-       }
-      else if (strcmp (name, "<>") == 0)
-       {
-         cmpltr = 9;
-       }
-      else if (strcmp (name, ">=") == 0)
-       {
-         cmpltr = 10;
-       }
-      else if (strcmp (name, ">") == 0)
-       {
-         cmpltr = 11;
-       }
-      else if (strcasecmp (name, "uv") == 0)
-       {
-         cmpltr = 12;
-       }
-      else if (strcasecmp (name, "*<>") == 0)
-       {
-         cmpltr = 13;
+           default:
+             abort ();
+           }
+         break;
        }
        }
-      else if (strcasecmp (name, "*>=") == 0)
+
+      /* If this instruction is specific to a particular architecture,
+        then set a new architecture.  This automatic promotion crud is
+        for compatibility with HP's old assemblers only.  */
+      if (match == TRUE
+         && bfd_get_mach (stdoutput) < insn->arch
+         && !bfd_set_arch_mach (stdoutput, bfd_arch_hppa, insn->arch))
        {
        {
-         cmpltr = 14;
+         as_warn (_("could not update architecture and machine"));
+         match = FALSE;
        }
        }
-      else if (strcasecmp (name, "*>") == 0)
+
+ failed:
+      /* Check if the args matched.  */
+      if (!match)
        {
        {
-         cmpltr = 15;
+         if (&insn[1] - pa_opcodes < (int) NUMOPCODES
+             && !strcmp (insn->name, insn[1].name))
+           {
+             ++insn;
+             s = argstart;
+             continue;
+           }
+         else
+           {
+             as_bad (_("Invalid operands %s"), error_message);
+             return;
+           }
        }
        }
-      /* If we have something like addb,n then there is no condition
-         completer.  */
-      else if (strcasecmp (name, "n") == 0)
+      break;
+    }
+
+  if (immediate_check)
+    {
+      if (pos != -1 && len != -1 && pos < len - 1)
+        as_warn (_("Immediates %d and %d will give undefined behavior."),
+                       pos, len);
+    }
+
+  the_insn.opcode = opcode;
+}
+
+/* Assemble a single instruction storing it into a frag.  */
+
+void
+md_assemble (char *str)
+{
+  char *to;
+
+  /* The had better be something to assemble.  */
+  gas_assert (str);
+
+  /* If we are within a procedure definition, make sure we've
+     defined a label for the procedure; handle case where the
+     label was defined after the .PROC directive.
+
+     Note there's not need to diddle with the segment or fragment
+     for the label symbol in this case.  We have already switched
+     into the new $CODE$ subspace at this point.  */
+  if (within_procedure && last_call_info->start_symbol == NULL)
+    {
+      label_symbol_struct *label_symbol = pa_get_label ();
+
+      if (label_symbol)
        {
        {
-         cmpltr = 0;
-         nullify = 1;
+         if (label_symbol->lss_label)
+           {
+             last_call_info->start_symbol = label_symbol->lss_label;
+             symbol_get_bfdsym (label_symbol->lss_label)->flags
+               |= BSF_FUNCTION;
+#ifdef OBJ_SOM
+             /* Also handle allocation of a fixup to hold the unwind
+                information when the label appears after the proc/procend.  */
+             if (within_entry_exit)
+               {
+                 char *where;
+                 unsigned int u;
+
+                 where = frag_more (0);
+                 u = UNWIND_LOW32 (&last_call_info->ci_unwind.descriptor);
+                 fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
+                               NULL, (offsetT) 0, NULL,
+                               0, R_HPPA_ENTRY, e_fsel, 0, 0, u);
+               }
+#endif
+           }
+         else
+           as_bad (_("Missing function name for .PROC (corrupted label chain)"));
        }
       else
        }
       else
-       {
-         cmpltr = -1;
-       }
-      **s = c;
+       as_bad (_("Missing function name for .PROC"));
     }
 
     }
 
-  /* Reset pointers if this was really a ,n for a branch instruction.  */
-  if (nullify)
-    *s = save_s;
+  /* Assemble the instruction.  Results are saved into "the_insn".  */
+  pa_ip (str);
 
 
-  return cmpltr;
+  /* Get somewhere to put the assembled instruction.  */
+  to = frag_more (4);
+
+  /* Output the opcode.  */
+  md_number_to_chars (to, the_insn.opcode, 4);
+
+  /* If necessary output more stuff.  */
+  if (the_insn.reloc != R_HPPA_NONE)
+    fix_new_hppa (frag_now, (to - frag_now->fr_literal), 4, NULL,
+                 (offsetT) 0, &the_insn.exp, the_insn.pcrel,
+                 the_insn.reloc, the_insn.field_selector,
+                 the_insn.format, the_insn.arg_reloc, 0);
+
+#ifdef OBJ_ELF
+  dwarf2_emit_insn (4);
+#endif
 }
 
 #ifdef OBJ_SOM
 /* Handle an alignment directive.  Special so that we can update the
    alignment of the subspace if necessary.  */
 static void
 }
 
 #ifdef OBJ_SOM
 /* Handle an alignment directive.  Special so that we can update the
    alignment of the subspace if necessary.  */
 static void
-pa_align (bytes)
-     int bytes;
+pa_align (int bytes)
 {
   /* We must have a valid space and subspace.  */
   pa_check_current_space_and_subspace ();
 {
   /* We must have a valid space and subspace.  */
   pa_check_current_space_and_subspace ();
@@ -5868,21 +5809,17 @@ pa_align (bytes)
 
   /* If bytes is a power of 2, then update the current subspace's
      alignment if necessary.  */
 
   /* If bytes is a power of 2, then update the current subspace's
      alignment if necessary.  */
-  if (log2 (bytes) != -1)
-    record_alignment (current_subspace->ssd_seg, log2 (bytes));
+  if (exact_log2 (bytes) != -1)
+    record_alignment (current_subspace->ssd_seg, exact_log2 (bytes));
 }
 #endif
 
 /* Handle a .BLOCK type pseudo-op.  */
 
 static void
 }
 #endif
 
 /* Handle a .BLOCK type pseudo-op.  */
 
 static void
-pa_block (z)
-     int z ATTRIBUTE_UNUSED;
+pa_block (int z ATTRIBUTE_UNUSED)
 {
 {
-  char *p;
-  long int temp_fill;
   unsigned int temp_size;
   unsigned int temp_size;
-  unsigned int i;
 
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -5891,20 +5828,16 @@ pa_block (z)
 
   temp_size = get_absolute_expression ();
 
 
   temp_size = get_absolute_expression ();
 
-  /* Always fill with zeros, that's what the HP assembler does.  */
-  temp_fill = 0;
-
-  p = frag_var (rs_fill, (int) temp_size, (int) temp_size,
-               (relax_substateT) 0, (symbolS *) 0, (offsetT) 1, NULL);
-  memset (p, 0, temp_size);
-
-  /* Convert 2 bytes at a time.  */
-
-  for (i = 0; i < temp_size; i += 2)
+  if (temp_size > 0x3FFFFFFF)
+    {
+      as_bad (_("Argument to .BLOCK/.BLOCKZ must be between 0 and 0x3fffffff"));
+      temp_size = 0;
+    }
+  else
     {
     {
-      md_number_to_chars (p + i,
-                         (valueT) temp_fill,
-                         (int) ((temp_size - i) > 2 ? 2 : (temp_size - i)));
+      /* Always fill with zeros, that's what the HP assembler does.  */
+      char *p = frag_var (rs_fill, 1, 1, 0, NULL, temp_size, NULL);
+      *p = 0;
     }
 
   pa_undefine_label ();
     }
 
   pa_undefine_label ();
@@ -5914,19 +5847,18 @@ pa_block (z)
 /* Handle a .begin_brtab and .end_brtab pseudo-op.  */
 
 static void
 /* Handle a .begin_brtab and .end_brtab pseudo-op.  */
 
 static void
-pa_brtab (begin)
-     int begin ATTRIBUTE_UNUSED;
+pa_brtab (int begin ATTRIBUTE_UNUSED)
 {
 
 #ifdef OBJ_SOM
 {
 
 #ifdef OBJ_SOM
-  /* The BRTAB relocations are only availble in SOM (to denote
+  /* The BRTAB relocations are only available in SOM (to denote
      the beginning and end of branch tables).  */
   char *where = frag_more (0);
 
   fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                NULL, (offsetT) 0, NULL,
                0, begin ? R_HPPA_BEGIN_BRTAB : R_HPPA_END_BRTAB,
      the beginning and end of branch tables).  */
   char *where = frag_more (0);
 
   fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                NULL, (offsetT) 0, NULL,
                0, begin ? R_HPPA_BEGIN_BRTAB : R_HPPA_END_BRTAB,
-               e_fsel, 0, 0, NULL);
+               e_fsel, 0, 0, 0);
 #endif
 
   demand_empty_rest_of_line ();
 #endif
 
   demand_empty_rest_of_line ();
@@ -5935,8 +5867,7 @@ pa_brtab (begin)
 /* Handle a .begin_try and .end_try pseudo-op.  */
 
 static void
 /* Handle a .begin_try and .end_try pseudo-op.  */
 
 static void
-pa_try (begin)
-     int begin ATTRIBUTE_UNUSED;
+pa_try (int begin ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   expressionS exp;
 {
 #ifdef OBJ_SOM
   expressionS exp;
@@ -5945,32 +5876,15 @@ pa_try (begin)
   if (! begin)
     expression (&exp);
 
   if (! begin)
     expression (&exp);
 
-  /* The TRY relocations are only availble in SOM (to denote
+  /* The TRY relocations are only available in SOM (to denote
      the beginning and end of exception handling regions).  */
 
   fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                NULL, (offsetT) 0, begin ? NULL : &exp,
                0, begin ? R_HPPA_BEGIN_TRY : R_HPPA_END_TRY,
      the beginning and end of exception handling regions).  */
 
   fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                NULL, (offsetT) 0, begin ? NULL : &exp,
                0, begin ? R_HPPA_BEGIN_TRY : R_HPPA_END_TRY,
-               e_fsel, 0, 0, NULL);
-#endif
-
-  demand_empty_rest_of_line ();
-}
-
-/* Handle a .CALL pseudo-op.  This involves storing away information
-   about where arguments are to be found so the linker can detect
-   (and correct) argument location mismatches between caller and callee.  */
-
-static void
-pa_call (unused)
-     int unused ATTRIBUTE_UNUSED;
-{
-#ifdef OBJ_SOM
-  /* We must have a valid space and subspace.  */
-  pa_check_current_space_and_subspace ();
+               e_fsel, 0, 0, 0);
 #endif
 
 #endif
 
-  pa_call_args (&last_call_desc);
   demand_empty_rest_of_line ();
 }
 
   demand_empty_rest_of_line ();
 }
 
@@ -5978,36 +5892,30 @@ pa_call (unused)
    where the caller placed arguments to a function call.  */
 
 static void
    where the caller placed arguments to a function call.  */
 
 static void
-pa_call_args (call_desc)
-     struct call_desc *call_desc;
+pa_call_args (struct call_desc *call_desc)
 {
 {
-  char *name, c, *p;
+  char *name, c;
   unsigned int temp, arg_reloc;
 
   while (!is_end_of_statement ())
     {
   unsigned int temp, arg_reloc;
 
   while (!is_end_of_statement ())
     {
-      name = input_line_pointer;
-      c = get_symbol_end ();
+      c = get_symbol_name (&name);
       /* Process a source argument.  */
       if ((strncasecmp (name, "argw", 4) == 0))
        {
          temp = atoi (name + 4);
       /* Process a source argument.  */
       if ((strncasecmp (name, "argw", 4) == 0))
        {
          temp = atoi (name + 4);
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          input_line_pointer++;
-         name = input_line_pointer;
-         c = get_symbol_end ();
+         c = get_symbol_name (&name);
          arg_reloc = pa_build_arg_reloc (name);
          call_desc->arg_reloc |= pa_align_arg_reloc (temp, arg_reloc);
        }
       /* Process a return value.  */
       else if ((strncasecmp (name, "rtnval", 6) == 0))
        {
          arg_reloc = pa_build_arg_reloc (name);
          call_desc->arg_reloc |= pa_align_arg_reloc (temp, arg_reloc);
        }
       /* Process a return value.  */
       else if ((strncasecmp (name, "rtnval", 6) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          input_line_pointer++;
-         name = input_line_pointer;
-         c = get_symbol_end ();
+         c = get_symbol_name (&name);
          arg_reloc = pa_build_arg_reloc (name);
          call_desc->arg_reloc |= (arg_reloc & 0x3);
        }
          arg_reloc = pa_build_arg_reloc (name);
          call_desc->arg_reloc |= (arg_reloc & 0x3);
        }
@@ -6015,31 +5923,27 @@ pa_call_args (call_desc)
        {
          as_bad (_("Invalid .CALL argument: %s"), name);
        }
        {
          as_bad (_("Invalid .CALL argument: %s"), name);
        }
-      p = input_line_pointer;
-      *p = c;
+
+      (void) restore_line_pointer (c);
       if (!is_end_of_statement ())
        input_line_pointer++;
     }
 }
 
       if (!is_end_of_statement ())
        input_line_pointer++;
     }
 }
 
-/* Return TRUE if FRAG1 and FRAG2 are the same.  */
+/* Handle a .CALL pseudo-op.  This involves storing away information
+   about where arguments are to be found so the linker can detect
+   (and correct) argument location mismatches between caller and callee.  */
 
 
-static int
-is_same_frag (frag1, frag2)
-     fragS *frag1;
-     fragS *frag2;
+static void
+pa_call (int unused ATTRIBUTE_UNUSED)
 {
 {
+#ifdef OBJ_SOM
+  /* We must have a valid space and subspace.  */
+  pa_check_current_space_and_subspace ();
+#endif
 
 
-  if (frag1 == NULL)
-    return (FALSE);
-  else if (frag2 == NULL)
-    return (FALSE);
-  else if (frag1 == frag2)
-    return (TRUE);
-  else if (frag2->fr_type == rs_fill && frag2->fr_fix == 0)
-    return (is_same_frag (frag1, frag2->fr_next));
-  else
-    return (FALSE);
+  pa_call_args (&last_call_desc);
+  demand_empty_rest_of_line ();
 }
 
 #ifdef OBJ_ELF
 }
 
 #ifdef OBJ_ELF
@@ -6049,21 +5953,48 @@ is_same_frag (frag1, frag2)
    of the unwind spaces.  */
 
 static void
    of the unwind spaces.  */
 
 static void
-pa_build_unwind_subspace (call_info)
-     struct call_info *call_info;
+pa_build_unwind_subspace (struct call_info *call_info)
 {
 {
-  char *unwind;
   asection *seg, *save_seg;
   subsegT save_subseg;
   asection *seg, *save_seg;
   subsegT save_subseg;
-  unsigned int i;
+  unsigned int unwind;
   int reloc;
   int reloc;
-  char c, *p;
+  char *name, *p;
+  symbolS *symbolP;
 
   if ((bfd_get_section_flags (stdoutput, now_seg)
        & (SEC_ALLOC | SEC_LOAD | SEC_READONLY))
       != (SEC_ALLOC | SEC_LOAD | SEC_READONLY))
     return;
 
 
   if ((bfd_get_section_flags (stdoutput, now_seg)
        & (SEC_ALLOC | SEC_LOAD | SEC_READONLY))
       != (SEC_ALLOC | SEC_LOAD | SEC_READONLY))
     return;
 
+  if (call_info->start_symbol == NULL)
+    /* This can happen if there were errors earlier on in the assembly.  */
+    return;
+
+  /* Replace the start symbol with a local symbol that will be reduced
+     to a section offset.  This avoids problems with weak functions with
+     multiple definitions, etc.  */
+  name = concat ("L$\001start_", S_GET_NAME (call_info->start_symbol),
+                (char *) NULL);
+
+  /* If we have a .procend preceded by a .exit, then the symbol will have
+     already been defined.  In that case, we don't want another unwind
+     entry.  */
+  symbolP = symbol_find (name);
+  if (symbolP)
+    {
+      xfree (name);
+      return;
+    }
+  else
+    {
+      symbolP = symbol_new (name, now_seg,
+                           S_GET_VALUE (call_info->start_symbol), frag_now);
+      gas_assert (symbolP);
+      S_CLEAR_EXTERNAL (symbolP);
+      symbol_table_insert (symbolP);
+    }
+
   reloc = R_PARISC_SEGREL32;
   save_seg = now_seg;
   save_subseg = now_subseg;
   reloc = R_PARISC_SEGREL32;
   save_seg = now_seg;
   save_subseg = now_subseg;
@@ -6084,17 +6015,14 @@ pa_build_unwind_subspace (call_info)
 
   /* Get some space to hold relocation information for the unwind
      descriptor.  */
 
   /* Get some space to hold relocation information for the unwind
      descriptor.  */
-  p = frag_more (4);
-  md_number_to_chars (p, 0, 4);
+  p = frag_more (16);
 
   /* Relocation info. for start offset of the function.  */
 
   /* Relocation info. for start offset of the function.  */
+  md_number_to_chars (p, 0, 4);
   fix_new_hppa (frag_now, p - frag_now->fr_literal, 4,
   fix_new_hppa (frag_now, p - frag_now->fr_literal, 4,
-               call_info->start_symbol, (offsetT) 0,
+               symbolP, (offsetT) 0,
                (expressionS *) NULL, 0, reloc,
                (expressionS *) NULL, 0, reloc,
-               e_fsel, 32, 0, NULL);
-
-  p = frag_more (4);
-  md_number_to_chars (p, 0, 4);
+               e_fsel, 32, 0, 0);
 
   /* Relocation info. for end offset of the function.
 
 
   /* Relocation info. for end offset of the function.
 
@@ -6103,21 +6031,18 @@ pa_build_unwind_subspace (call_info)
      symbol into the symbol table.  It (should) end up giving the same
      value as call_info->start_symbol + function size once the linker is
      finished with its work.  */
      symbol into the symbol table.  It (should) end up giving the same
      value as call_info->start_symbol + function size once the linker is
      finished with its work.  */
-
-  fix_new_hppa (frag_now, p - frag_now->fr_literal, 4,
+  md_number_to_chars (p + 4, 0, 4);
+  fix_new_hppa (frag_now, p + 4 - frag_now->fr_literal, 4,
                call_info->end_symbol, (offsetT) 0,
                (expressionS *) NULL, 0, reloc,
                call_info->end_symbol, (offsetT) 0,
                (expressionS *) NULL, 0, reloc,
-               e_fsel, 32, 0, NULL);
+               e_fsel, 32, 0, 0);
 
 
-  /* Dump it.  */
-  unwind = (char *) &call_info->ci_unwind;
-  for (i = 8; i < sizeof (struct unwind_table); i++)
-    {
-      c = *(unwind + i);
-      {
-       FRAG_APPEND_1_CHAR (c);
-      }
-    }
+  /* Dump the descriptor.  */
+  unwind = UNWIND_LOW32 (&call_info->ci_unwind.descriptor);
+  md_number_to_chars (p + 8, unwind, 4);
+
+  unwind = UNWIND_HIGH32 (&call_info->ci_unwind.descriptor);
+  md_number_to_chars (p + 12, unwind, 4);
 
   /* Return back to the original segment/subsegment.  */
   subseg_set (save_seg, save_subseg);
 
   /* Return back to the original segment/subsegment.  */
   subseg_set (save_seg, save_subseg);
@@ -6129,10 +6054,9 @@ pa_build_unwind_subspace (call_info)
    .ENTER and .LEAVE.  */
 
 static void
    .ENTER and .LEAVE.  */
 
 static void
-pa_callinfo (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_callinfo (int unused ATTRIBUTE_UNUSED)
 {
 {
-  char *name, c, *p;
+  char *name, c;
   int temp;
 
 #ifdef OBJ_SOM
   int temp;
 
 #ifdef OBJ_SOM
@@ -6151,13 +6075,11 @@ pa_callinfo (unused)
   /* Iterate over the .CALLINFO arguments.  */
   while (!is_end_of_statement ())
     {
   /* Iterate over the .CALLINFO arguments.  */
   while (!is_end_of_statement ())
     {
-      name = input_line_pointer;
-      c = get_symbol_end ();
+      c = get_symbol_name (&name);
       /* Frame size specification.  */
       if ((strncasecmp (name, "frame", 5) == 0))
        {
       /* Frame size specification.  */
       if ((strncasecmp (name, "frame", 5) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          temp = get_absolute_expression ();
          if ((temp & 0x3) != 0)
          input_line_pointer++;
          temp = get_absolute_expression ();
          if ((temp & 0x3) != 0)
@@ -6168,13 +6090,11 @@ pa_callinfo (unused)
 
          /* callinfo is in bytes and unwind_desc is in 8 byte units.  */
          last_call_info->ci_unwind.descriptor.frame_size = temp / 8;
 
          /* callinfo is in bytes and unwind_desc is in 8 byte units.  */
          last_call_info->ci_unwind.descriptor.frame_size = temp / 8;
-
        }
       /* Entry register (GR, GR and SR) specifications.  */
       else if ((strncasecmp (name, "entry_gr", 8) == 0))
        {
        }
       /* Entry register (GR, GR and SR) specifications.  */
       else if ((strncasecmp (name, "entry_gr", 8) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          temp = get_absolute_expression ();
          /* The HP assembler accepts 19 as the high bound for ENTRY_GR
          input_line_pointer++;
          temp = get_absolute_expression ();
          /* The HP assembler accepts 19 as the high bound for ENTRY_GR
@@ -6186,8 +6106,7 @@ pa_callinfo (unused)
        }
       else if ((strncasecmp (name, "entry_fr", 8) == 0))
        {
        }
       else if ((strncasecmp (name, "entry_fr", 8) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          temp = get_absolute_expression ();
          /* Similarly the HP assembler takes 31 as the high bound even
          input_line_pointer++;
          temp = get_absolute_expression ();
          /* Similarly the HP assembler takes 31 as the high bound even
@@ -6198,53 +6117,46 @@ pa_callinfo (unused)
        }
       else if ((strncasecmp (name, "entry_sr", 8) == 0))
        {
        }
       else if ((strncasecmp (name, "entry_sr", 8) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          temp = get_absolute_expression ();
          if (temp != 3)
            as_bad (_("Value for ENTRY_SR must be 3\n"));
        }
       /* Note whether or not this function performs any calls.  */
          input_line_pointer++;
          temp = get_absolute_expression ();
          if (temp != 3)
            as_bad (_("Value for ENTRY_SR must be 3\n"));
        }
       /* Note whether or not this function performs any calls.  */
-      else if ((strncasecmp (name, "calls", 5) == 0) ||
-              (strncasecmp (name, "caller", 6) == 0))
+      else if ((strncasecmp (name, "calls", 5) == 0)
+              || (strncasecmp (name, "caller", 6) == 0))
        {
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
        }
       else if ((strncasecmp (name, "no_calls", 8) == 0))
        {
        }
       else if ((strncasecmp (name, "no_calls", 8) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
        }
       /* Should RP be saved into the stack.  */
       else if ((strncasecmp (name, "save_rp", 7) == 0))
        {
        }
       /* Should RP be saved into the stack.  */
       else if ((strncasecmp (name, "save_rp", 7) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          last_call_info->ci_unwind.descriptor.save_rp = 1;
        }
       /* Likewise for SP.  */
       else if ((strncasecmp (name, "save_sp", 7) == 0))
        {
          last_call_info->ci_unwind.descriptor.save_rp = 1;
        }
       /* Likewise for SP.  */
       else if ((strncasecmp (name, "save_sp", 7) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          last_call_info->ci_unwind.descriptor.save_sp = 1;
        }
       /* Is this an unwindable procedure.  If so mark it so
          last_call_info->ci_unwind.descriptor.save_sp = 1;
        }
       /* Is this an unwindable procedure.  If so mark it so
-         in the unwind descriptor.  */
+        in the unwind descriptor.  */
       else if ((strncasecmp (name, "no_unwind", 9) == 0))
        {
       else if ((strncasecmp (name, "no_unwind", 9) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          last_call_info->ci_unwind.descriptor.cannot_unwind = 1;
        }
       /* Is this an interrupt routine.  If so mark it in the
          last_call_info->ci_unwind.descriptor.cannot_unwind = 1;
        }
       /* Is this an interrupt routine.  If so mark it in the
-         unwind descriptor.  */
+        unwind descriptor.  */
       else if ((strncasecmp (name, "hpux_int", 7) == 0))
        {
       else if ((strncasecmp (name, "hpux_int", 7) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          last_call_info->ci_unwind.descriptor.hpux_interrupt_marker = 1;
        }
       /* Is this a millicode routine.  "millicode" isn't in my
          last_call_info->ci_unwind.descriptor.hpux_interrupt_marker = 1;
        }
       /* Is this a millicode routine.  "millicode" isn't in my
@@ -6253,15 +6165,15 @@ pa_callinfo (unused)
         to drop the information, so we'll accept it too.  */
       else if ((strncasecmp (name, "millicode", 9) == 0))
        {
         to drop the information, so we'll accept it too.  */
       else if ((strncasecmp (name, "millicode", 9) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          last_call_info->ci_unwind.descriptor.millicode = 1;
        }
       else
        {
          as_bad (_("Invalid .CALLINFO argument: %s"), name);
          last_call_info->ci_unwind.descriptor.millicode = 1;
        }
       else
        {
          as_bad (_("Invalid .CALLINFO argument: %s"), name);
-         *input_line_pointer = c;
+         (void) restore_line_pointer (c);
        }
        }
+
       if (!is_end_of_statement ())
        input_line_pointer++;
     }
       if (!is_end_of_statement ())
        input_line_pointer++;
     }
@@ -6269,12 +6181,12 @@ pa_callinfo (unused)
   demand_empty_rest_of_line ();
 }
 
   demand_empty_rest_of_line ();
 }
 
-#if !(defined (OBJ_ELF) && defined (TE_LINUX))
+#if !(defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD)))
 /* Switch to the text space.  Like s_text, but delete our
    label when finished.  */
 /* Switch to the text space.  Like s_text, but delete our
    label when finished.  */
+
 static void
 static void
-pa_text (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_text (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   current_space = is_defined_space ("$TEXT$");
 {
 #ifdef OBJ_SOM
   current_space = is_defined_space ("$TEXT$");
@@ -6287,9 +6199,9 @@ pa_text (unused)
 }
 
 /* Switch to the data space.  As usual delete our label.  */
 }
 
 /* Switch to the data space.  As usual delete our label.  */
+
 static void
 static void
-pa_data (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_data (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   current_space = is_defined_space ("$PRIVATE$");
 {
 #ifdef OBJ_SOM
   current_space = is_defined_space ("$PRIVATE$");
@@ -6301,7 +6213,7 @@ pa_data (unused)
 }
 
 /* This is different than the standard GAS s_comm(). On HP9000/800 machines,
 }
 
 /* This is different than the standard GAS s_comm(). On HP9000/800 machines,
-   the .comm pseudo-op has the following symtax:
+   the .comm pseudo-op has the following syntax:
 
    <label> .comm <length>
 
 
    <label> .comm <length>
 
@@ -6321,8 +6233,7 @@ pa_data (unused)
    This also makes error detection all but impossible.  */
 
 static void
    This also makes error detection all but impossible.  */
 
 static void
-pa_comm (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_comm (int unused ATTRIBUTE_UNUSED)
 {
   unsigned int size;
   symbolS *symbol;
 {
   unsigned int size;
   symbolS *symbol;
@@ -6338,32 +6249,32 @@ pa_comm (unused)
 
   if (symbol)
     {
 
   if (symbol)
     {
+      symbol_get_bfdsym (symbol)->flags |= BSF_OBJECT;
       S_SET_VALUE (symbol, size);
       S_SET_VALUE (symbol, size);
-      S_SET_SEGMENT (symbol, bfd_und_section_ptr);
+      S_SET_SEGMENT (symbol, bfd_com_section_ptr);
       S_SET_EXTERNAL (symbol);
 
       /* colon() has already set the frag to the current location in the
       S_SET_EXTERNAL (symbol);
 
       /* colon() has already set the frag to the current location in the
-         current subspace; we need to reset the fragment to the zero address
-         fragment.  We also need to reset the segment pointer.  */
+        current subspace; we need to reset the fragment to the zero address
+        fragment.  We also need to reset the segment pointer.  */
       symbol_set_frag (symbol, &zero_address_frag);
     }
   demand_empty_rest_of_line ();
 }
       symbol_set_frag (symbol, &zero_address_frag);
     }
   demand_empty_rest_of_line ();
 }
-#endif /* !(defined (OBJ_ELF) && defined (TE_LINUX)) */
+#endif /* !(defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD))) */
 
 /* Process a .END pseudo-op.  */
 
 static void
 
 /* Process a .END pseudo-op.  */
 
 static void
-pa_end (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_end (int unused ATTRIBUTE_UNUSED)
 {
   demand_empty_rest_of_line ();
 }
 
 /* Process a .ENTER pseudo-op.  This is not supported.  */
 {
   demand_empty_rest_of_line ();
 }
 
 /* Process a .ENTER pseudo-op.  This is not supported.  */
+
 static void
 static void
-pa_enter (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_enter (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -6375,10 +6286,10 @@ pa_enter (unused)
 }
 
 /* Process a .ENTRY pseudo-op.  .ENTRY marks the beginning of the
 }
 
 /* Process a .ENTRY pseudo-op.  .ENTRY marks the beginning of the
-   procesure.  */
+   procedure.  */
+
 static void
 static void
-pa_entry (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_entry (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -6407,12 +6318,14 @@ pa_entry (unused)
      denote the entry and exit points.  */
   if (last_call_info->start_symbol != NULL)
     {
      denote the entry and exit points.  */
   if (last_call_info->start_symbol != NULL)
     {
-      char *where = frag_more (0);
+      char *where;
+      unsigned int u;
 
 
+      where = frag_more (0);
+      u = UNWIND_LOW32 (&last_call_info->ci_unwind.descriptor);
       fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                    NULL, (offsetT) 0, NULL,
       fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                    NULL, (offsetT) 0, NULL,
-                   0, R_HPPA_ENTRY, e_fsel, 0, 0,
-                   (int *) &last_call_info->ci_unwind.descriptor);
+                   0, R_HPPA_ENTRY, e_fsel, 0, 0, u);
     }
 #endif
 }
     }
 #endif
 }
@@ -6423,10 +6336,9 @@ pa_entry (unused)
 static int fudge_reg_expressions;
 
 int
 static int fudge_reg_expressions;
 
 int
-hppa_force_reg_syms_absolute (resultP, op, rightP)
-     expressionS *resultP;
-     operatorT op ATTRIBUTE_UNUSED;
-     expressionS *rightP;
+hppa_force_reg_syms_absolute (expressionS *resultP,
+                             operatorT op ATTRIBUTE_UNUSED,
+                             expressionS *rightP)
 {
   if (fudge_reg_expressions
       && rightP->X_op == O_register
 {
   if (fudge_reg_expressions
       && rightP->X_op == O_register
@@ -6441,8 +6353,7 @@ hppa_force_reg_syms_absolute (resultP, op, rightP)
 /* Handle a .EQU pseudo-op.  */
 
 static void
 /* Handle a .EQU pseudo-op.  */
 
 static void
-pa_equ (reg)
-     int reg;
+pa_equ (int reg)
 {
   label_symbol_struct *label_symbol = pa_get_label ();
   symbolS *symbol;
 {
   label_symbol_struct *label_symbol = pa_get_label ();
   symbolS *symbol;
@@ -6490,12 +6401,65 @@ pa_equ (reg)
   demand_empty_rest_of_line ();
 }
 
   demand_empty_rest_of_line ();
 }
 
+#ifdef OBJ_ELF
+/* Mark the end of a function so that it's possible to compute
+   the size of the function in elf_hppa_final_processing.  */
+
+static void
+hppa_elf_mark_end_of_function (void)
+{
+  /* ELF does not have EXIT relocations.  All we do is create a
+     temporary symbol marking the end of the function.  */
+  char *name;
+  symbolS *symbolP;
+
+  if (last_call_info == NULL || last_call_info->start_symbol == NULL)
+    {
+      /* We have already warned about a missing label,
+        or other problems.  */
+      return;
+    }
+
+  name = concat ("L$\001end_", S_GET_NAME (last_call_info->start_symbol),
+                (char *) NULL);
+
+  /* If we have a .exit followed by a .procend, then the
+     symbol will have already been defined.  */
+  symbolP = symbol_find (name);
+  if (symbolP)
+    {
+      /* The symbol has already been defined!  This can
+        happen if we have a .exit followed by a .procend.
+
+        This is *not* an error.  All we want to do is free
+        the memory we just allocated for the name and continue.  */
+      xfree (name);
+    }
+  else
+    {
+      /* symbol value should be the offset of the
+        last instruction of the function */
+      symbolP = symbol_new (name, now_seg, (valueT) (frag_now_fix () - 4),
+                           frag_now);
+
+      gas_assert (symbolP);
+      S_CLEAR_EXTERNAL (symbolP);
+      symbol_table_insert (symbolP);
+    }
+
+  if (symbolP)
+    last_call_info->end_symbol = symbolP;
+  else
+    as_bad (_("Symbol '%s' could not be created."), name);
+}
+#endif
+
 /* Helper function.  Does processing for the end of a function.  This
    usually involves creating some relocations or building special
    symbols to mark the end of the function.  */
 
 static void
 /* Helper function.  Does processing for the end of a function.  This
    usually involves creating some relocations or building special
    symbols to mark the end of the function.  */
 
 static void
-process_exit ()
+process_exit (void)
 {
   char *where;
 
 {
   char *where;
 
@@ -6505,6 +6469,7 @@ process_exit ()
   /* Mark the end of the function, stuff away the location of the frag
      for the end of the function, and finally call pa_build_unwind_subspace
      to add an entry in the unwind table.  */
   /* Mark the end of the function, stuff away the location of the frag
      for the end of the function, and finally call pa_build_unwind_subspace
      to add an entry in the unwind table.  */
+  (void) where;
   hppa_elf_mark_end_of_function ();
   pa_build_unwind_subspace (last_call_info);
 #else
   hppa_elf_mark_end_of_function ();
   pa_build_unwind_subspace (last_call_info);
 #else
@@ -6520,15 +6485,14 @@ process_exit ()
   fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                NULL, (offsetT) 0,
                NULL, 0, R_HPPA_EXIT, e_fsel, 0, 0,
   fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                NULL, (offsetT) 0,
                NULL, 0, R_HPPA_EXIT, e_fsel, 0, 0,
-               (int *) &last_call_info->ci_unwind.descriptor + 1);
+               UNWIND_HIGH32 (&last_call_info->ci_unwind.descriptor));
 #endif
 }
 
 /* Process a .EXIT pseudo-op.  */
 
 static void
 #endif
 }
 
 /* Process a .EXIT pseudo-op.  */
 
 static void
-pa_exit (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_exit (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -6555,62 +6519,17 @@ pa_exit (unused)
   demand_empty_rest_of_line ();
 }
 
   demand_empty_rest_of_line ();
 }
 
-/* Process a .EXPORT directive.  This makes functions external
-   and provides information such as argument relocation entries
-   to callers.  */
-
-static void
-pa_export (unused)
-     int unused ATTRIBUTE_UNUSED;
-{
-  char *name, c, *p;
-  symbolS *symbol;
-
-  name = input_line_pointer;
-  c = get_symbol_end ();
-  /* Make sure the given symbol exists.  */
-  if ((symbol = symbol_find_or_make (name)) == NULL)
-    {
-      as_bad (_("Cannot define export symbol: %s\n"), name);
-      p = input_line_pointer;
-      *p = c;
-      input_line_pointer++;
-    }
-  else
-    {
-      /* OK.  Set the external bits and process argument relocations.
-         For the HP, weak and global are not mutually exclusive.
-         S_SET_EXTERNAL will not set BSF_GLOBAL if WEAK is set.
-         Call S_SET_EXTERNAL to get the other processing.  Manually
-         set BSF_GLOBAL when we get back.  */
-      S_SET_EXTERNAL (symbol);
-      symbol_get_bfdsym (symbol)->flags |= BSF_GLOBAL;
-      p = input_line_pointer;
-      *p = c;
-      if (!is_end_of_statement ())
-       {
-         input_line_pointer++;
-         pa_type_args (symbol, 1);
-       }
-    }
-
-  demand_empty_rest_of_line ();
-}
-
 /* Helper function to process arguments to a .EXPORT pseudo-op.  */
 
 static void
 /* Helper function to process arguments to a .EXPORT pseudo-op.  */
 
 static void
-pa_type_args (symbolP, is_export)
-     symbolS *symbolP;
-     int is_export;
+pa_type_args (symbolS *symbolP, int is_export)
 {
 {
-  char *name, c, *p;
+  char *name, c;
   unsigned int temp, arg_reloc;
   pa_symbol_type type = SYMBOL_TYPE_UNKNOWN;
   asymbol *bfdsym = symbol_get_bfdsym (symbolP);
 
   if (strncasecmp (input_line_pointer, "absolute", 8) == 0)
   unsigned int temp, arg_reloc;
   pa_symbol_type type = SYMBOL_TYPE_UNKNOWN;
   asymbol *bfdsym = symbol_get_bfdsym (symbolP);
 
   if (strncasecmp (input_line_pointer, "absolute", 8) == 0)
-
     {
       input_line_pointer += 8;
       bfdsym->flags &= ~BSF_FUNCTION;
     {
       input_line_pointer += 8;
       bfdsym->flags &= ~BSF_FUNCTION;
@@ -6621,11 +6540,11 @@ pa_type_args (symbolP, is_export)
     {
       input_line_pointer += 4;
       /* IMPORTing/EXPORTing CODE types for functions is meaningless for SOM,
     {
       input_line_pointer += 4;
       /* IMPORTing/EXPORTing CODE types for functions is meaningless for SOM,
-         instead one should be IMPORTing/EXPORTing ENTRY types.
+        instead one should be IMPORTing/EXPORTing ENTRY types.
 
 
-         Complain if one tries to EXPORT a CODE type since that's never
-         done.  Both GCC and HP C still try to IMPORT CODE types, so
-         silently fix them to be ENTRY types.  */
+        Complain if one tries to EXPORT a CODE type since that's never
+        done.  Both GCC and HP C still try to IMPORT CODE types, so
+        silently fix them to be ENTRY types.  */
       if (S_IS_FUNCTION (symbolP))
        {
          if (is_export)
       if (S_IS_FUNCTION (symbolP))
        {
          if (is_export)
@@ -6692,6 +6611,8 @@ pa_type_args (symbolP, is_export)
      to the SOM BFD backend.  */
 #ifdef obj_set_symbol_type
   obj_set_symbol_type (bfdsym, (int) type);
      to the SOM BFD backend.  */
 #ifdef obj_set_symbol_type
   obj_set_symbol_type (bfdsym, (int) type);
+#else
+  (void) type;
 #endif
 
   /* Now that the type of the exported symbol has been handled,
 #endif
 
   /* Now that the type of the exported symbol has been handled,
@@ -6700,74 +6621,110 @@ pa_type_args (symbolP, is_export)
     {
       if (*input_line_pointer == ',')
        input_line_pointer++;
     {
       if (*input_line_pointer == ',')
        input_line_pointer++;
-      name = input_line_pointer;
-      c = get_symbol_end ();
+      c = get_symbol_name (&name);
       /* Argument sources.  */
       if ((strncasecmp (name, "argw", 4) == 0))
        {
       /* Argument sources.  */
       if ((strncasecmp (name, "argw", 4) == 0))
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          temp = atoi (name + 4);
          input_line_pointer++;
          temp = atoi (name + 4);
-         name = input_line_pointer;
-         c = get_symbol_end ();
+         c = get_symbol_name (&name);
          arg_reloc = pa_align_arg_reloc (temp, pa_build_arg_reloc (name));
 #if defined (OBJ_SOM) || defined (ELF_ARG_RELOC)
          symbol_arg_reloc_info (symbolP) |= arg_reloc;
          arg_reloc = pa_align_arg_reloc (temp, pa_build_arg_reloc (name));
 #if defined (OBJ_SOM) || defined (ELF_ARG_RELOC)
          symbol_arg_reloc_info (symbolP) |= arg_reloc;
+#else
+         (void) arg_reloc;
 #endif
 #endif
-         *input_line_pointer = c;
+         (void) restore_line_pointer (c);
        }
       /* The return value.  */
       else if ((strncasecmp (name, "rtnval", 6)) == 0)
        {
        }
       /* The return value.  */
       else if ((strncasecmp (name, "rtnval", 6)) == 0)
        {
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          input_line_pointer++;
-         name = input_line_pointer;
-         c = get_symbol_end ();
+         c = get_symbol_name (&name);
          arg_reloc = pa_build_arg_reloc (name);
 #if defined (OBJ_SOM) || defined (ELF_ARG_RELOC)
          symbol_arg_reloc_info (symbolP) |= arg_reloc;
          arg_reloc = pa_build_arg_reloc (name);
 #if defined (OBJ_SOM) || defined (ELF_ARG_RELOC)
          symbol_arg_reloc_info (symbolP) |= arg_reloc;
+#else
+         (void) arg_reloc;
 #endif
 #endif
-         *input_line_pointer = c;
+         (void) restore_line_pointer (c);
        }
        }
-      /* Privelege level.  */
+      /* Privilege level.  */
       else if ((strncasecmp (name, "priv_lev", 8)) == 0)
        {
       else if ((strncasecmp (name, "priv_lev", 8)) == 0)
        {
-         p = input_line_pointer;
-         *p = c;
+         char *priv;
+
+         (void) restore_line_pointer (c);
          input_line_pointer++;
          temp = atoi (input_line_pointer);
 #ifdef OBJ_SOM
          ((obj_symbol_type *) bfdsym)->tc_data.ap.hppa_priv_level = temp;
 #endif
          input_line_pointer++;
          temp = atoi (input_line_pointer);
 #ifdef OBJ_SOM
          ((obj_symbol_type *) bfdsym)->tc_data.ap.hppa_priv_level = temp;
 #endif
-         c = get_symbol_end ();
-         *input_line_pointer = c;
+         c = get_symbol_name (&priv);
+         (void) restore_line_pointer (c);
        }
       else
        {
          as_bad (_("Undefined .EXPORT/.IMPORT argument (ignored): %s"), name);
        }
       else
        {
          as_bad (_("Undefined .EXPORT/.IMPORT argument (ignored): %s"), name);
-         p = input_line_pointer;
-         *p = c;
+         (void) restore_line_pointer (c);
        }
        }
+
       if (!is_end_of_statement ())
        input_line_pointer++;
     }
 }
 
       if (!is_end_of_statement ())
        input_line_pointer++;
     }
 }
 
+/* Process a .EXPORT directive.  This makes functions external
+   and provides information such as argument relocation entries
+   to callers.  */
+
+static void
+pa_export (int unused ATTRIBUTE_UNUSED)
+{
+  char *name, c;
+  symbolS *symbol;
+
+  c = get_symbol_name (&name);
+  /* Make sure the given symbol exists.  */
+  if ((symbol = symbol_find_or_make (name)) == NULL)
+    {
+      as_bad (_("Cannot define export symbol: %s\n"), name);
+      restore_line_pointer (c);
+      input_line_pointer++;
+    }
+  else
+    {
+      /* OK.  Set the external bits and process argument relocations.
+        For the HP, weak and global are not mutually exclusive.
+        S_SET_EXTERNAL will not set BSF_GLOBAL if WEAK is set.
+        Call S_SET_EXTERNAL to get the other processing.  Manually
+        set BSF_GLOBAL when we get back.  */
+      S_SET_EXTERNAL (symbol);
+      symbol_get_bfdsym (symbol)->flags |= BSF_GLOBAL;
+      (void) restore_line_pointer (c);
+      if (!is_end_of_statement ())
+       {
+         input_line_pointer++;
+         pa_type_args (symbol, 1);
+       }
+    }
+
+  demand_empty_rest_of_line ();
+}
+
 /* Handle an .IMPORT pseudo-op.  Any symbol referenced in a given
    assembly file must either be defined in the assembly file, or
    explicitly IMPORTED from another.  */
 
 static void
 /* Handle an .IMPORT pseudo-op.  Any symbol referenced in a given
    assembly file must either be defined in the assembly file, or
    explicitly IMPORTED from another.  */
 
 static void
-pa_import (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_import (int unused ATTRIBUTE_UNUSED)
 {
 {
-  char *name, c, *p;
+  char *name, c;
   symbolS *symbol;
 
   symbolS *symbol;
 
-  name = input_line_pointer;
-  c = get_symbol_end ();
+  c = get_symbol_name (&name);
 
   symbol = symbol_find (name);
   /* Ugh.  We might be importing a symbol defined earlier in the file,
 
   symbol = symbol_find (name);
   /* Ugh.  We might be importing a symbol defined earlier in the file,
@@ -6776,8 +6733,7 @@ pa_import (unused)
   if (symbol == NULL || !S_IS_DEFINED (symbol))
     {
       symbol = symbol_find_or_make (name);
   if (symbol == NULL || !S_IS_DEFINED (symbol))
     {
       symbol = symbol_find_or_make (name);
-      p = input_line_pointer;
-      *p = c;
+      (void) restore_line_pointer (c);
 
       if (!is_end_of_statement ())
        {
 
       if (!is_end_of_statement ())
        {
@@ -6786,9 +6742,9 @@ pa_import (unused)
        }
       else
        {
        }
       else
        {
-         /* Sigh.  To be compatable with the HP assembler and to help
+         /* Sigh.  To be compatible with the HP assembler and to help
             poorly written assembly code, we assign a type based on
             poorly written assembly code, we assign a type based on
-            the the current segment.  Note only BSF_FUNCTION really
+            the current segment.  Note only BSF_FUNCTION really
             matters, we do not need to set the full SYMBOL_TYPE_* info.  */
          if (now_seg == text_section)
            symbol_get_bfdsym (symbol)->flags |= BSF_FUNCTION;
             matters, we do not need to set the full SYMBOL_TYPE_* info.  */
          if (now_seg == text_section)
            symbol_get_bfdsym (symbol)->flags |= BSF_FUNCTION;
@@ -6812,19 +6768,16 @@ pa_import (unused)
 /* Handle a .LABEL pseudo-op.  */
 
 static void
 /* Handle a .LABEL pseudo-op.  */
 
 static void
-pa_label (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_label (int unused ATTRIBUTE_UNUSED)
 {
 {
-  char *name, c, *p;
+  char *name, c;
 
 
-  name = input_line_pointer;
-  c = get_symbol_end ();
+  c = get_symbol_name (&name);
 
   if (strlen (name) > 0)
     {
       colon (name);
 
   if (strlen (name) > 0)
     {
       colon (name);
-      p = input_line_pointer;
-      *p = c;
+      (void) restore_line_pointer (c);
     }
   else
     {
     }
   else
     {
@@ -6842,8 +6795,7 @@ pa_label (unused)
 /* Handle a .LEAVE pseudo-op.  This is not supported yet.  */
 
 static void
 /* Handle a .LEAVE pseudo-op.  This is not supported yet.  */
 
 static void
-pa_leave (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_leave (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -6857,8 +6809,7 @@ pa_leave (unused)
 /* Handle a .LEVEL pseudo-op.  */
 
 static void
 /* Handle a .LEVEL pseudo-op.  */
 
 static void
-pa_level (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_level (int unused ATTRIBUTE_UNUSED)
 {
   char *level;
 
 {
   char *level;
 
@@ -6898,8 +6849,7 @@ pa_level (unused)
 /* Handle a .ORIGIN pseudo-op.  */
 
 static void
 /* Handle a .ORIGIN pseudo-op.  */
 
 static void
-pa_origin (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_origin (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -6914,27 +6864,23 @@ pa_origin (unused)
    is for static functions.  FIXME.  Should share more code with .EXPORT.  */
 
 static void
    is for static functions.  FIXME.  Should share more code with .EXPORT.  */
 
 static void
-pa_param (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_param (int unused ATTRIBUTE_UNUSED)
 {
 {
-  char *name, c, *p;
+  char *name, c;
   symbolS *symbol;
 
   symbolS *symbol;
 
-  name = input_line_pointer;
-  c = get_symbol_end ();
+  c = get_symbol_name (&name);
 
   if ((symbol = symbol_find_or_make (name)) == NULL)
     {
       as_bad (_("Cannot define static symbol: %s\n"), name);
 
   if ((symbol = symbol_find_or_make (name)) == NULL)
     {
       as_bad (_("Cannot define static symbol: %s\n"), name);
-      p = input_line_pointer;
-      *p = c;
+      (void) restore_line_pointer (c);
       input_line_pointer++;
     }
   else
     {
       S_CLEAR_EXTERNAL (symbol);
       input_line_pointer++;
     }
   else
     {
       S_CLEAR_EXTERNAL (symbol);
-      p = input_line_pointer;
-      *p = c;
+      (void) restore_line_pointer (c);
       if (!is_end_of_statement ())
        {
          input_line_pointer++;
       if (!is_end_of_statement ())
        {
          input_line_pointer++;
@@ -6949,8 +6895,7 @@ pa_param (unused)
    of a procedure from a syntactical point of view.  */
 
 static void
    of a procedure from a syntactical point of view.  */
 
 static void
-pa_proc (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_proc (int unused ATTRIBUTE_UNUSED)
 {
   struct call_info *call_info;
 
 {
   struct call_info *call_info;
 
@@ -6967,7 +6912,7 @@ pa_proc (unused)
   within_procedure = TRUE;
 
   /* Create another call_info structure.  */
   within_procedure = TRUE;
 
   /* Create another call_info structure.  */
-  call_info = (struct call_info *) xmalloc (sizeof (struct call_info));
+  call_info = XNEW (struct call_info);
 
   if (!call_info)
     as_fatal (_("Cannot allocate unwind descriptor\n"));
 
   if (!call_info)
     as_fatal (_("Cannot allocate unwind descriptor\n"));
@@ -7015,14 +6960,12 @@ pa_proc (unused)
   demand_empty_rest_of_line ();
 }
 
   demand_empty_rest_of_line ();
 }
 
-/* Process the syntatical end of a procedure.  Make sure all the
+/* Process the syntactical end of a procedure.  Make sure all the
    appropriate pseudo-ops were found within the procedure.  */
 
 static void
    appropriate pseudo-ops were found within the procedure.  */
 
 static void
-pa_procend (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_procend (int unused ATTRIBUTE_UNUSED)
 {
 {
-
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
   pa_check_current_space_and_subspace ();
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
   pa_check_current_space_and_subspace ();
@@ -7051,12 +6994,14 @@ pa_procend (unused)
                 information when the label appears after the proc/procend.  */
              if (within_entry_exit)
                {
                 information when the label appears after the proc/procend.  */
              if (within_entry_exit)
                {
-                 char *where = frag_more (0);
+                 char *where;
+                 unsigned int u;
 
 
+                 where = frag_more (0);
+                 u = UNWIND_LOW32 (&last_call_info->ci_unwind.descriptor);
                  fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                                NULL, (offsetT) 0, NULL,
                  fix_new_hppa (frag_now, where - frag_now->fr_literal, 0,
                                NULL, (offsetT) 0, NULL,
-                               0, R_HPPA_ENTRY, e_fsel, 0, 0,
-                               (int *) &last_call_info->ci_unwind.descriptor);
+                               0, R_HPPA_ENTRY, e_fsel, 0, 0, u);
                }
 #endif
            }
                }
 #endif
            }
@@ -7092,8 +7037,7 @@ pa_procend (unused)
    return log2 (VALUE).  Else return -1.  */
 
 static int
    return log2 (VALUE).  Else return -1.  */
 
 static int
-log2 (value)
-     int value;
+exact_log2 (int value)
 {
   int shift = 0;
 
 {
   int shift = 0;
 
@@ -7109,7 +7053,7 @@ log2 (value)
 /* Check to make sure we have a valid space and subspace.  */
 
 static void
 /* Check to make sure we have a valid space and subspace.  */
 
 static void
-pa_check_current_space_and_subspace ()
+pa_check_current_space_and_subspace (void)
 {
   if (current_space == NULL)
     as_fatal (_("Not in a space.\n"));
 {
   if (current_space == NULL)
     as_fatal (_("Not in a space.\n"));
@@ -7123,9 +7067,7 @@ pa_check_current_space_and_subspace ()
    by the parameters to the .SPACE directive.  */
 
 static sd_chain_struct *
    by the parameters to the .SPACE directive.  */
 
 static sd_chain_struct *
-pa_parse_space_stmt (space_name, create_flag)
-     char *space_name;
-     int create_flag;
+pa_parse_space_stmt (const char *space_name, int create_flag)
 {
   char *name, *ptemp, c;
   char loadable, defined, private, sort;
 {
   char *name, *ptemp, c;
   char loadable, defined, private, sort;
@@ -7133,7 +7075,7 @@ pa_parse_space_stmt (space_name, create_flag)
   asection *seg = NULL;
   sd_chain_struct *space;
 
   asection *seg = NULL;
   sd_chain_struct *space;
 
-  /* load default values */
+  /* Load default values.  */
   spnum = 0;
   sort = 0;
   loadable = TRUE;
   spnum = 0;
   sort = 0;
   loadable = TRUE;
@@ -7161,8 +7103,8 @@ pa_parse_space_stmt (space_name, create_flag)
       print_errors = FALSE;
       ptemp = input_line_pointer + 1;
       /* First see if the space was specified as a number rather than
       print_errors = FALSE;
       ptemp = input_line_pointer + 1;
       /* First see if the space was specified as a number rather than
-         as a name.  According to the PA assembly manual the rest of
-         the line should be ignored.  */
+        as a name.  According to the PA assembly manual the rest of
+        the line should be ignored.  */
       strict = 0;
       pa_parse_number (&ptemp, 0);
       if (pa_number >= 0)
       strict = 0;
       pa_parse_number (&ptemp, 0);
       if (pa_number >= 0)
@@ -7175,39 +7117,38 @@ pa_parse_space_stmt (space_name, create_flag)
          while (!is_end_of_statement ())
            {
              input_line_pointer++;
          while (!is_end_of_statement ())
            {
              input_line_pointer++;
-             name = input_line_pointer;
-             c = get_symbol_end ();
+             c = get_symbol_name (&name);
              if ((strncasecmp (name, "spnum", 5) == 0))
                {
              if ((strncasecmp (name, "spnum", 5) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  input_line_pointer++;
                  spnum = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "sort", 4) == 0))
                {
                  input_line_pointer++;
                  spnum = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "sort", 4) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  input_line_pointer++;
                  sort = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "unloadable", 10) == 0))
                {
                  input_line_pointer++;
                  sort = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "unloadable", 10) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  loadable = FALSE;
                }
              else if ((strncasecmp (name, "notdefined", 10) == 0))
                {
                  loadable = FALSE;
                }
              else if ((strncasecmp (name, "notdefined", 10) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  defined = FALSE;
                }
              else if ((strncasecmp (name, "private", 7) == 0))
                {
                  defined = FALSE;
                }
              else if ((strncasecmp (name, "private", 7) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  private = TRUE;
                }
              else
                {
                  as_bad (_("Invalid .SPACE argument"));
                  private = TRUE;
                }
              else
                {
                  as_bad (_("Invalid .SPACE argument"));
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  if (!is_end_of_statement ())
                    input_line_pointer++;
                }
                  if (!is_end_of_statement ())
                    input_line_pointer++;
                }
@@ -7222,7 +7163,7 @@ pa_parse_space_stmt (space_name, create_flag)
   /* If create_flag is nonzero, then create the new space with
      the attributes computed above.  Else set the values in
      an already existing space -- this can only happen for
   /* If create_flag is nonzero, then create the new space with
      the attributes computed above.  Else set the values in
      an already existing space -- this can only happen for
-     the first occurence of a built-in space.  */
+     the first occurrence of a built-in space.  */
   if (create_flag)
     space = create_new_space (space_name, spnum, loadable, defined,
                              private, sort, seg, 1);
   if (create_flag)
     space = create_new_space (space_name, spnum, loadable, defined,
                              private, sort, seg, 1);
@@ -7245,8 +7186,7 @@ pa_parse_space_stmt (space_name, create_flag)
    given space, creating the new space if necessary.  */
 
 static void
    given space, creating the new space if necessary.  */
 
 static void
-pa_space (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_space (int unused ATTRIBUTE_UNUSED)
 {
   char *name, c, *space_name, *save_s;
   sd_chain_struct *sd_chain;
 {
   char *name, c, *space_name, *save_s;
   sd_chain_struct *sd_chain;
@@ -7259,8 +7199,8 @@ pa_space (unused)
   else
     {
       /* Check for some of the predefined spaces.   FIXME: most of the code
   else
     {
       /* Check for some of the predefined spaces.   FIXME: most of the code
-         below is repeated several times, can we extract the common parts
-         and place them into a subroutine or something similar?  */
+        below is repeated several times, can we extract the common parts
+        and place them into a subroutine or something similar?  */
       /* FIXME Is this (and the next IF stmt) really right?
         What if INPUT_LINE_POINTER points to "$TEXT$FOO"?  */
       if (strncmp (input_line_pointer, "$TEXT$", 6) == 0)
       /* FIXME Is this (and the next IF stmt) really right?
         What if INPUT_LINE_POINTER points to "$TEXT$FOO"?  */
       if (strncmp (input_line_pointer, "$TEXT$", 6) == 0)
@@ -7346,11 +7286,9 @@ pa_space (unused)
       /* Not a number, attempt to create a new space.  */
       print_errors = 1;
       input_line_pointer = save_s;
       /* Not a number, attempt to create a new space.  */
       print_errors = 1;
       input_line_pointer = save_s;
-      name = input_line_pointer;
-      c = get_symbol_end ();
-      space_name = xmalloc (strlen (name) + 1);
-      strcpy (space_name, name);
-      *input_line_pointer = c;
+      c = get_symbol_name (&name);
+      space_name = xstrdup (name);
+      (void) restore_line_pointer (c);
 
       sd_chain = pa_parse_space_stmt (space_name, 1);
       current_space = sd_chain;
 
       sd_chain = pa_parse_space_stmt (space_name, 1);
       current_space = sd_chain;
@@ -7365,16 +7303,14 @@ pa_space (unused)
 /* Switch to a new space.  (I think).  FIXME.  */
 
 static void
 /* Switch to a new space.  (I think).  FIXME.  */
 
 static void
-pa_spnum (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_spnum (int unused ATTRIBUTE_UNUSED)
 {
   char *name;
   char c;
   char *p;
   sd_chain_struct *space;
 
 {
   char *name;
   char c;
   char *p;
   sd_chain_struct *space;
 
-  name = input_line_pointer;
-  c = get_symbol_end ();
+  c = get_symbol_name (&name);
   space = is_defined_space (name);
   if (space)
     {
   space = is_defined_space (name);
   if (space)
     {
@@ -7384,7 +7320,7 @@ pa_spnum (unused)
   else
     as_warn (_("Undefined space: '%s' Assuming space number = 0."), name);
 
   else
     as_warn (_("Undefined space: '%s' Assuming space number = 0."), name);
 
-  *input_line_pointer = c;
+  (void) restore_line_pointer (c);
   demand_empty_rest_of_line ();
 }
 
   demand_empty_rest_of_line ();
 }
 
@@ -7395,12 +7331,11 @@ pa_spnum (unused)
    they're broken up into subroutines.  */
 
 static void
    they're broken up into subroutines.  */
 
 static void
-pa_subspace (create_new)
-     int create_new;
+pa_subspace (int create_new)
 {
   char *name, *ss_name, c;
 {
   char *name, *ss_name, c;
-  char loadable, code_only, common, dup_common, zero, sort;
-  int i, access, space_index, alignment, quadrant, applicable, flags;
+  char loadable, code_only, comdat, common, dup_common, zero, sort;
+  int i, access_ctr, space_index, alignment, quadrant, applicable, flags;
   sd_chain_struct *space;
   ssd_chain_struct *ssd;
   asection *section;
   sd_chain_struct *space;
   ssd_chain_struct *ssd;
   asection *section;
@@ -7415,16 +7350,15 @@ pa_subspace (create_new)
     }
   else
     {
     }
   else
     {
-      name = input_line_pointer;
-      c = get_symbol_end ();
-      ss_name = xmalloc (strlen (name) + 1);
-      strcpy (ss_name, name);
-      *input_line_pointer = c;
+      c = get_symbol_name (&name);
+      ss_name = xstrdup (name);
+      (void) restore_line_pointer (c);
 
       /* Load default values.  */
       sort = 0;
 
       /* Load default values.  */
       sort = 0;
-      access = 0x7f;
+      access_ctr = 0x7f;
       loadable = 1;
       loadable = 1;
+      comdat = 0;
       common = 0;
       dup_common = 0;
       code_only = 0;
       common = 0;
       dup_common = 0;
       code_only = 0;
@@ -7439,7 +7373,7 @@ pa_subspace (create_new)
       else
        ssd = is_defined_subspace (ss_name);
       /* Allow user to override the builtin attributes of subspaces.  But
       else
        ssd = is_defined_subspace (ss_name);
       /* Allow user to override the builtin attributes of subspaces.  But
-         only allow the attributes to be changed once!  */
+        only allow the attributes to be changed once!  */
       if (ssd && SUBSPACE_DEFINED (ssd))
        {
          subseg_set (ssd->ssd_seg, ssd->ssd_subseg);
       if (ssd && SUBSPACE_DEFINED (ssd))
        {
          subseg_set (ssd->ssd_seg, ssd->ssd_subseg);
@@ -7459,6 +7393,7 @@ pa_subspace (create_new)
              if (strcasecmp (pa_def_subspaces[i].name, ss_name) == 0)
                {
                  loadable = pa_def_subspaces[i].loadable;
              if (strcasecmp (pa_def_subspaces[i].name, ss_name) == 0)
                {
                  loadable = pa_def_subspaces[i].loadable;
+                 comdat = pa_def_subspaces[i].comdat;
                  common = pa_def_subspaces[i].common;
                  dup_common = pa_def_subspaces[i].dup_common;
                  code_only = pa_def_subspaces[i].code_only;
                  common = pa_def_subspaces[i].common;
                  dup_common = pa_def_subspaces[i].dup_common;
                  code_only = pa_def_subspaces[i].code_only;
@@ -7466,7 +7401,7 @@ pa_subspace (create_new)
                  space_index = pa_def_subspaces[i].space_index;
                  alignment = pa_def_subspaces[i].alignment;
                  quadrant = pa_def_subspaces[i].quadrant;
                  space_index = pa_def_subspaces[i].space_index;
                  alignment = pa_def_subspaces[i].alignment;
                  quadrant = pa_def_subspaces[i].quadrant;
-                 access = pa_def_subspaces[i].access;
+                 access_ctr = pa_def_subspaces[i].access;
                  sort = pa_def_subspaces[i].sort;
                  break;
                }
                  sort = pa_def_subspaces[i].sort;
                  break;
                }
@@ -7475,26 +7410,25 @@ pa_subspace (create_new)
        }
 
       /* We should be working with a new subspace now.  Fill in
        }
 
       /* We should be working with a new subspace now.  Fill in
-         any information as specified by the user.  */
+        any information as specified by the user.  */
       if (!is_end_of_statement ())
        {
          input_line_pointer++;
          while (!is_end_of_statement ())
            {
       if (!is_end_of_statement ())
        {
          input_line_pointer++;
          while (!is_end_of_statement ())
            {
-             name = input_line_pointer;
-             c = get_symbol_end ();
+             c = get_symbol_name (&name);
              if ((strncasecmp (name, "quad", 4) == 0))
                {
              if ((strncasecmp (name, "quad", 4) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  input_line_pointer++;
                  quadrant = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "align", 5) == 0))
                {
                  input_line_pointer++;
                  quadrant = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "align", 5) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  input_line_pointer++;
                  alignment = get_absolute_expression ();
                  input_line_pointer++;
                  alignment = get_absolute_expression ();
-                 if (log2 (alignment) == -1)
+                 if (exact_log2 (alignment) == -1)
                    {
                      as_bad (_("Alignment must be a power of 2"));
                      alignment = 1;
                    {
                      as_bad (_("Alignment must be a power of 2"));
                      alignment = 1;
@@ -7502,60 +7436,75 @@ pa_subspace (create_new)
                }
              else if ((strncasecmp (name, "access", 6) == 0))
                {
                }
              else if ((strncasecmp (name, "access", 6) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  input_line_pointer++;
                  input_line_pointer++;
-                 access = get_absolute_expression ();
+                 access_ctr = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "sort", 4) == 0))
                {
                }
              else if ((strncasecmp (name, "sort", 4) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  input_line_pointer++;
                  sort = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "code_only", 9) == 0))
                {
                  input_line_pointer++;
                  sort = get_absolute_expression ();
                }
              else if ((strncasecmp (name, "code_only", 9) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  code_only = 1;
                }
              else if ((strncasecmp (name, "unloadable", 10) == 0))
                {
                  code_only = 1;
                }
              else if ((strncasecmp (name, "unloadable", 10) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  loadable = 0;
                }
                  loadable = 0;
                }
+             else if ((strncasecmp (name, "comdat", 6) == 0))
+               {
+                 (void) restore_line_pointer (c);
+                 comdat = 1;
+               }
              else if ((strncasecmp (name, "common", 6) == 0))
                {
              else if ((strncasecmp (name, "common", 6) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  common = 1;
                }
              else if ((strncasecmp (name, "dup_comm", 8) == 0))
                {
                  common = 1;
                }
              else if ((strncasecmp (name, "dup_comm", 8) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  dup_common = 1;
                }
              else if ((strncasecmp (name, "zero", 4) == 0))
                {
                  dup_common = 1;
                }
              else if ((strncasecmp (name, "zero", 4) == 0))
                {
-                 *input_line_pointer = c;
+                 (void) restore_line_pointer (c);
                  zero = 1;
                }
              else if ((strncasecmp (name, "first", 5) == 0))
                as_bad (_("FIRST not supported as a .SUBSPACE argument"));
              else
                as_bad (_("Invalid .SUBSPACE argument"));
                  zero = 1;
                }
              else if ((strncasecmp (name, "first", 5) == 0))
                as_bad (_("FIRST not supported as a .SUBSPACE argument"));
              else
                as_bad (_("Invalid .SUBSPACE argument"));
+
              if (!is_end_of_statement ())
                input_line_pointer++;
            }
        }
 
       /* Compute a reasonable set of BFD flags based on the information
              if (!is_end_of_statement ())
                input_line_pointer++;
            }
        }
 
       /* Compute a reasonable set of BFD flags based on the information
-         in the .subspace directive.  */
+        in the .subspace directive.  */
       applicable = bfd_applicable_section_flags (stdoutput);
       flags = 0;
       if (loadable)
        flags |= (SEC_ALLOC | SEC_LOAD);
       if (code_only)
        flags |= SEC_CODE;
       applicable = bfd_applicable_section_flags (stdoutput);
       flags = 0;
       if (loadable)
        flags |= (SEC_ALLOC | SEC_LOAD);
       if (code_only)
        flags |= SEC_CODE;
-      if (common || dup_common)
-       flags |= SEC_IS_COMMON;
+
+      /* These flags are used to implement various flavors of initialized
+        common.  The SOM linker discards duplicate subspaces when they
+        have the same "key" symbol name.  This support is more like
+        GNU linkonce than BFD common.  Further, pc-relative relocations
+        are converted to section relative relocations in BFD common
+        sections.  This complicates the handling of relocations in
+        common sections containing text and isn't currently supported
+        correctly in the SOM BFD backend.  */
+      if (comdat || common || dup_common)
+       flags |= SEC_LINK_ONCE;
 
       flags |= SEC_RELOC | SEC_HAS_CONTENTS;
 
 
       flags |= SEC_RELOC | SEC_HAS_CONTENTS;
 
@@ -7566,12 +7515,12 @@ pa_subspace (create_new)
       applicable &= flags;
 
       /* If this is an existing subspace, then we want to use the
       applicable &= flags;
 
       /* If this is an existing subspace, then we want to use the
-         segment already associated with the subspace.
+        segment already associated with the subspace.
 
 
-         FIXME NOW!  ELF BFD doesn't appear to be ready to deal with
-         lots of sections.  It might be a problem in the PA ELF
-         code, I do not know yet.  For now avoid creating anything
-         but the "standard" sections for ELF.  */
+        FIXME NOW!  ELF BFD doesn't appear to be ready to deal with
+        lots of sections.  It might be a problem in the PA ELF
+        code, I do not know yet.  For now avoid creating anything
+        but the "standard" sections for ELF.  */
       if (create_new)
        section = subseg_force_new (ss_name, 0);
       else if (ssd)
       if (create_new)
        section = subseg_force_new (ss_name, 0);
       else if (ssd)
@@ -7586,27 +7535,27 @@ pa_subspace (create_new)
       bfd_set_section_flags (stdoutput, section, applicable);
 
       /* Record any alignment request for this section.  */
       bfd_set_section_flags (stdoutput, section, applicable);
 
       /* Record any alignment request for this section.  */
-      record_alignment (section, log2 (alignment));
+      record_alignment (section, exact_log2 (alignment));
 
       /* Set the starting offset for this section.  */
       bfd_set_section_vma (stdoutput, section,
                           pa_subspace_start (space, quadrant));
 
       /* Now that all the flags are set, update an existing subspace,
 
       /* Set the starting offset for this section.  */
       bfd_set_section_vma (stdoutput, section,
                           pa_subspace_start (space, quadrant));
 
       /* Now that all the flags are set, update an existing subspace,
-         or create a new one.  */
+        or create a new one.  */
       if (ssd)
 
        current_subspace = update_subspace (space, ss_name, loadable,
       if (ssd)
 
        current_subspace = update_subspace (space, ss_name, loadable,
-                                           code_only, common, dup_common,
-                                           sort, zero, access, space_index,
-                                           alignment, quadrant,
+                                           code_only, comdat, common,
+                                           dup_common, sort, zero, access_ctr,
+                                           space_index, alignment, quadrant,
                                            section);
       else
        current_subspace = create_new_subspace (space, ss_name, loadable,
                                            section);
       else
        current_subspace = create_new_subspace (space, ss_name, loadable,
-                                               code_only, common,
+                                               code_only, comdat, common,
                                                dup_common, zero, sort,
                                                dup_common, zero, sort,
-                                               access, space_index,
-                                             alignment, quadrant, section);
+                                               access_ctr, space_index,
+                                               alignment, quadrant, section);
 
       demand_empty_rest_of_line ();
       current_subspace->ssd_seg = section;
 
       demand_empty_rest_of_line ();
       current_subspace->ssd_seg = section;
@@ -7618,7 +7567,7 @@ pa_subspace (create_new)
 /* Create default space and subspace dictionaries.  */
 
 static void
 /* Create default space and subspace dictionaries.  */
 
 static void
-pa_spaces_begin ()
+pa_spaces_begin (void)
 {
   int i;
 
 {
   int i;
 
@@ -7628,7 +7577,7 @@ pa_spaces_begin ()
   i = 0;
   while (pa_def_spaces[i].name)
     {
   i = 0;
   while (pa_def_spaces[i].name)
     {
-      char *name;
+      const char *name;
 
       /* Pick the right name to use for the new section.  */
       name = pa_def_spaces[i].name;
 
       /* Pick the right name to use for the new section.  */
       name = pa_def_spaces[i].name;
@@ -7644,13 +7593,13 @@ pa_spaces_begin ()
   i = 0;
   while (pa_def_subspaces[i].name)
     {
   i = 0;
   while (pa_def_subspaces[i].name)
     {
-      char *name;
+      const char *name;
       int applicable, subsegment;
       asection *segment = NULL;
       sd_chain_struct *space;
 
       /* Pick the right name for the new section and pick the right
       int applicable, subsegment;
       asection *segment = NULL;
       sd_chain_struct *space;
 
       /* Pick the right name for the new section and pick the right
-         subsegment number.  */
+        subsegment number.  */
       name = pa_def_subspaces[i].name;
       subsegment = 0;
 
       name = pa_def_subspaces[i].name;
       subsegment = 0;
 
@@ -7658,7 +7607,7 @@ pa_spaces_begin ()
       segment = subseg_new (name, subsegment);
 
       /* For SOM we want to replace the standard .text, .data, and .bss
       segment = subseg_new (name, subsegment);
 
       /* For SOM we want to replace the standard .text, .data, and .bss
-         sections with our own.   We also want to set BFD flags for
+        sections with our own.   We also want to set BFD flags for
         all the built-in subspaces.  */
       if (!strcmp (pa_def_subspaces[i].name, "$CODE$"))
        {
         all the built-in subspaces.  */
       if (!strcmp (pa_def_subspaces[i].name, "$CODE$"))
        {
@@ -7727,6 +7676,7 @@ pa_spaces_begin ()
       create_new_subspace (space, name,
                           pa_def_subspaces[i].loadable,
                           pa_def_subspaces[i].code_only,
       create_new_subspace (space, name,
                           pa_def_subspaces[i].loadable,
                           pa_def_subspaces[i].code_only,
+                          pa_def_subspaces[i].comdat,
                           pa_def_subspaces[i].common,
                           pa_def_subspaces[i].dup_common,
                           pa_def_subspaces[i].zero,
                           pa_def_subspaces[i].common,
                           pa_def_subspaces[i].dup_common,
                           pa_def_subspaces[i].zero,
@@ -7744,26 +7694,19 @@ pa_spaces_begin ()
    by the given parameters.  */
 
 static sd_chain_struct *
    by the given parameters.  */
 
 static sd_chain_struct *
-create_new_space (name, spnum, loadable, defined, private,
-                 sort, seg, user_defined)
-     char *name;
-     int spnum;
-     int loadable;
-     int defined;
-     int private;
-     int sort;
-     asection *seg;
-     int user_defined;
+create_new_space (const char *name,
+                 int spnum,
+                 int loadable ATTRIBUTE_UNUSED,
+                 int defined,
+                 int private,
+                 int sort,
+                 asection *seg,
+                 int user_defined)
 {
   sd_chain_struct *chain_entry;
 
 {
   sd_chain_struct *chain_entry;
 
-  chain_entry = (sd_chain_struct *) xmalloc (sizeof (sd_chain_struct));
-  if (!chain_entry)
-    as_fatal (_("Out of memory: could not allocate new space chain entry: %s\n"),
-             name);
-
-  SPACE_NAME (chain_entry) = (char *) xmalloc (strlen (name) + 1);
-  strcpy (SPACE_NAME (chain_entry), name);
+  chain_entry = XNEW (sd_chain_struct);
+  SPACE_NAME (chain_entry) = xstrdup (name);
   SPACE_DEFINED (chain_entry) = defined;
   SPACE_USER_DEFINED (chain_entry) = user_defined;
   SPACE_SPNUM (chain_entry) = spnum;
   SPACE_DEFINED (chain_entry) = defined;
   SPACE_USER_DEFINED (chain_entry) = user_defined;
   SPACE_SPNUM (chain_entry) = spnum;
@@ -7794,7 +7737,7 @@ create_new_space (name, spnum, loadable, defined, private,
        }
 
       /* At this point we've found the correct place to add the new
        }
 
       /* At this point we've found the correct place to add the new
-         entry.  So add it and update the linked lists as appropriate.  */
+        entry.  So add it and update the linked lists as appropriate.  */
       if (prev_chain_pointer)
        {
          chain_entry->sd_next = chain_pointer;
       if (prev_chain_pointer)
        {
          chain_entry->sd_next = chain_pointer;
@@ -7828,27 +7771,25 @@ create_new_space (name, spnum, loadable, defined, private,
    order as defined by the SORT entries.  */
 
 static ssd_chain_struct *
    order as defined by the SORT entries.  */
 
 static ssd_chain_struct *
-create_new_subspace (space, name, loadable, code_only, common,
-                    dup_common, is_zero, sort, access, space_index,
-                    alignment, quadrant, seg)
-     sd_chain_struct *space;
-     char *name;
-     int loadable, code_only, common, dup_common, is_zero;
-     int sort;
-     int access;
-     int space_index;
-     int alignment;
-     int quadrant;
-     asection *seg;
+create_new_subspace (sd_chain_struct *space,
+                    const char *name,
+                    int loadable ATTRIBUTE_UNUSED,
+                    int code_only ATTRIBUTE_UNUSED,
+                    int comdat,
+                    int common,
+                    int dup_common,
+                    int is_zero ATTRIBUTE_UNUSED,
+                    int sort,
+                    int access_ctr,
+                    int space_index ATTRIBUTE_UNUSED,
+                    int alignment ATTRIBUTE_UNUSED,
+                    int quadrant,
+                    asection *seg)
 {
   ssd_chain_struct *chain_entry;
 
 {
   ssd_chain_struct *chain_entry;
 
-  chain_entry = (ssd_chain_struct *) xmalloc (sizeof (ssd_chain_struct));
-  if (!chain_entry)
-    as_fatal (_("Out of memory: could not allocate new subspace chain entry: %s\n"), name);
-
-  SUBSPACE_NAME (chain_entry) = (char *) xmalloc (strlen (name) + 1);
-  strcpy (SUBSPACE_NAME (chain_entry), name);
+  chain_entry = XNEW (ssd_chain_struct);
+  SUBSPACE_NAME (chain_entry) = xstrdup (name);
 
   /* Initialize subspace_defined.  When we hit a .subspace directive
      we'll set it to 1 which "locks-in" the subspace attributes.  */
 
   /* Initialize subspace_defined.  When we hit a .subspace directive
      we'll set it to 1 which "locks-in" the subspace attributes.  */
@@ -7876,7 +7817,7 @@ create_new_subspace (space, name, loadable, code_only, common,
        }
 
       /* Now we have somewhere to put the new entry.  Insert it and update
        }
 
       /* Now we have somewhere to put the new entry.  Insert it and update
-         the links.  */
+        the links.  */
       if (prev_chain_pointer)
        {
          chain_entry->ssd_next = chain_pointer;
       if (prev_chain_pointer)
        {
          chain_entry->ssd_next = chain_pointer;
@@ -7890,8 +7831,8 @@ create_new_subspace (space, name, loadable, code_only, common,
     }
 
 #ifdef obj_set_subsection_attributes
     }
 
 #ifdef obj_set_subsection_attributes
-  obj_set_subsection_attributes (seg, space->sd_seg, access,
-                                sort, quadrant);
+  obj_set_subsection_attributes (seg, space->sd_seg, access_ctr, sort,
+                                quadrant, comdat, common, dup_common);
 #endif
 
   return chain_entry;
 #endif
 
   return chain_entry;
@@ -7901,29 +7842,28 @@ create_new_subspace (space, name, loadable, code_only, common,
    various arguments.   Return the modified subspace chain entry.  */
 
 static ssd_chain_struct *
    various arguments.   Return the modified subspace chain entry.  */
 
 static ssd_chain_struct *
-update_subspace (space, name, loadable, code_only, common, dup_common, sort,
-                zero, access, space_index, alignment, quadrant, section)
-     sd_chain_struct *space;
-     char *name;
-     int loadable;
-     int code_only;
-     int common;
-     int dup_common;
-     int zero;
-     int sort;
-     int access;
-     int space_index;
-     int alignment;
-     int quadrant;
-     asection *section;
+update_subspace (sd_chain_struct *space,
+                char *name,
+                int loadable ATTRIBUTE_UNUSED,
+                int code_only ATTRIBUTE_UNUSED,
+                int comdat,
+                int common,
+                int dup_common,
+                int sort,
+                int zero ATTRIBUTE_UNUSED,
+                int access_ctr,
+                int space_index ATTRIBUTE_UNUSED,
+                int alignment ATTRIBUTE_UNUSED,
+                int quadrant,
+                asection *section)
 {
   ssd_chain_struct *chain_entry;
 
   chain_entry = is_defined_subspace (name);
 
 #ifdef obj_set_subsection_attributes
 {
   ssd_chain_struct *chain_entry;
 
   chain_entry = is_defined_subspace (name);
 
 #ifdef obj_set_subsection_attributes
-  obj_set_subsection_attributes (section, space->sd_seg, access,
-                                sort, quadrant);
+  obj_set_subsection_attributes (section, space->sd_seg, access_ctr, sort,
+                                quadrant, comdat, common, dup_common);
 #endif
 
   return chain_entry;
 #endif
 
   return chain_entry;
@@ -7933,18 +7873,15 @@ update_subspace (space, name, loadable, code_only, common, dup_common, sort,
    NULL if no such space exists.  */
 
 static sd_chain_struct *
    NULL if no such space exists.  */
 
 static sd_chain_struct *
-is_defined_space (name)
-     char *name;
+is_defined_space (const char *name)
 {
   sd_chain_struct *chain_pointer;
 
   for (chain_pointer = space_dict_root;
        chain_pointer;
        chain_pointer = chain_pointer->sd_next)
 {
   sd_chain_struct *chain_pointer;
 
   for (chain_pointer = space_dict_root;
        chain_pointer;
        chain_pointer = chain_pointer->sd_next)
-    {
-      if (strcmp (SPACE_NAME (chain_pointer), name) == 0)
-       return chain_pointer;
-    }
+    if (strcmp (SPACE_NAME (chain_pointer), name) == 0)
+      return chain_pointer;
 
   /* No mapping from segment to space was found.  Return NULL.  */
   return NULL;
 
   /* No mapping from segment to space was found.  Return NULL.  */
   return NULL;
@@ -7957,8 +7894,7 @@ is_defined_space (name)
    so a linear exhaustive search is OK here.  */
 
 static sd_chain_struct *
    so a linear exhaustive search is OK here.  */
 
 static sd_chain_struct *
-pa_segment_to_space (seg)
-     asection *seg;
+pa_segment_to_space (asection *seg)
 {
   sd_chain_struct *space_chain;
 
 {
   sd_chain_struct *space_chain;
 
@@ -7966,25 +7902,27 @@ pa_segment_to_space (seg)
   for (space_chain = space_dict_root;
        space_chain;
        space_chain = space_chain->sd_next)
   for (space_chain = space_dict_root;
        space_chain;
        space_chain = space_chain->sd_next)
-    {
-      if (space_chain->sd_seg == seg)
-       return space_chain;
-    }
+    if (space_chain->sd_seg == seg)
+      return space_chain;
 
   /* Mapping was not found.  Return NULL.  */
   return NULL;
 }
 
 
   /* Mapping was not found.  Return NULL.  */
   return NULL;
 }
 
-/* Return the space chain entry for the subspace with the name NAME or
-   NULL if no such subspace exists.
+/* Return the first space chain entry for the subspace with the name
+   NAME or NULL if no such subspace exists.
+
+   When there are multiple subspaces with the same name, switching to
+   the first (i.e., default) subspace is preferable in most situations.
+   For example, it wouldn't be desirable to merge COMDAT data with non
+   COMDAT data.
 
    Uses a linear search through all the spaces and subspaces, this may
    not be appropriate if we ever being placing each function in its
    own subspace.  */
 
 static ssd_chain_struct *
 
    Uses a linear search through all the spaces and subspaces, this may
    not be appropriate if we ever being placing each function in its
    own subspace.  */
 
 static ssd_chain_struct *
-is_defined_subspace (name)
-     char *name;
+is_defined_subspace (const char *name)
 {
   sd_chain_struct *space_chain;
   ssd_chain_struct *subspace_chain;
 {
   sd_chain_struct *space_chain;
   ssd_chain_struct *subspace_chain;
@@ -8014,9 +7952,7 @@ is_defined_subspace (name)
    to become more efficient.  */
 
 static ssd_chain_struct *
    to become more efficient.  */
 
 static ssd_chain_struct *
-pa_subsegment_to_subspace (seg, subseg)
-     asection *seg;
-     subsegT subseg;
+pa_subsegment_to_subspace (asection *seg, subsegT subseg)
 {
   sd_chain_struct *space_chain;
   ssd_chain_struct *subspace_chain;
 {
   sd_chain_struct *space_chain;
   ssd_chain_struct *subspace_chain;
@@ -8048,8 +7984,7 @@ pa_subsegment_to_subspace (seg, subseg)
    that was found or NULL on failure.  */
 
 static sd_chain_struct *
    that was found or NULL on failure.  */
 
 static sd_chain_struct *
-pa_find_space_by_number (number)
-     int number;
+pa_find_space_by_number (int number)
 {
   sd_chain_struct *space_chain;
 
 {
   sd_chain_struct *space_chain;
 
@@ -8069,9 +8004,7 @@ pa_find_space_by_number (number)
    address is unknown then return zero.  */
 
 static unsigned int
    address is unknown then return zero.  */
 
 static unsigned int
-pa_subspace_start (space, quadrant)
-     sd_chain_struct *space;
-     int quadrant;
+pa_subspace_start (sd_chain_struct *space, int quadrant)
 {
   /* FIXME.  Assumes everyone puts read/write data at 0x4000000, this
      is not correct for the PA OSF1 port.  */
 {
   /* FIXME.  Assumes everyone puts read/write data at 0x4000000, this
      is not correct for the PA OSF1 port.  */
@@ -8083,24 +8016,13 @@ pa_subspace_start (space, quadrant)
     return 0;
   return 0;
 }
     return 0;
   return 0;
 }
-
-/* FIXME.  Needs documentation.  */
-static int
-pa_next_subseg (space)
-     sd_chain_struct *space;
-{
-
-  space->sd_last_subseg++;
-  return space->sd_last_subseg;
-}
 #endif
 
 /* Helper function for pa_stringer.  Used to find the end of
    a string.  */
 
 static unsigned int
 #endif
 
 /* Helper function for pa_stringer.  Used to find the end of
    a string.  */
 
 static unsigned int
-pa_stringer_aux (s)
-     char *s;
+pa_stringer_aux (char *s)
 {
   unsigned int c = *s & CHAR_MASK;
 
 {
   unsigned int c = *s & CHAR_MASK;
 
@@ -8118,8 +8040,7 @@ pa_stringer_aux (s)
 /* Handle a .STRING type pseudo-op.  */
 
 static void
 /* Handle a .STRING type pseudo-op.  */
 
 static void
-pa_stringer (append_zero)
-     int append_zero;
+pa_stringer (int append_zero)
 {
   char *s, num_buf[4];
   unsigned int c;
 {
   char *s, num_buf[4];
   unsigned int c;
@@ -8156,11 +8077,11 @@ pa_stringer (append_zero)
                s++;
                for (num_digit = 0, number = 0, dg = *s;
                     num_digit < 2
                s++;
                for (num_digit = 0, number = 0, dg = *s;
                     num_digit < 2
-                    && (isdigit (dg) || (dg >= 'a' && dg <= 'f')
+                    && (ISDIGIT (dg) || (dg >= 'a' && dg <= 'f')
                         || (dg >= 'A' && dg <= 'F'));
                     num_digit++)
                  {
                         || (dg >= 'A' && dg <= 'F'));
                     num_digit++)
                  {
-                   if (isdigit (dg))
+                   if (ISDIGIT (dg))
                      number = number * 16 + dg - '0';
                    else if (dg >= 'a' && dg <= 'f')
                      number = number * 16 + dg - 'a' + 10;
                      number = number * 16 + dg - '0';
                    else if (dg >= 'a' && dg <= 'f')
                      number = number * 16 + dg - 'a' + 10;
@@ -8193,15 +8114,14 @@ pa_stringer (append_zero)
            }
        }
     }
            }
        }
     }
-  stringer (append_zero);
+  stringer (8 + append_zero);
   pa_undefine_label ();
 }
 
 /* Handle a .VERSION pseudo-op.  */
 
 static void
   pa_undefine_label ();
 }
 
 /* Handle a .VERSION pseudo-op.  */
 
 static void
-pa_version (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_version (int unused ATTRIBUTE_UNUSED)
 {
   obj_version (0);
   pa_undefine_label ();
 {
   obj_version (0);
   pa_undefine_label ();
@@ -8212,8 +8132,7 @@ pa_version (unused)
 /* Handle a .COMPILER pseudo-op.  */
 
 static void
 /* Handle a .COMPILER pseudo-op.  */
 
 static void
-pa_compiler (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_compiler (int unused ATTRIBUTE_UNUSED)
 {
   obj_som_compiler (0);
   pa_undefine_label ();
 {
   obj_som_compiler (0);
   pa_undefine_label ();
@@ -8224,8 +8143,7 @@ pa_compiler (unused)
 /* Handle a .COPYRIGHT pseudo-op.  */
 
 static void
 /* Handle a .COPYRIGHT pseudo-op.  */
 
 static void
-pa_copyright (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_copyright (int unused ATTRIBUTE_UNUSED)
 {
   obj_copyright (0);
   pa_undefine_label ();
 {
   obj_copyright (0);
   pa_undefine_label ();
@@ -8235,8 +8153,7 @@ pa_copyright (unused)
    the latest space label.  */
 
 static void
    the latest space label.  */
 
 static void
-pa_cons (nbytes)
-     int nbytes;
+pa_cons (int nbytes)
 {
   cons (nbytes);
   pa_undefine_label ();
 {
   cons (nbytes);
   pa_undefine_label ();
@@ -8245,8 +8162,7 @@ pa_cons (nbytes)
 /* Like float_cons, but we need to undefine our label.  */
 
 static void
 /* Like float_cons, but we need to undefine our label.  */
 
 static void
-pa_float_cons (float_type)
-     int float_type;
+pa_float_cons (int float_type)
 {
   float_cons (float_type);
   pa_undefine_label ();
 {
   float_cons (float_type);
   pa_undefine_label ();
@@ -8255,8 +8171,7 @@ pa_float_cons (float_type)
 /* Like s_fill, but delete our label when finished.  */
 
 static void
 /* Like s_fill, but delete our label when finished.  */
 
 static void
-pa_fill (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_fill (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -8270,8 +8185,7 @@ pa_fill (unused)
 /* Like lcomm, but delete our label when finished.  */
 
 static void
 /* Like lcomm, but delete our label when finished.  */
 
 static void
-pa_lcomm (needs_align)
-     int needs_align;
+pa_lcomm (int needs_align)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -8285,8 +8199,7 @@ pa_lcomm (needs_align)
 /* Like lsym, but delete our label when finished.  */
 
 static void
 /* Like lsym, but delete our label when finished.  */
 
 static void
-pa_lsym (unused)
-     int unused ATTRIBUTE_UNUSED;
+pa_lsym (int unused ATTRIBUTE_UNUSED)
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
 {
 #ifdef OBJ_SOM
   /* We must have a valid space and subspace.  */
@@ -8297,6 +8210,79 @@ pa_lsym (unused)
   pa_undefine_label ();
 }
 
   pa_undefine_label ();
 }
 
+/* This function is called once, at assembler startup time.  It should
+   set up all the tables, etc. that the MD part of the assembler will need.  */
+
+void
+md_begin (void)
+{
+  const char *retval = NULL;
+  int lose = 0;
+  unsigned int i = 0;
+
+  last_call_info = NULL;
+  call_info_root = NULL;
+
+  /* Set the default machine type.  */
+  if (!bfd_set_arch_mach (stdoutput, bfd_arch_hppa, DEFAULT_LEVEL))
+    as_warn (_("could not set architecture and machine"));
+
+  /* Folding of text and data segments fails miserably on the PA.
+     Warn user and disable "-R" option.  */
+  if (flag_readonly_data_in_text)
+    {
+      as_warn (_("-R option not supported on this target."));
+      flag_readonly_data_in_text = 0;
+    }
+
+#ifdef OBJ_SOM
+  pa_spaces_begin ();
+#endif
+
+  op_hash = hash_new ();
+
+  while (i < NUMOPCODES)
+    {
+      const char *name = pa_opcodes[i].name;
+
+      retval = hash_insert (op_hash, name, (struct pa_opcode *) &pa_opcodes[i]);
+      if (retval != NULL && *retval != '\0')
+       {
+         as_fatal (_("Internal error: can't hash `%s': %s\n"), name, retval);
+         lose = 1;
+       }
+
+      do
+       {
+         if ((pa_opcodes[i].match & pa_opcodes[i].mask)
+             != pa_opcodes[i].match)
+           {
+             fprintf (stderr, _("internal error: losing opcode: `%s' \"%s\"\n"),
+                      pa_opcodes[i].name, pa_opcodes[i].args);
+             lose = 1;
+           }
+         ++i;
+       }
+      while (i < NUMOPCODES && !strcmp (pa_opcodes[i].name, name));
+    }
+
+  if (lose)
+    as_fatal (_("Broken assembler.  No assembly attempted."));
+
+#ifdef OBJ_SOM
+  /* SOM will change text_section.  To make sure we never put
+     anything into the old one switch to the new one now.  */
+  subseg_set (text_section, 0);
+#endif
+
+#ifdef OBJ_SOM
+  dummy_symbol = symbol_find_or_make ("L$dummy");
+  S_SET_SEGMENT (dummy_symbol, text_section);
+  /* Force the symbol to be converted to a real symbol.  */
+  symbol_get_bfdsym (dummy_symbol)->flags |= BSF_KEEP;
+#endif
+}
+
 /* On the PA relocations which involve function symbols must not be
    adjusted.  This so that the linker can know when/how to create argument
    relocation stubs for indirect calls and calls to static functions.
 /* On the PA relocations which involve function symbols must not be
    adjusted.  This so that the linker can know when/how to create argument
    relocation stubs for indirect calls and calls to static functions.
@@ -8315,37 +8301,59 @@ pa_lsym (unused)
    any fixup which creates entries in the DLT (eg they use "T" field
    selectors).
 
    any fixup which creates entries in the DLT (eg they use "T" field
    selectors).
 
-   Reject reductions involving symbols with external scope; such
-   reductions make life a living hell for object file editors.
-
-   FIXME.  Also reject R_HPPA relocations which are 32bits wide in
-   the code space.  The SOM BFD backend doesn't know how to pull the
-   right bits out of an instruction.  */
+   ??? Reject reductions involving symbols with external scope; such
+   reductions make life a living hell for object file editors.  */
 
 int
 
 int
-hppa_fix_adjustable (fixp)
-     fixS *fixp;
+hppa_fix_adjustable (fixS *fixp)
 {
 {
+#ifdef OBJ_ELF
+  reloc_type code;
+#endif
   struct hppa_fix_struct *hppa_fix;
 
   hppa_fix = (struct hppa_fix_struct *) fixp->tc_fix_data;
 
   struct hppa_fix_struct *hppa_fix;
 
   hppa_fix = (struct hppa_fix_struct *) fixp->tc_fix_data;
 
-#ifdef OBJ_SOM
-  /* Reject reductions of symbols in 32bit relocs.  */
-  if (fixp->fx_r_type == R_HPPA && hppa_fix->fx_r_format == 32)
-    return 0;
-#endif
-
 #ifdef OBJ_ELF
 #ifdef OBJ_ELF
-  if (fixp->fx_r_type == (int) R_PARISC_GNU_VTINHERIT
-      || fixp->fx_r_type ==  (int) R_PARISC_GNU_VTENTRY)
-    return 0;
+  /* LR/RR selectors are implicitly used for a number of different relocation
+     types.  We must ensure that none of these types are adjusted (see below)
+     even if they occur with a different selector.  */
+  code = elf_hppa_reloc_final_type (stdoutput, fixp->fx_r_type,
+                                   hppa_fix->fx_r_format,
+                                   hppa_fix->fx_r_field);
+
+  switch (code)
+    {
+    /* Relocation types which use e_lrsel.  */
+    case R_PARISC_DIR21L:
+    case R_PARISC_DLTREL21L:
+    case R_PARISC_DPREL21L:
+    case R_PARISC_PLTOFF21L:
+
+    /* Relocation types which use e_rrsel.  */
+    case R_PARISC_DIR14R:
+    case R_PARISC_DIR14DR:
+    case R_PARISC_DIR14WR:
+    case R_PARISC_DIR17R:
+    case R_PARISC_DLTREL14R:
+    case R_PARISC_DLTREL14DR:
+    case R_PARISC_DLTREL14WR:
+    case R_PARISC_DPREL14R:
+    case R_PARISC_DPREL14DR:
+    case R_PARISC_DPREL14WR:
+    case R_PARISC_PLTOFF14R:
+    case R_PARISC_PLTOFF14DR:
+    case R_PARISC_PLTOFF14WR:
+
+    /* Other types that we reject for reduction.  */
+    case R_PARISC_GNU_VTENTRY:
+    case R_PARISC_GNU_VTINHERIT:
+      return 0;
+    default:
+      break;
+    }
 #endif
 
 #endif
 
-  if (fixp->fx_addsy && (S_IS_EXTERNAL (fixp->fx_addsy)
-                        || S_IS_WEAK (fixp->fx_addsy)))
-    return 0;
-
   /* Reject reductions of symbols in sym1-sym2 expressions when
      the fixup will occur in a CODE subspace.
 
   /* Reject reductions of symbols in sym1-sym2 expressions when
      the fixup will occur in a CODE subspace.
 
@@ -8355,11 +8363,7 @@ hppa_fix_adjustable (fixp)
   if (fixp->fx_addsy
       && fixp->fx_subsy
       && (hppa_fix->segment->flags & SEC_CODE))
   if (fixp->fx_addsy
       && fixp->fx_subsy
       && (hppa_fix->segment->flags & SEC_CODE))
-    {
-      /* Apparently sy_used_in_reloc never gets set for sub symbols.  */
-      symbol_mark_used_in_reloc (fixp->fx_subsy);
-      return 0;
-    }
+    return 0;
 
   /* We can't adjust any relocs that use LR% and RR% field selectors.
 
 
   /* We can't adjust any relocs that use LR% and RR% field selectors.
 
@@ -8383,7 +8387,7 @@ hppa_fix_adjustable (fixp)
      .                 RR%sect+4092 == (R%sect)+4092
      .                 RR%sect+4096 == (R%sect)-4096
      and the last address loses because rounding the addend to 8k
      .                 RR%sect+4092 == (R%sect)+4092
      .                 RR%sect+4096 == (R%sect)-4096
      and the last address loses because rounding the addend to 8k
-     mutiples takes us up to 8192 with an offset of -4096.
+     multiples takes us up to 8192 with an offset of -4096.
 
      In cases where the LR% expression is identical to the RR% one we
      will never have a problem, but is so happens that gcc rounds
 
      In cases where the LR% expression is identical to the RR% one we
      will never have a problem, but is so happens that gcc rounds
@@ -8422,8 +8426,7 @@ hppa_fix_adjustable (fixp)
    within GAS.  */
 
 int
    within GAS.  */
 
 int
-hppa_force_relocation (fixp)
-     struct fix *fixp;
+hppa_force_relocation (struct fix *fixp)
 {
   struct hppa_fix_struct *hppa_fixp;
 
 {
   struct hppa_fix_struct *hppa_fixp;
 
@@ -8445,15 +8448,15 @@ hppa_force_relocation (fixp)
     return 1;
 #endif
 
     return 1;
 #endif
 
-  assert (fixp->fx_addsy != NULL);
+  gas_assert (fixp->fx_addsy != NULL);
 
   /* Ensure we emit a relocation for global symbols so that dynamic
      linking works.  */
 
   /* Ensure we emit a relocation for global symbols so that dynamic
      linking works.  */
-  if (S_IS_EXTERNAL (fixp->fx_addsy) || S_IS_WEAK (fixp->fx_addsy))
+  if (S_FORCE_RELOC (fixp->fx_addsy, 1))
     return 1;
 
   /* It is necessary to force PC-relative calls/jumps to have a relocation
     return 1;
 
   /* It is necessary to force PC-relative calls/jumps to have a relocation
-     entry if they're going to need either a argument relocation or long
+     entry if they're going to need either an argument relocation or long
      call stub.  */
   if (fixp->fx_pcrel
       && arg_reloc_stub_needed (symbol_arg_reloc_info (fixp->fx_addsy),
      call stub.  */
   if (fixp->fx_pcrel
       && arg_reloc_stub_needed (symbol_arg_reloc_info (fixp->fx_addsy),
@@ -8463,15 +8466,27 @@ hppa_force_relocation (fixp)
   /* Now check to see if we're going to need a long-branch stub.  */
   if (fixp->fx_r_type == (int) R_HPPA_PCREL_CALL)
     {
   /* Now check to see if we're going to need a long-branch stub.  */
   if (fixp->fx_r_type == (int) R_HPPA_PCREL_CALL)
     {
-      valueT distance;
+      long pc = md_pcrel_from (fixp);
+      valueT distance, min_stub_distance;
 
 
-      distance = (fixp->fx_offset + S_GET_VALUE (fixp->fx_addsy)
-                 - md_pcrel_from (fixp) - 8);
-      if (distance + 8388608 >= 16777216
-         || (hppa_fixp->fx_r_format == 17 && distance + 262144 >= 524288)
-#ifdef OBJ_ELF
-         || (hppa_fixp->fx_r_format == 12 && distance + 8192 >= 16384)
+      distance = fixp->fx_offset + S_GET_VALUE (fixp->fx_addsy) - pc - 8;
+
+      /* Distance to the closest possible stub.  This will detect most
+        but not all circumstances where a stub will not work.  */
+      min_stub_distance = pc + 16;
+#ifdef OBJ_SOM
+      if (last_call_info != NULL)
+       min_stub_distance -= S_GET_VALUE (last_call_info->start_symbol);
 #endif
 #endif
+
+      if ((distance + 8388608 >= 16777216
+          && min_stub_distance <= 8388608)
+         || (hppa_fixp->fx_r_format == 17
+             && distance + 262144 >= 524288
+             && min_stub_distance <= 262144)
+         || (hppa_fixp->fx_r_format == 12
+             && distance + 8192 >= 16384
+             && min_stub_distance <= 8192)
          )
        return 1;
     }
          )
        return 1;
     }
@@ -8485,75 +8500,13 @@ hppa_force_relocation (fixp)
 
 /* Now for some ELF specific code.  FIXME.  */
 #ifdef OBJ_ELF
 
 /* Now for some ELF specific code.  FIXME.  */
 #ifdef OBJ_ELF
-/* Mark the end of a function so that it's possible to compute
-   the size of the function in hppa_elf_final_processing.  */
-
-static void
-hppa_elf_mark_end_of_function ()
-{
-  /* ELF does not have EXIT relocations.  All we do is create a
-     temporary symbol marking the end of the function.  */
-  char *name;
-
-  if (last_call_info == NULL || last_call_info->start_symbol == NULL)
-    {
-      /* We have already warned about a missing label,
-        or other problems.  */
-      return;
-    }
-
-  name = (char *) xmalloc (strlen ("L$\001end_")
-                          + strlen (S_GET_NAME (last_call_info->start_symbol))
-                          + 1);
-  if (name)
-    {
-      symbolS *symbolP;
-
-      strcpy (name, "L$\001end_");
-      strcat (name, S_GET_NAME (last_call_info->start_symbol));
-
-      /* If we have a .exit followed by a .procend, then the
-        symbol will have already been defined.  */
-      symbolP = symbol_find (name);
-      if (symbolP)
-       {
-         /* The symbol has already been defined!  This can
-            happen if we have a .exit followed by a .procend.
-
-            This is *not* an error.  All we want to do is free
-            the memory we just allocated for the name and continue.  */
-         xfree (name);
-       }
-      else
-       {
-         /* symbol value should be the offset of the
-            last instruction of the function */
-         symbolP = symbol_new (name, now_seg, (valueT) (frag_now_fix () - 4),
-                               frag_now);
-
-         assert (symbolP);
-         S_CLEAR_EXTERNAL (symbolP);
-         symbol_table_insert (symbolP);
-       }
-
-      if (symbolP)
-       last_call_info->end_symbol = symbolP;
-      else
-       as_bad (_("Symbol '%s' could not be created."), name);
-
-    }
-  else
-    as_bad (_("No memory for symbol name."));
-
-}
-
 /* For ELF, this function serves one purpose:  to setup the st_size
    field of STT_FUNC symbols.  To do this, we need to scan the
    call_info structure list, determining st_size in by taking the
    difference in the address of the beginning/end marker symbols.  */
 
 void
 /* For ELF, this function serves one purpose:  to setup the st_size
    field of STT_FUNC symbols.  To do this, we need to scan the
    call_info structure list, determining st_size in by taking the
    difference in the address of the beginning/end marker symbols.  */
 
 void
-elf_hppa_final_processing ()
+elf_hppa_final_processing (void)
 {
   struct call_info *call_info_pointer;
 
 {
   struct call_info *call_info_pointer;
 
@@ -8571,8 +8524,7 @@ elf_hppa_final_processing ()
 }
 
 static void
 }
 
 static void
-pa_vtable_entry (ignore)
-     int ignore ATTRIBUTE_UNUSED;
+pa_vtable_entry (int ignore ATTRIBUTE_UNUSED)
 {
   struct fix *new_fix;
 
 {
   struct fix *new_fix;
 
@@ -8580,8 +8532,8 @@ pa_vtable_entry (ignore)
 
   if (new_fix)
     {
 
   if (new_fix)
     {
-      struct hppa_fix_struct *hppa_fix = (struct hppa_fix_struct *)
-       obstack_alloc (&notes, sizeof (struct hppa_fix_struct));
+      struct hppa_fix_struct * hppa_fix = XOBNEW (&notes, struct hppa_fix_struct);
+
       hppa_fix->fx_r_type = R_HPPA;
       hppa_fix->fx_r_field = e_fsel;
       hppa_fix->fx_r_format = 32;
       hppa_fix->fx_r_type = R_HPPA;
       hppa_fix->fx_r_field = e_fsel;
       hppa_fix->fx_r_format = 32;
@@ -8593,8 +8545,7 @@ pa_vtable_entry (ignore)
 }
 
 static void
 }
 
 static void
-pa_vtable_inherit (ignore)
-     int ignore ATTRIBUTE_UNUSED;
+pa_vtable_inherit (int ignore ATTRIBUTE_UNUSED)
 {
   struct fix *new_fix;
 
 {
   struct fix *new_fix;
 
@@ -8602,8 +8553,8 @@ pa_vtable_inherit (ignore)
 
   if (new_fix)
     {
 
   if (new_fix)
     {
-      struct hppa_fix_struct *hppa_fix = (struct hppa_fix_struct *)
-       obstack_alloc (&notes, sizeof (struct hppa_fix_struct));
+      struct hppa_fix_struct * hppa_fix = XOBNEW (&notes, struct hppa_fix_struct);
+
       hppa_fix->fx_r_type = R_HPPA;
       hppa_fix->fx_r_field = e_fsel;
       hppa_fix->fx_r_format = 32;
       hppa_fix->fx_r_type = R_HPPA;
       hppa_fix->fx_r_field = e_fsel;
       hppa_fix->fx_r_format = 32;
@@ -8614,3 +8565,145 @@ pa_vtable_inherit (ignore)
     }
 }
 #endif
     }
 }
 #endif
+
+/* Table of pseudo ops for the PA.  FIXME -- how many of these
+   are now redundant with the overall GAS and the object file
+   dependent tables?  */
+const pseudo_typeS md_pseudo_table[] =
+{
+  /* align pseudo-ops on the PA specify the actual alignment requested,
+     not the log2 of the requested alignment.  */
+#ifdef OBJ_SOM
+  {"align", pa_align, 8},
+#endif
+#ifdef OBJ_ELF
+  {"align", s_align_bytes, 8},
+#endif
+  {"begin_brtab", pa_brtab, 1},
+  {"begin_try", pa_try, 1},
+  {"block", pa_block, 1},
+  {"blockz", pa_block, 0},
+  {"byte", pa_cons, 1},
+  {"call", pa_call, 0},
+  {"callinfo", pa_callinfo, 0},
+#if defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD))
+  {"code", obj_elf_text, 0},
+#else
+  {"code", pa_text, 0},
+  {"comm", pa_comm, 0},
+#endif
+#ifdef OBJ_SOM
+  {"compiler", pa_compiler, 0},
+#endif
+  {"copyright", pa_copyright, 0},
+#if !(defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD)))
+  {"data", pa_data, 0},
+#endif
+  {"double", pa_float_cons, 'd'},
+  {"dword", pa_cons, 8},
+  {"end", pa_end, 0},
+  {"end_brtab", pa_brtab, 0},
+#if !(defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD)))
+  {"end_try", pa_try, 0},
+#endif
+  {"enter", pa_enter, 0},
+  {"entry", pa_entry, 0},
+  {"equ", pa_equ, 0},
+  {"exit", pa_exit, 0},
+  {"export", pa_export, 0},
+  {"fill", pa_fill, 0},
+  {"float", pa_float_cons, 'f'},
+  {"half", pa_cons, 2},
+  {"import", pa_import, 0},
+  {"int", pa_cons, 4},
+  {"label", pa_label, 0},
+  {"lcomm", pa_lcomm, 0},
+  {"leave", pa_leave, 0},
+  {"level", pa_level, 0},
+  {"long", pa_cons, 4},
+  {"lsym", pa_lsym, 0},
+#ifdef OBJ_SOM
+  {"nsubspa", pa_subspace, 1},
+#endif
+  {"octa", pa_cons, 16},
+  {"org", pa_origin, 0},
+  {"origin", pa_origin, 0},
+  {"param", pa_param, 0},
+  {"proc", pa_proc, 0},
+  {"procend", pa_procend, 0},
+  {"quad", pa_cons, 8},
+  {"reg", pa_equ, 1},
+  {"short", pa_cons, 2},
+  {"single", pa_float_cons, 'f'},
+#ifdef OBJ_SOM
+  {"space", pa_space, 0},
+  {"spnum", pa_spnum, 0},
+#endif
+  {"string", pa_stringer, 0},
+  {"stringz", pa_stringer, 1},
+#ifdef OBJ_SOM
+  {"subspa", pa_subspace, 0},
+#endif
+#if !(defined (OBJ_ELF) && (defined (TE_LINUX) || defined (TE_NetBSD)))
+  {"text", pa_text, 0},
+#endif
+  {"version", pa_version, 0},
+#ifdef OBJ_ELF
+  {"vtable_entry", pa_vtable_entry, 0},
+  {"vtable_inherit", pa_vtable_inherit, 0},
+#endif
+  {"word", pa_cons, 4},
+  {NULL, 0, 0}
+};
+
+#ifdef OBJ_ELF
+void
+hppa_cfi_frame_initial_instructions (void)
+{
+  cfi_add_CFA_def_cfa (30, 0);
+}
+
+int
+hppa_regname_to_dw2regnum (char *regname)
+{
+  unsigned int regnum = -1;
+  unsigned int i;
+  const char *p;
+  char *q;
+  static struct { const char *name; int dw2regnum; } regnames[] =
+    {
+      { "sp", 30 }, { "rp", 2 },
+    };
+
+  for (i = 0; i < ARRAY_SIZE (regnames); ++i)
+    if (strcmp (regnames[i].name, regname) == 0)
+      return regnames[i].dw2regnum;
+
+  if (regname[0] == 'r')
+    {
+      p = regname + 1;
+      regnum = strtoul (p, &q, 10);
+      if (p == q || *q || regnum >= 32)
+       return -1;
+    }
+  else if (regname[0] == 'f' && regname[1] == 'r')
+    {
+      p = regname + 2;
+      regnum = strtoul (p, &q, 10);
+#if TARGET_ARCH_SIZE == 64
+      if (p == q || *q || regnum <= 4 || regnum >= 32)
+       return -1;
+      regnum += 32 - 4;
+#else
+      if (p == q
+         || (*q  && ((*q != 'L' && *q != 'R') || *(q + 1)))
+         || regnum <= 4 || regnum >= 32)
+       return -1;
+      regnum = (regnum - 4) * 2 + 32;
+      if (*q == 'R')
+       regnum++;
+#endif
+    }
+  return regnum;
+}
+#endif
This page took 0.193689 seconds and 4 git commands to generate.