2004-11-05 Tomer Levi <Tomer.Levi@nsc.com>
[external/binutils.git] / gas / config / tc-crx.c
1 /* tc-crx.c -- Assembler code for the CRX CPU core.
2    Copyright 2004 Free Software Foundation, Inc.
3
4    Contributed by Tomer Levi, NSC, Israel.
5    Originally written for GAS 2.12 by Tomer Levi, NSC, Israel.
6    Updates, BFDizing, GNUifying and ELF support by Tomer Levi.
7
8    This file is part of GAS, the GNU Assembler.
9
10    GAS is free software; you can redistribute it and/or modify
11    it under the terms of the GNU General Public License as published by
12    the Free Software Foundation; either version 2, or (at your option)
13    any later version.
14
15    GAS is distributed in the hope that it will be useful,
16    but WITHOUT ANY WARRANTY; without even the implied warranty of
17    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18    GNU General Public License for more details.
19
20    You should have received a copy of the GNU General Public License
21    along with GAS; see the file COPYING.  If not, write to the
22    Free Software Foundation, 59 Temple Place - Suite 330, Boston,
23    MA 02111-1307, USA.  */
24
25 #include "as.h"
26 #include "safe-ctype.h"
27 #include "dwarf2dbg.h"
28 #include "opcode/crx.h"
29 #include "elf/crx.h"
30
31 #include <limits.h>
32
33 /* Word is considered here as a 16-bit unsigned short int.  */
34 #define WORD_SIZE   16
35 #define WORD_SHIFT  16
36
37 /* Register is 4-bit size.  */
38 #define REG_SIZE   4
39
40 /* Maximum size of a single instruction (in words).  */
41 #define INSN_MAX_SIZE   3
42
43 /* Maximum bits which may be set in a `mask16' operand.  */
44 #define MAX_REGS_IN_MASK16  8
45
46 /* Escape to 16-bit immediate.  */
47 #define ESC_16  0xE
48 /* Escape to 32-bit immediate.  */
49 #define ESC_32  0xF
50
51 /* Utility macros for string comparison.  */
52 #define streq(a, b)           (strcmp (a, b) == 0)
53 #define strneq(a, b, c)       (strncmp (a, b, c) == 0)
54
55 /* A mask to set n_bits starting from offset offs.  */
56 #define SET_BITS_MASK(offs,n_bits)    ((((1 << (n_bits)) - 1) << (offs)))
57 /* A mask to clear n_bits starting from offset offs.  */
58 #define CLEAR_BITS_MASK(offs,n_bits)  (~(((1 << (n_bits)) - 1) << (offs)))
59
60 /* Get the argument type for each operand of a given instruction.  */
61 #define GET_ACTUAL_TYPE                                           \
62   for (i = 0; i < insn->nargs; i++)                               \
63     atyp_act[i] = getarg_type (instruction->operands[i].op_type)
64
65 /* Get the size (in bits) for each operand of a given instruction.  */
66 #define GET_ACTUAL_SIZE                                           \
67   for (i = 0; i < insn->nargs; i++)                               \
68     bits_act[i] = getbits (instruction->operands[i].op_type)
69
70 /* Non-zero if OP is instruction with no operands.  */
71 #define NO_OPERANDS_INST(OP)                      \
72   (streq (OP, "di") || streq (OP, "nop")          \
73    || streq (OP, "retx") || streq (OP, "ei")      \
74    || streq (OP, "wait") || streq (OP, "eiwait"))
75
76 /* Print a number NUM, shifted by SHIFT bytes, into a location
77    pointed by index BYTE of array 'output_opcode'.  */
78 #define CRX_PRINT(BYTE, NUM, SHIFT)   output_opcode[BYTE] |= (NUM << SHIFT)
79
80 /* Opcode mnemonics hash table.  */
81 static struct hash_control *crx_inst_hash;
82 /* CRX registers hash table.  */
83 static struct hash_control *reg_hash;
84 /* CRX coprocessor registers hash table.  */
85 static struct hash_control *copreg_hash;
86 /* Current instruction we're assembling.  */
87 const inst *instruction;
88
89 /* Initialize global variables.  */
90 long output_opcode[2];
91 /* Nonzero means a relocatable symbol.  */
92 int relocatable;
93 /* Nonzero means a constant's bit-size was already set.  */
94 int size_was_set;
95 /* Nonzero means a negative constant.  */
96 int signflag;
97 /* Nonzero means a CST4 instruction.  */
98 int cst4flag;
99 /* A copy of the original instruction (used in error messages).  */
100 char ins_parse[MAX_INST_LEN];
101 /* Holds the current processed argument number.  */
102 int cur_arg_num;
103
104 /* Generic assembler global variables which must be defined by all targets.  */
105
106 /* Characters which always start a comment.  */
107 const char comment_chars[] = "#";
108
109 /* Characters which start a comment at the beginning of a line.  */
110 const char line_comment_chars[] = "#";
111
112 /* This array holds machine specific line separator characters.  */
113 const char line_separator_chars[] = ";";
114
115 /* Chars that can be used to separate mant from exp in floating point nums.  */
116 const char EXP_CHARS[] = "eE";
117
118 /* Chars that mean this number is a floating point constant as in 0f12.456  */
119 const char FLT_CHARS[] = "f'";
120
121 /* Target-specific multicharacter options, not const-declared at usage.  */
122 const char *md_shortopts = "";
123 struct option md_longopts[] =
124 {
125   {NULL, no_argument, NULL, 0}
126 };
127 size_t md_longopts_size = sizeof (md_longopts);
128
129 /* This table describes all the machine specific pseudo-ops
130    the assembler has to support.  The fields are:
131    *** Pseudo-op name without dot.
132    *** Function to call to execute this pseudo-op.
133    *** Integer arg to pass to the function.  */
134
135 const pseudo_typeS md_pseudo_table[] =
136 {
137   /* In CRX machine, align is in bytes (not a ptwo boundary).  */
138   {"align", s_align_bytes, 0},
139   {0, 0, 0}
140 };
141
142 const relax_typeS md_relax_table[] =
143 {
144   /* bCC  */
145   {0xfa, -0x100, 2, 1},                 /*  8 */
146   {0xfffe, -0x10000, 4, 2},             /* 16 */
147   {0xfffffffe, -0xfffffffe, 6, 0},      /* 32 */
148
149   /* bal  */
150   {0xfffe, -0x10000, 4, 4},             /* 16 */
151   {0xfffffffe, -0xfffffffe, 6, 0},      /* 32 */
152
153   /* cmpbr  */
154   {0xfe, -0x100, 4, 6},                 /*  8 */
155   {0xfffffe, -0x1000000, 6, 0}          /* 24 */
156 };
157
158 static void    reset_vars               (char *, ins *);
159 static reg     get_register             (char *);
160 static copreg  get_copregister          (char *);
161 static void    set_operand_size         (ins *);
162 static argtype getarg_type              (operand_type);
163 static int     getbits                  (operand_type);
164 static int     get_flags                (operand_type);
165 static int     get_number_of_operands   (void);
166 static void    parse_operand            (char *, ins *);
167 static int     gettrap                  (char *);
168 static void    handle_LoadStor          (char *);
169 static int     get_cinv_parameters      (char *);
170 static unsigned long getconstant        (unsigned long, int);
171 static int     getreg_image             (reg);
172 static void    parse_operands           (ins *, char *);
173 static void    parse_insn               (ins *, char *);
174 static void    print_operand            (int, int, argument *);
175 static void    print_constant           (int, int, argument *);
176 static int     exponent2scale           (int);
177 static void    mask_const               (unsigned long *, int);
178 static void    mask_reg                 (int, unsigned short *);
179 static int     process_label_constant   (char *, ins *);
180 static void    set_operand              (char *, ins *);
181 static char *  preprocess_reglist       (char *, int *);
182 static int     assemble_insn            (char *, ins *);
183 static void    print_insn               (ins *);
184
185 /* Return the bit size for a given operand.  */
186
187 static int
188 getbits (operand_type op)
189 {
190   if (op < MAX_OPRD)
191     return crx_optab[op].bit_size;
192   else
193     return 0;
194 }
195
196 /* Return the argument type of a given operand.  */
197
198 static argtype
199 getarg_type (operand_type op)
200 {
201   if (op < MAX_OPRD)
202     return crx_optab[op].arg_type;
203   else
204     return nullargs;
205 }
206
207 /* Return the flags of a given operand.  */
208
209 static int
210 get_flags (operand_type op)
211 {
212   if (op < MAX_OPRD)
213     return crx_optab[op].flags;
214   else
215     return 0;
216 }
217
218 /* Get the core processor register 'reg_name'.  */
219
220 static reg
221 get_register (char *reg_name)
222 {
223   const reg_entry *reg;
224
225   reg = (const reg_entry *) hash_find (reg_hash, reg_name);
226
227   if (reg != NULL)
228     return reg->value.reg_val;
229   else
230     return nullregister;
231 }
232
233 /* Get the coprocessor register 'copreg_name'.  */
234
235 static copreg
236 get_copregister (char *copreg_name)
237 {
238   const reg_entry *copreg;
239
240   copreg = (const reg_entry *) hash_find (copreg_hash, copreg_name);
241
242   if (copreg != NULL)
243     return copreg->value.copreg_val;
244   else
245     return nullcopregister;
246 }
247
248 /* Mask a constant to the number of bits it is to be mapped to.  */
249
250 static void
251 mask_const (unsigned long int *t, int size)
252 {
253   *t &= (((LONGLONG)1 << size) - 1);
254 }
255
256 /* Round up a section size to the appropriate boundary.  */
257
258 valueT
259 md_section_align (segT seg, valueT val)
260 {
261   /* Round .text section to a multiple of 2.  */
262   if (seg == text_section)
263     return (val + 1) & ~1;
264   return val;
265 }
266
267 /* Parse an operand that is machine-specific (remove '*').  */
268
269 void
270 md_operand (expressionS * exp)
271 {
272   char c = *input_line_pointer;
273
274   switch (c)
275     {
276     case '*':
277       input_line_pointer++;
278       expression (exp);
279       break;
280     default:
281       break;
282     }
283 }
284
285 /* Reset global variables before parsing a new instruction.  */
286
287 static void
288 reset_vars (char *op, ins *crx_ins)
289 {
290   unsigned int i;
291
292   cur_arg_num = relocatable = size_was_set = signflag = cst4flag = 0;
293   memset (& output_opcode, '\0', sizeof (output_opcode));
294
295   /* Memset the 'signflag' field in every argument.  */
296   for (i = 0; i < MAX_OPERANDS; i++)
297     crx_ins->arg[i].signflag = 0;
298
299   /* Save a copy of the original OP (used in error messages).  */
300   strcpy (ins_parse, op);
301 }
302
303 /* This macro decides whether a particular reloc is an entry in a
304    switch table.  It is used when relaxing, because the linker needs
305    to know about all such entries so that it can adjust them if
306    necessary.  */
307
308 #define SWITCH_TABLE(fix)                                 \
309   (   (fix)->fx_addsy != NULL                             \
310    && (fix)->fx_subsy != NULL                             \
311    && S_GET_SEGMENT ((fix)->fx_addsy) ==                  \
312       S_GET_SEGMENT ((fix)->fx_subsy)                     \
313    && S_GET_SEGMENT (fix->fx_addsy) != undefined_section  \
314    && (   (fix)->fx_r_type == BFD_RELOC_CRX_NUM8          \
315        || (fix)->fx_r_type == BFD_RELOC_CRX_NUM16         \
316        || (fix)->fx_r_type == BFD_RELOC_CRX_NUM32))
317
318 /* See whether we need to force a relocation into the output file.
319    This is used to force out switch and PC relative relocations when
320    relaxing.  */
321
322 int
323 crx_force_relocation (fixS *fix)
324 {
325   if (generic_force_reloc (fix) || SWITCH_TABLE (fix))
326     return 1;
327
328   return 0;
329 }
330
331 /* Generate a relocation entry for a fixup.  */
332
333 arelent *
334 tc_gen_reloc (asection *section ATTRIBUTE_UNUSED, fixS * fixP)
335 {
336   arelent * reloc;
337
338   reloc = xmalloc (sizeof (arelent));
339   reloc->sym_ptr_ptr  = xmalloc (sizeof (asymbol *));
340   *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixP->fx_addsy);
341   reloc->address = fixP->fx_frag->fr_address + fixP->fx_where;
342   reloc->addend = fixP->fx_offset;
343
344   if (fixP->fx_subsy != NULL)
345     {
346       if (SWITCH_TABLE (fixP))
347         {
348           /* Keep the current difference in the addend.  */
349           reloc->addend = (S_GET_VALUE (fixP->fx_addsy)
350                            - S_GET_VALUE (fixP->fx_subsy) + fixP->fx_offset);
351
352           switch (fixP->fx_r_type)
353             {
354             case BFD_RELOC_CRX_NUM8:
355               fixP->fx_r_type = BFD_RELOC_CRX_SWITCH8;
356               break;
357             case BFD_RELOC_CRX_NUM16:
358               fixP->fx_r_type = BFD_RELOC_CRX_SWITCH16;
359               break;
360             case BFD_RELOC_CRX_NUM32:
361               fixP->fx_r_type = BFD_RELOC_CRX_SWITCH32;
362               break;
363             default:
364               abort ();
365               break;
366             }
367         }
368       else
369         {
370           /* We only resolve difference expressions in the same section.  */
371           as_bad_where (fixP->fx_file, fixP->fx_line,
372                         _("can't resolve `%s' {%s section} - `%s' {%s section}"),
373                         fixP->fx_addsy ? S_GET_NAME (fixP->fx_addsy) : "0",
374                         segment_name (fixP->fx_addsy
375                                       ? S_GET_SEGMENT (fixP->fx_addsy)
376                                       : absolute_section),
377                         S_GET_NAME (fixP->fx_subsy),
378                         segment_name (S_GET_SEGMENT (fixP->fx_addsy)));
379         }
380     }
381
382   assert ((int) fixP->fx_r_type > 0);
383   reloc->howto = bfd_reloc_type_lookup (stdoutput, fixP->fx_r_type);
384
385   if (reloc->howto == (reloc_howto_type *) NULL)
386     {
387       as_bad_where (fixP->fx_file, fixP->fx_line,
388                     _("internal error: reloc %d (`%s') not supported by object file format"),
389                     fixP->fx_r_type,
390                     bfd_get_reloc_code_name (fixP->fx_r_type));
391       return NULL;
392     }
393   assert (!fixP->fx_pcrel == !reloc->howto->pc_relative);
394
395   return reloc;
396 }
397
398 /* Prepare machine-dependent frags for relaxation.  */
399
400 int
401 md_estimate_size_before_relax (fragS *fragp, asection *seg)
402 {
403   /* If symbol is undefined or located in a different section,
404      select the largest supported relocation.  */
405   relax_substateT subtype;
406   relax_substateT rlx_state[] = {0, 2,
407                                  3, 4,
408                                  5, 6};
409
410   for (subtype = 0; subtype < ARRAY_SIZE (rlx_state); subtype += 2)
411     {
412       if (fragp->fr_subtype == rlx_state[subtype]
413           && (!S_IS_DEFINED (fragp->fr_symbol)
414               || seg != S_GET_SEGMENT (fragp->fr_symbol)))
415         {
416           fragp->fr_subtype = rlx_state[subtype + 1];
417           break;
418         }
419     }
420
421   if (fragp->fr_subtype >= ARRAY_SIZE (md_relax_table))
422     abort ();
423
424   return md_relax_table[fragp->fr_subtype].rlx_length;
425 }
426
427 void
428 md_convert_frag (bfd *abfd ATTRIBUTE_UNUSED, asection *sec, fragS *fragP)
429 {
430   /* 'opcode' points to the start of the instruction, whether
431      we need to change the instruction's fixed encoding.  */
432   char *opcode = fragP->fr_literal + fragP->fr_fix;
433   bfd_reloc_code_real_type reloc;
434
435   subseg_change (sec, 0);
436
437   switch (fragP->fr_subtype)
438     {
439     case 0:
440       reloc = BFD_RELOC_CRX_REL8;
441       break;
442     case 1:
443       *opcode = 0x7e;
444       reloc = BFD_RELOC_CRX_REL16;
445       break;
446     case 2:
447       *opcode = 0x7f;
448       reloc = BFD_RELOC_CRX_REL32;
449       break;
450     case 3:
451       reloc = BFD_RELOC_CRX_REL16;
452       break;
453     case 4:
454       *++opcode = 0x31;
455       reloc = BFD_RELOC_CRX_REL32;
456       break;
457     case 5:
458       reloc = BFD_RELOC_CRX_REL8_CMP;
459       break;
460     case 6:
461       *++opcode = 0x31;
462       reloc = BFD_RELOC_CRX_REL24;
463       break;
464     default:
465       abort ();
466       break;
467     }
468
469     fix_new (fragP, fragP->fr_fix,
470              bfd_get_reloc_size (bfd_reloc_type_lookup (stdoutput, reloc)),
471              fragP->fr_symbol, fragP->fr_offset, 1, reloc);
472     fragP->fr_var = 0;
473     fragP->fr_fix += md_relax_table[fragP->fr_subtype].rlx_length;
474 }
475
476 /* Process machine-dependent command line options.  Called once for
477    each option on the command line that the machine-independent part of
478    GAS does not understand.  */
479
480 int
481 md_parse_option (int c ATTRIBUTE_UNUSED, char *arg ATTRIBUTE_UNUSED)
482 {
483   return 0;
484 }
485
486 /* Machine-dependent usage-output.  */
487
488 void
489 md_show_usage (FILE *stream ATTRIBUTE_UNUSED)
490 {
491   return;
492 }
493
494 /* Turn a string in input_line_pointer into a floating point constant
495    of type TYPE, and store the appropriate bytes in *LITP.  The number
496    of LITTLENUMS emitted is stored in *SIZEP.  An error message is
497    returned, or NULL on OK.  */
498
499 char *
500 md_atof (int type, char *litP, int *sizeP)
501 {
502   int prec;
503   LITTLENUM_TYPE words[4];
504   char *t;
505   int i;
506
507   switch (type)
508     {
509     case 'f':
510       prec = 2;
511       break;
512
513     case 'd':
514       prec = 4;
515       break;
516
517     default:
518       *sizeP = 0;
519       return _("bad call to md_atof");
520     }
521
522   t = atof_ieee (input_line_pointer, type, words);
523   if (t)
524     input_line_pointer = t;
525
526   *sizeP = prec * 2;
527
528   if (! target_big_endian)
529     {
530       for (i = prec - 1; i >= 0; i--)
531         {
532           md_number_to_chars (litP, (valueT) words[i], 2);
533           litP += 2;
534         }
535     }
536   else
537     {
538       for (i = 0; i < prec; i++)
539         {
540           md_number_to_chars (litP, (valueT) words[i], 2);
541           litP += 2;
542         }
543     }
544
545   return NULL;
546 }
547
548 /* Apply a fixS (fixup of an instruction or data that we didn't have
549    enough info to complete immediately) to the data in a frag.
550    Since linkrelax is nonzero and TC_LINKRELAX_FIXUP is defined to disable
551    relaxation of debug sections, this function is called only when
552    fixuping relocations of debug sections.  */
553
554 void
555 md_apply_fix3 (fixS *fixP, valueT *valP, segT seg)
556 {
557   valueT val = * valP;
558   char *buf = fixP->fx_frag->fr_literal + fixP->fx_where;
559   fixP->fx_offset = 0;
560
561   switch (fixP->fx_r_type)
562     {
563     case BFD_RELOC_CRX_NUM8:
564       bfd_put_8 (stdoutput, (unsigned char) val, buf);
565       break;
566     case BFD_RELOC_CRX_NUM16:
567       bfd_put_16 (stdoutput, val, buf);
568       break;
569     case BFD_RELOC_CRX_NUM32:
570       bfd_put_32 (stdoutput, val, buf);
571       break;
572     default:
573       /* We shouldn't ever get here because linkrelax is nonzero.  */
574       abort ();
575       break;
576     }
577
578   fixP->fx_done = 0;
579
580   if (fixP->fx_addsy == NULL
581       && fixP->fx_pcrel == 0)
582     fixP->fx_done = 1;
583
584   if (fixP->fx_pcrel == 1
585       && fixP->fx_addsy != NULL
586       && S_GET_SEGMENT (fixP->fx_addsy) == seg)
587     fixP->fx_done = 1;
588 }
589
590 /* The location from which a PC relative jump should be calculated,
591    given a PC relative reloc.  */
592
593 long
594 md_pcrel_from (fixS *fixp)
595 {
596   return fixp->fx_frag->fr_address + fixp->fx_where;
597 }
598
599 /* This function is called once, at assembler startup time.  This should
600    set up all the tables, etc that the MD part of the assembler needs.  */
601
602 void
603 md_begin (void)
604 {
605   const char *hashret = NULL;
606   int i = 0;
607
608   /* Set up a hash table for the instructions.  */
609   if ((crx_inst_hash = hash_new ()) == NULL)
610     as_fatal (_("Virtual memory exhausted"));
611   
612   while (crx_instruction[i].mnemonic != NULL)
613     {
614       const char *mnemonic = crx_instruction[i].mnemonic;
615
616       hashret = hash_insert (crx_inst_hash, mnemonic,
617         (PTR) &crx_instruction[i]);
618
619       if (hashret != NULL && *hashret != '\0')
620         as_fatal (_("Can't hash `%s': %s\n"), crx_instruction[i].mnemonic,
621                   *hashret == 0 ? _("(unknown reason)") : hashret);
622
623       /* Insert unique names into hash table.  The CRX instruction set
624          has many identical opcode names that have different opcodes based
625          on the operands.  This hash table then provides a quick index to
626          the first opcode with a particular name in the opcode table.  */
627       do
628         {
629           ++i;
630         }
631       while (crx_instruction[i].mnemonic != NULL
632              && streq (crx_instruction[i].mnemonic, mnemonic));
633     }
634
635   /* Initialize reg_hash hash table.  */
636   if ((reg_hash = hash_new ()) == NULL)
637     as_fatal (_("Virtual memory exhausted"));
638
639   {
640     const reg_entry *regtab;
641
642     for (regtab = crx_regtab;
643          regtab < (crx_regtab + NUMREGS); regtab++)
644       {
645         hashret = hash_insert (reg_hash, regtab->name, (PTR) regtab);
646         if (hashret)
647           as_fatal (_("Internal Error:  Can't hash %s: %s"),
648                     regtab->name,
649                     hashret);
650       }
651   }
652
653   /* Initialize copreg_hash hash table.  */
654   if ((copreg_hash = hash_new ()) == NULL)
655     as_fatal (_("Virtual memory exhausted"));
656
657   {
658     const reg_entry *copregtab;
659
660     for (copregtab = crx_copregtab; copregtab < (crx_copregtab + NUMCOPREGS);
661          copregtab++)
662       {
663         hashret = hash_insert (copreg_hash, copregtab->name, (PTR) copregtab);
664         if (hashret)
665           as_fatal (_("Internal Error:  Can't hash %s: %s"),
666                     copregtab->name,
667                     hashret);
668       }
669   }
670   /*  Set linkrelax here to avoid fixups in most sections.  */
671   linkrelax = 1;
672 }
673
674 /* Set the number of bits corresponding to a constant -
675    here we check for possible overflow cases.  */
676
677 static void
678 set_operand_size (ins * crx_ins)
679 {
680   int cnt_bits = 0;
681   const cst4_entry *cst4_op;
682   argument *cur_arg = &crx_ins->arg[cur_arg_num];  /* Current argument.  */
683   unsigned long int temp = cur_arg->constant;
684
685   /* If the constant's size was already set - nothing to do.  */
686   if (size_was_set)
687     return;
688
689   /* Already dealt with negative numbers in process_label_constants.  */
690   while (temp > 0)
691     {
692       temp >>= 1;
693       cnt_bits++;
694     }
695
696   /* Arithmetic instructions :
697      16-bit positive signed immediate -->> represent as 32-bit.  */
698   if (IS_INSN_TYPE (ARITH_INS) && !relocatable && !signflag)
699     {
700       if (cnt_bits == 16)
701         {
702           cur_arg->size = 32;
703           return;
704         }
705     }
706   /* Index addressing mode :
707      6-bit positive signed immediate -->> represent as 22-bit.  */
708   if (IS_INSN_TYPE (LD_STOR_INS)
709       || IS_INSN_TYPE (STOR_IMM_INS)
710       || IS_INSN_TYPE (CSTBIT_INS))
711     {
712       if (!signflag && cur_arg->type == arg_idxr)
713         {
714           if (cnt_bits == 6)
715             {
716               cur_arg->size = 22;
717               return;
718             }
719           if (cnt_bits == 22)
720             as_bad (_("Offset out of range in Instruction `%s'"), ins_parse);
721         }
722     }
723   /* load/stor instructions :
724      16-bit positive signed immediate -->> represent as 32-bit.  */
725   if (IS_INSN_TYPE (LD_STOR_INS))
726     {
727       if (!signflag && cur_arg->type == arg_cr)
728         {
729           if (cnt_bits == 16)
730             {
731               cur_arg->size = 32;
732               return;
733             }
734           if (cnt_bits == 32)
735             as_bad (_("Offset out of range in Instruction `%s'"), ins_parse);
736         }
737     }
738   /* Post-increment mode :
739      12-bit positive signed immediate -->> represent as 28-bit.  */
740   if (IS_INSN_TYPE (CSTBIT_INS)
741       || IS_INSN_TYPE (LD_STOR_INS_INC)
742       || IS_INSN_TYPE (STOR_IMM_INS))
743     {
744       if (!signflag && cur_arg->type == arg_cr)
745         {
746           if (cnt_bits == 12)
747             {
748               cur_arg->size = 28;
749               if (IS_INSN_TYPE (LD_STOR_INS_INC))
750                 as_bad (_("Offset out of range in Instruction `%s'"), ins_parse);
751               return;
752             }
753           if (IS_INSN_TYPE (CSTBIT_INS) || IS_INSN_TYPE (STOR_IMM_INS))
754             {
755               if (cnt_bits == 28)
756                 as_bad (_("Offset out of range in Instruction `%s'"), ins_parse);
757             }
758
759         }
760     }
761
762   /* Handle negative cst4 mapping for arithmetic/cmp&br operations.  */
763   if (signflag && !relocatable
764       && ((IS_INSN_TYPE (ARITH_INS) || IS_INSN_TYPE (ARITH_BYTE_INS))
765       || ((IS_INSN_TYPE (CMPBR_INS) && cur_arg_num == 0))))
766     {
767       for (cst4_op = cst4_map; cst4_op < (cst4_map + cst4_maps); cst4_op++)
768         {
769           if (cur_arg->constant == (unsigned int)(-cst4_op->value))
770             {
771               cur_arg->size = 4;
772               cur_arg->constant = cst4_op->binary;
773               cur_arg->signflag = 0;
774               return;
775             }
776         }
777     }
778   /* Because of the cst4 mapping -- -1 and -4 already handled above
779      as well as for relocatable cases.  */
780   if (signflag && IS_INSN_TYPE (ARITH_BYTE_INS))
781     {
782       if (!relocatable)
783         {
784           if (cur_arg->constant <= 0xffff)
785             cur_arg->size = 16;
786           else
787             /* Setting to 18 so that there is no match.  */
788             cur_arg->size = 18;
789         }
790       else
791         cur_arg->size = 16;
792       return;
793     }
794
795   if (signflag && IS_INSN_TYPE (ARITH_INS))
796     {
797       /* For all immediates which can be expressed in less than 16 bits.  */
798       if (cur_arg->constant <= 0xffff && !relocatable)
799         {
800           cur_arg->size = 16;
801           return;
802         }
803       /* Either it is relocatable or not representable in 16 bits.  */
804       if (cur_arg->constant < 0xffffffff || relocatable)
805         {
806           cur_arg->size = 32;
807           return;
808         }
809       cur_arg->size = 33;
810       return;
811     }
812
813   if (signflag && !relocatable)
814     return;
815
816   if (!relocatable)
817     cur_arg->size = cnt_bits;
818
819   /* Checking for Error Conditions.  */
820   if (IS_INSN_TYPE (ARITH_INS) && !signflag)
821     {
822       if (cnt_bits > 32)
823         as_bad (_("Cannot represent Immediate in %d bits in Instruction `%s'"),
824                 cnt_bits, ins_parse);
825     }
826   else if (IS_INSN_TYPE (ARITH_BYTE_INS) && !signflag)
827     {
828       if (cnt_bits > 16)
829         as_bad (_("Cannot represent Immediate in %d bits in Instruction `%s'"),
830                 cnt_bits, ins_parse);
831     }
832 }
833
834 /* Handle the constants immediate/absolute values and
835    Labels (jump targets/Memory locations).  */
836
837 static int
838 process_label_constant (char *str, ins * crx_ins)
839 {
840   char *save;
841   unsigned long int temp, cnt;
842   const cst4_entry *cst4_op;
843   int is_cst4 = 0;
844   int constant_val = 0;
845   argument *cur_arg = &crx_ins->arg[cur_arg_num];  /* Current argument.  */
846
847   save = input_line_pointer;
848   signflag = 0;
849
850   if (str[0] == '-')
851     {
852       signflag = 1;
853       str++;
854     }
855   else if (str[0] == '+')
856     str++;
857
858   input_line_pointer = str;
859
860   expression (&crx_ins->exp);
861
862   switch (crx_ins->exp.X_op)
863     {
864     case O_big:
865     case O_absent:
866       /* Missing or bad expr becomes absolute 0.  */
867       as_bad (_("missing or invalid displacement expression `%s' taken as 0"),
868               str);
869       crx_ins->exp.X_op = O_constant;
870       crx_ins->exp.X_add_number = 0;
871       crx_ins->exp.X_add_symbol = (symbolS *) 0;
872       crx_ins->exp.X_op_symbol = (symbolS *) 0;
873       break;
874
875     case O_constant:
876       cur_arg->constant = constant_val = crx_ins->exp.X_add_number;
877       if ((IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS))
878            && cur_arg_num == 2)
879         {
880           LONGLONG temp64 = 0;
881           char ptr;
882           char temp_str[30];
883           unsigned int jump_value = 0;
884           int BR_MASK = 0, BR_SIZE = 0;
885           temp_str[0] = '\0';
886           if (signflag)
887             {
888               temp_str[0] = '-';
889               temp_str[1] = '\0';
890             }
891           strncat (temp_str, str, strlen (str));
892           temp64 = strtoll (temp_str, (char **) &ptr,0);
893
894           if (temp64 % 2 != 0)
895             as_bad (_("Odd Offset in displacement in Instruction `%s'"),
896                     ins_parse);
897
898           /* Determine the branch size.  */
899           jump_value = (unsigned int)temp64 & 0xFFFFFFFF;
900           if (((jump_value & 0xFFFFFF00) == 0xFFFFFF00)
901               || ((jump_value & 0xFFFFFF00) == 0x0))
902             {
903               BR_MASK = 0xFF;
904               BR_SIZE = 8;
905             }
906           else
907             if (((jump_value & 0xFF000000) == 0xFF000000)
908                 || ((jump_value & 0xFF000000) == 0x0))
909             {
910               BR_MASK = 0xFFFFFF;
911               BR_SIZE = 24;
912             }
913           jump_value = jump_value >> 1;
914           cur_arg->constant = jump_value & BR_MASK;
915           cur_arg->size = BR_SIZE;
916           size_was_set = 1;
917           cur_arg->signflag = signflag;
918           input_line_pointer = save;
919           return crx_ins->exp.X_op;
920         }
921
922       if (IS_INSN_TYPE (BRANCH_INS)
923           || IS_INSN_MNEMONIC ("bal")
924           || IS_INSN_TYPE (DCR_BRANCH_INS))
925         {
926           LONGLONG temp64 = 0;
927           char ptr;
928           char temp_str[30];
929           unsigned int jump_value = 0;
930           int BR_MASK = 0, BR_SIZE = 0;
931
932           temp_str[0] = '\0';
933           if (signflag)
934             {
935               temp_str[0] = '-';
936               temp_str[1] = '\0';
937             }
938           strncat (temp_str, str, strlen (str));
939           temp64 = strtoll (temp_str, (char **) &ptr,0);
940
941           if (temp64 % 2 != 0)
942             as_bad (_("Odd Offset in displacement in Instruction `%s'"),
943             ins_parse);
944
945           /* Determine the branch size.  */
946           jump_value = (unsigned int)temp64 & 0xFFFFFFFF;
947           if (!IS_INSN_MNEMONIC ("bal") && !IS_INSN_TYPE (DCR_BRANCH_INS)
948               && (((jump_value & 0xFFFFFF00) == 0xFFFFFF00)
949                   || ((jump_value & 0xFFFFFF00) == 0x0)))
950             {
951               BR_MASK = 0xFF;
952               BR_SIZE = 8;
953             }
954           else if (((jump_value & 0xFFFF0000) == 0xFFFF0000)
955                    || ((jump_value & 0xFFFF0000) == 0x0))
956             {
957               BR_MASK = 0xFFFF;
958               BR_SIZE = 16;
959             }
960           else
961             {
962               BR_MASK = 0xFFFFFFFF;
963               BR_SIZE = 32;
964             }
965           jump_value = jump_value >> 1;
966           cur_arg->constant = jump_value & BR_MASK;
967           cur_arg->size = BR_SIZE;
968           size_was_set = 1;
969           cur_arg->signflag = signflag;
970           input_line_pointer = save;
971           return crx_ins->exp.X_op;
972         }
973       /* Fix for movd $0xF12344, r0 -- signflag has to be set.  */
974       if (constant_val < 0 && signflag != 1
975           && !IS_INSN_TYPE (LD_STOR_INS) && !IS_INSN_TYPE (LD_STOR_INS_INC)
976           && !IS_INSN_TYPE (CSTBIT_INS) && !IS_INSN_TYPE (STOR_IMM_INS)
977           && !IS_INSN_TYPE (BRANCH_INS) && !IS_INSN_MNEMONIC ("bal"))
978         {
979           cur_arg->constant = ~(cur_arg->constant) + 1;
980           signflag = 1;
981         }
982       /* For load/store instruction when the value is in the offset part.  */
983       if (constant_val < 0 && signflag != 1
984           && (IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (LD_STOR_INS_INC)
985               || IS_INSN_TYPE (CSTBIT_INS) || IS_INSN_TYPE (STOR_IMM_INS)))
986         {
987           if (cur_arg->type == arg_cr || cur_arg->type == arg_idxr)
988             {
989               cur_arg->constant = ~(cur_arg->constant) + 1;
990               signflag = 1;
991             }
992         }
993       if (signflag)
994         {
995           /* Signflag in never set in case of load store instructions
996              Mapping in case of only the arithinsn case.  */
997           if ((cur_arg->constant != 1 && cur_arg->constant != 4)
998              || (!IS_INSN_TYPE (ARITH_INS)
999                  && !IS_INSN_TYPE (ARITH_BYTE_INS)
1000                  && !IS_INSN_TYPE (CMPBR_INS)))
1001             {
1002               /* Counting the number of bits required to represent
1003                  the constant.  */
1004               cnt = 0;
1005               temp = cur_arg->constant - 1;
1006               while (temp > 0)
1007                 {
1008                   temp >>= 1;
1009                   cnt++;
1010                 }
1011               cur_arg->size = cnt + 1;
1012               cur_arg->constant = ~(cur_arg->constant) + 1;
1013               if (IS_INSN_TYPE (ARITH_INS) || IS_INSN_TYPE (ARITH_BYTE_INS))
1014                 {
1015                   char ptr;
1016                   LONGLONG temp64;
1017
1018                   temp64 = strtoull (str, (char **) &ptr, 0);
1019                   if (cnt < 4)
1020                     cur_arg->size = 5;
1021
1022                   if (IS_INSN_TYPE (ARITH_INS))
1023                     {
1024                       if (cur_arg->size > 32
1025                           || (temp64 > ULONG_MAX))
1026                         {
1027                           if (cur_arg->size > 32)
1028                             as_bad (_("In Instruction `%s': Immediate size is \
1029                                     %lu bits cannot be accomodated"),
1030                                     ins_parse, cnt + 1);
1031
1032                           if (temp64 > ULONG_MAX)
1033                             as_bad (_("Value given more than 32 bits in \
1034                                     Instruction `%s'"), ins_parse);
1035                         }
1036                     }
1037                   if (IS_INSN_TYPE (ARITH_BYTE_INS))
1038                     {
1039                       if (cur_arg->size > 16
1040                           || !((temp64 & 0xFFFF0000) == 0xFFFF0000
1041                                || (temp64 & 0xFFFF0000) == 0x0))
1042                         {
1043                           if (cur_arg->size > 16)
1044                             as_bad (_("In Instruction `%s': Immediate size is \
1045                                     %lu bits cannot be accomodated"),
1046                                     ins_parse, cnt + 1);
1047
1048                           if (!((temp64 & 0xFFFF0000) == 0xFFFF0000
1049                                 || (temp64 & 0xFFFF0000) == 0x0))
1050                             as_bad (_("Value given more than 16 bits in \
1051                                     Instruction `%s'"), ins_parse);
1052                         }
1053                     }
1054                 }
1055               if (IS_INSN_TYPE (LD_STOR_INS) && cur_arg->type == arg_cr)
1056                 {
1057                   /* Cases handled ---
1058                      dispub4/dispuw4/dispud4 and for load store dispubwd4
1059                      is applicable only.  */
1060                   if (cur_arg->size <= 4)
1061                     cur_arg->size = 5;
1062                 }
1063               /* Argument number is checked to distinguish between
1064                  immediate and displacement in cmpbranch and bcopcond.  */
1065               if ((IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS))
1066                    && cur_arg_num == 2)
1067                 {
1068                   if (cur_arg->size != 32)
1069                     cur_arg->constant >>= 1;
1070                 }
1071
1072               mask_const (&cur_arg->constant, (int) cur_arg->size);
1073             }
1074         }
1075       else
1076         {
1077           /* Argument number is checked to distinguish between
1078              immediate and displacement in cmpbranch and bcopcond.  */
1079           if (((IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS))
1080                   && cur_arg_num == 2)
1081                 || IS_INSN_TYPE (BRANCH_NEQ_INS))
1082             {
1083               if (IS_INSN_TYPE (BRANCH_NEQ_INS))
1084                 {
1085                   if (cur_arg->constant == 0)
1086                     as_bad (_("Instruction `%s' has Zero offset"), ins_parse);
1087                 }
1088
1089               if (cur_arg->constant % 2 != 0)
1090                 as_bad (_("Instruction `%s' has odd offset"), ins_parse);
1091
1092               if (IS_INSN_TYPE (BRANCH_NEQ_INS))
1093                 {
1094                   if (cur_arg->constant > 32 || cur_arg->constant < 2)
1095                       as_bad (_("Instruction `%s' has illegal offset (%ld)"),
1096                               ins_parse, cur_arg->constant);
1097
1098                   cur_arg->constant -= 2;
1099                 }
1100
1101               cur_arg->constant >>= 1;
1102               set_operand_size (crx_ins);
1103             }
1104
1105           /* Compare branch argument number zero to be compared -
1106              mapped to cst4.  */
1107           if (IS_INSN_TYPE (CMPBR_INS) && cur_arg_num == 0)
1108             {
1109               for (cst4_op = cst4_map; cst4_op < (cst4_map + cst4_maps); cst4_op++)
1110                 {
1111                   if (cur_arg->constant == (unsigned int)cst4_op->value)
1112                     {
1113                       cur_arg->constant = cst4_op->binary;
1114                       is_cst4 = 1;
1115                       break;
1116                     }
1117                 }
1118               if (!is_cst4)
1119                 as_bad (_("Instruction `%s' has invalid imm value as an \
1120                           operand"), ins_parse);
1121             }
1122         }
1123       break;
1124
1125     case O_symbol:
1126     case O_subtract:
1127       crx_ins->rtype = BFD_RELOC_NONE;
1128       relocatable = 1;
1129
1130       switch (cur_arg->type)
1131         {
1132         case arg_cr:
1133           /* Have to consider various cases here.  */
1134           if (IS_INSN_TYPE (LD_STOR_INS_INC))
1135             /* 'load/stor <num>(reg)+'.  */
1136             crx_ins->rtype = BFD_RELOC_CRX_REGREL12;
1137           else if (IS_INSN_TYPE (CSTBIT_INS)
1138                    || IS_INSN_TYPE (STOR_IMM_INS))
1139             /* 'stor imm' and '[stc]bit'.  */
1140             crx_ins->rtype = BFD_RELOC_CRX_REGREL28;
1141           else
1142             /* General load/stor instruction.  */
1143             crx_ins->rtype = BFD_RELOC_CRX_REGREL32;
1144             break;
1145         case arg_idxr:
1146           /* Index Mode 22 bits relocation.  */
1147             crx_ins->rtype = BFD_RELOC_CRX_REGREL22;
1148           break;
1149         case arg_c:
1150           /* Absolute types.  */
1151           /* Case for jumps...dx  types.  */
1152           /* For bal.  */
1153           if (IS_INSN_MNEMONIC ("bal") || IS_INSN_TYPE (DCR_BRANCH_INS))
1154             crx_ins->rtype = BFD_RELOC_CRX_REL16;
1155           else if (IS_INSN_TYPE (BRANCH_INS))
1156             crx_ins->rtype = BFD_RELOC_CRX_REL8;
1157           else if (IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (STOR_IMM_INS)
1158                    || IS_INSN_TYPE (CSTBIT_INS))
1159             crx_ins->rtype = BFD_RELOC_CRX_ABS32;
1160           else if (IS_INSN_TYPE (BRANCH_NEQ_INS))
1161             crx_ins->rtype = BFD_RELOC_CRX_REL4;
1162           else if (IS_INSN_TYPE (CMPBR_INS) || IS_INSN_TYPE (COP_BRANCH_INS))
1163             crx_ins->rtype = BFD_RELOC_CRX_REL8_CMP;
1164           break;
1165         case arg_ic:
1166           if (IS_INSN_TYPE (ARITH_INS))
1167             crx_ins->rtype = BFD_RELOC_CRX_IMM32;
1168           else if (IS_INSN_TYPE (ARITH_BYTE_INS))
1169             crx_ins->rtype = BFD_RELOC_CRX_IMM16;
1170           break;
1171         default:
1172           break;
1173       }
1174       cur_arg->size = (bfd_reloc_type_lookup (stdoutput, crx_ins->rtype))->bitsize;
1175       break;
1176
1177     default:
1178       break;
1179     }
1180
1181   input_line_pointer = save;
1182   cur_arg->signflag = signflag;
1183   return crx_ins->exp.X_op;
1184 }
1185
1186 /* Get the values of the scale to be encoded -
1187    used for the scaled index mode of addressing.  */
1188
1189 static int
1190 exponent2scale (int val)
1191 {
1192   int exponent;
1193
1194   /* If 'val' is 0, the following 'for' will be an endless loop.  */
1195   if (val == 0)
1196     return 0;
1197
1198   for (exponent = 0; (val != 1); val >>= 1, exponent++)
1199     ;
1200
1201   return exponent;
1202 }
1203
1204 /* Parsing different types of operands
1205    -> constants             Immediate/Absolute/Relative numbers
1206    -> Labels                Relocatable symbols
1207    -> (rbase)               Register base
1208    -> disp(rbase)           Register relative
1209    -> disp(rbase)+          Post-increment mode
1210    -> disp(rbase,ridx,scl)  Register index mode  */
1211
1212 static void
1213 set_operand (char *operand, ins * crx_ins)
1214 {
1215   char *operandS; /* Pointer to start of sub-opearand.  */
1216   char *operandE; /* Pointer to end of sub-opearand.  */
1217   expressionS scale;
1218   int scale_val;
1219   char *input_save, c;
1220   argument *cur_arg = &crx_ins->arg[cur_arg_num]; /* Current argument.  */
1221
1222   /* Initialize pointers.  */
1223   operandS = operandE = operand;
1224
1225   switch (cur_arg->type)
1226     {
1227     case arg_sc:    /* Case *+0x18.  */
1228     case arg_ic:    /* Case $0x18.  */
1229       operandS++;
1230     case arg_c:     /* Case 0x18.  */
1231       /* Set constant.  */
1232       process_label_constant (operandS, crx_ins/*, op_num*/);
1233       
1234       if (cur_arg->type != arg_ic)
1235         cur_arg->type = arg_c;
1236       break;
1237
1238     case arg_icr:   /* Case $0x18(r1).  */
1239       operandS++;
1240     case arg_cr:    /* Case 0x18(r1).   */
1241       /* Set displacement constant.  */
1242       while (*operandE != '(')
1243         operandE++;
1244       *operandE = '\0';
1245       process_label_constant (operandS, crx_ins/*, op_num*/);
1246       operandS = operandE;    
1247     case arg_rbase: /* Case (r1).  */
1248       operandS++;
1249       /* Set register base.  */
1250       while (*operandE != ')')
1251         operandE++;
1252       *operandE = '\0';
1253       if ((cur_arg->r = get_register (operandS)) == nullregister)
1254         as_bad (_("Illegal register `%s' in Instruction `%s'"),
1255                 operandS, ins_parse);
1256
1257       if (cur_arg->type != arg_rbase)
1258         cur_arg->type = arg_cr;
1259       break;
1260
1261     case arg_idxr:
1262       /* Set displacement constant.  */
1263       while (*operandE != '(')
1264         operandE++;
1265       *operandE = '\0';
1266       process_label_constant (operandS, crx_ins);
1267       operandS = ++operandE;
1268       
1269       /* Set register base.  */
1270       while ((*operandE != ',') && (! ISSPACE (*operandE)))
1271         operandE++;
1272       *operandE++ = '\0';
1273       if ((cur_arg->r = get_register (operandS)) == nullregister)
1274         as_bad (_("Illegal register `%s' in Instruction `%s'"),
1275                 operandS, ins_parse);
1276
1277       /* Skip leading white space.  */
1278       while (ISSPACE (*operandE))
1279         operandE++;
1280       operandS = operandE;
1281
1282       /* Set register index.  */
1283       while ((*operandE != ')') && (*operandE != ','))
1284         operandE++;
1285       c = *operandE;
1286       *operandE++ = '\0';
1287
1288       if ((cur_arg->i_r = get_register (operandS)) == nullregister)
1289         as_bad (_("Illegal register `%s' in Instruction `%s'"),
1290                 operandS, ins_parse);
1291
1292       /* Skip leading white space.  */
1293       while (ISSPACE (*operandE))
1294         operandE++;
1295       operandS = operandE;
1296
1297       /* Set the scale.  */
1298       if (c == ')')
1299         cur_arg->scale = 0;
1300       else
1301         {
1302           while (*operandE != ')')
1303             operandE++;
1304           *operandE = '\0';
1305
1306           /* Preprocess the scale string.  */
1307           input_save = input_line_pointer;
1308           input_line_pointer = operandS;
1309           expression (&scale);
1310           input_line_pointer = input_save;
1311
1312           scale_val = scale.X_add_number;
1313
1314           /* Check if the scale value is legal.  */
1315           if (scale_val != 1 && scale_val != 2
1316               && scale_val != 4 && scale_val != 8)
1317             as_bad (_("Illegal Scale - `%d'"), scale_val);
1318
1319           cur_arg->scale = exponent2scale (scale_val);
1320         }
1321       break;
1322
1323     default:
1324       break;
1325     }
1326 }
1327
1328 /* Parse a single operand.
1329    operand - Current operand to parse.
1330    crx_ins - Current assembled instruction.  */
1331
1332 static void
1333 parse_operand (char *operand, ins * crx_ins)
1334 {
1335   int ret_val;
1336   argument *cur_arg = &crx_ins->arg[cur_arg_num]; /* Current argument.  */
1337
1338   /* Initialize the type to NULL before parsing.  */
1339   cur_arg->type = nullargs;
1340
1341   /* Check whether this is a general processor register.  */
1342   if ((ret_val = get_register (operand)) != nullregister)
1343     {
1344       cur_arg->type = arg_r;
1345       cur_arg->r = ret_val;
1346       goto set_size;
1347     }
1348
1349   /* Check whether this is a core [special] coprocessor register.  */
1350   if ((ret_val = get_copregister (operand)) != nullcopregister)
1351     {
1352       cur_arg->type = arg_copr;
1353       if (ret_val >= cs0)
1354         cur_arg->type = arg_copsr;
1355       cur_arg->cr = ret_val;
1356       goto set_size;
1357     }
1358
1359   /* Deal with special characters.  */
1360   switch (operand[0])
1361     {
1362     case '$':
1363       if (strchr (operand, '(') != NULL)
1364         cur_arg->type = arg_icr;
1365       else
1366         cur_arg->type = arg_ic;
1367       goto set_params;
1368       break;
1369
1370     case '*':
1371       cur_arg->type = arg_sc;
1372       goto set_params;
1373       break;
1374
1375     case '(':
1376       cur_arg->type = arg_rbase;
1377       goto set_params;
1378       break;
1379
1380     default:
1381         break;
1382     }
1383       
1384   if (strchr (operand, '(') != NULL)
1385     {
1386       if (strchr (operand, ',') != NULL
1387           && (strchr (operand, ',') > strchr (operand, '(')))
1388             cur_arg->type = arg_idxr;
1389       else
1390         cur_arg->type = arg_cr;
1391     }
1392   else
1393     cur_arg->type = arg_c;
1394   goto set_params;
1395
1396 /* Parse an operand according to its type.  */
1397 set_params:
1398   cur_arg->constant = 0;
1399   set_operand (operand, crx_ins);
1400
1401 /* Determine argument size.  */
1402 set_size:
1403   switch (cur_arg->type)
1404     {
1405     /* The following are all registers, so set their size to REG_SIZE.  */
1406     case arg_r:
1407     case arg_copr:
1408     case arg_copsr:
1409     case arg_rbase:
1410       cur_arg->size = REG_SIZE;
1411       break;
1412
1413     case arg_c:
1414     case arg_ic:
1415     case arg_sc:
1416     case arg_cr:
1417     case arg_icr:
1418     case arg_idxr:
1419       set_operand_size (crx_ins);
1420       break;
1421
1422     default:
1423       as_bad (_("Illegal argument type in instruction `%s'"), ins_parse);
1424       break;
1425     }
1426 }
1427
1428 /* Parse the various operands. Each operand is then analyzed to fillup 
1429    the fields in the crx_ins data structure.  */
1430
1431 static void
1432 parse_operands (ins * crx_ins, char *operands)
1433 {
1434   char *operandS;              /* Operands string.  */
1435   char *operandH, *operandT;   /* Single operand head/tail pointers.  */
1436   int allocated = 0;           /* Indicates a new operands string was allocated.  */
1437   char *operand[MAX_OPERANDS]; /* Separating the operands.  */
1438   int op_num = 0;              /* Current operand number we are parsing.  */
1439   int bracket_flag = 0;        /* Indicates a bracket '(' was found.  */
1440   int sq_bracket_flag = 0;     /* Indicates a square bracket '[' was found.  */
1441
1442   /* Preprocess the list of registers, if necessary.  */
1443   operandS = operandH = operandT = (INST_HAS_REG_LIST) ?
1444     preprocess_reglist (operands, &allocated) : operands;
1445
1446   while (*operandT != '\0')
1447     {
1448       if (*operandT == ',' && bracket_flag != 1 && sq_bracket_flag != 1)
1449         {
1450           *operandT++ = '\0';
1451           operand[op_num++] = strdup (operandH);
1452           operandH = operandT;
1453           continue;
1454         }
1455
1456       if (*operandT == ' ')
1457         as_bad (_("Illegal operands (whitespace): `%s'"), ins_parse);
1458
1459       if (*operandT == '(')
1460         bracket_flag = 1;
1461       else if (*operandT == '[')
1462         sq_bracket_flag = 1;
1463
1464       if (*operandT == ')')
1465         {
1466           if (bracket_flag)
1467             bracket_flag = 0;
1468           else
1469             as_fatal (_("Missing matching brackets : `%s'"), ins_parse);
1470         }
1471       else if (*operandT == ']')
1472         {
1473           if (sq_bracket_flag)
1474             sq_bracket_flag = 0;
1475           else
1476             as_fatal (_("Missing matching brackets : `%s'"), ins_parse);
1477         }
1478
1479       if (bracket_flag == 1 && *operandT == ')')
1480         bracket_flag = 0;
1481       else if (sq_bracket_flag == 1 && *operandT == ']')
1482         sq_bracket_flag = 0;
1483
1484       operandT++;
1485     }
1486
1487   /* Adding the last operand.  */
1488   operand[op_num++] = strdup (operandH);
1489   crx_ins->nargs = op_num;
1490
1491   /* Verifying correct syntax of operands (all brackets should be closed).  */
1492   if (bracket_flag || sq_bracket_flag)
1493     as_fatal (_("Missing matching brackets : `%s'"), ins_parse);
1494
1495   /* Now we parse each operand separately.  */
1496   for (op_num = 0; op_num < crx_ins->nargs; op_num++)
1497     {
1498       cur_arg_num = op_num;
1499       parse_operand (operand[op_num], crx_ins);
1500       free (operand[op_num]);
1501     }
1502
1503   if (allocated)
1504     free (operandS);
1505 }
1506
1507 /* Get the trap index in dispatch table, given its name.
1508    This routine is used by assembling the 'excp' instruction.  */
1509
1510 static int
1511 gettrap (char *s)
1512 {
1513   const trap_entry *trap;
1514
1515   for (trap = crx_traps; trap < (crx_traps + NUMTRAPS); trap++)
1516     if (strcasecmp (trap->name, s) == 0)
1517       return trap->entry;
1518
1519   as_bad (_("Unknown exception: `%s'"), s);
1520   return 0;
1521 }
1522
1523 /* Post-Increment instructions, as well as Store-Immediate instructions, are a 
1524    sub-group within load/stor instruction groups. 
1525    Therefore, when parsing a Post-Increment/Store-Immediate insn, we have to 
1526    advance the instruction pointer to the start of that sub-group (that is, up 
1527    to the first instruction of that type).
1528    Otherwise, the insn will be mistakenly identified as of type LD_STOR_INS.  */
1529
1530 static void
1531 handle_LoadStor (char *operands)
1532 {
1533   /* Post-Increment instructions precede Store-Immediate instructions in 
1534      CRX instruction table, hence they are handled before. 
1535      This synchronization should be kept.  */
1536
1537   /* Assuming Post-Increment insn has the following format :
1538      'MNEMONIC DISP(REG)+, REG' (e.g. 'loadw 12(r5)+, r6').
1539      LD_STOR_INS_INC are the only store insns containing a plus sign (+).  */
1540   if (strstr (operands, ")+") != NULL)
1541     {
1542       while (! IS_INSN_TYPE (LD_STOR_INS_INC))
1543         instruction++;
1544       return;
1545     }
1546
1547   /* Assuming Store-Immediate insn has the following format :
1548      'MNEMONIC $DISP, ...' (e.g. 'storb $1, 12(r5)').
1549      STOR_IMM_INS are the only store insns containing a dollar sign ($).  */
1550   if (strstr (operands, "$") != NULL)
1551     while (! IS_INSN_TYPE (STOR_IMM_INS))
1552       instruction++;
1553 }
1554
1555 /* Top level module where instruction parsing starts.
1556    crx_ins - data structure holds some information.
1557    operands - holds the operands part of the whole instruction.  */
1558
1559 static void
1560 parse_insn (ins *insn, char *operands)
1561 {
1562   /* Handle 'excp'/'cinv' */
1563   if (IS_INSN_MNEMONIC ("excp") || IS_INSN_MNEMONIC ("cinv"))
1564     {
1565       insn->nargs = 1;
1566       insn->arg[0].type = arg_ic;
1567       insn->arg[0].size = 4;
1568       insn->arg[0].constant = IS_INSN_MNEMONIC ("excp") ?
1569         gettrap (operands) : get_cinv_parameters (operands);
1570       return;
1571     }
1572
1573   /* Handle load/stor unique instructions before parsing.  */
1574   if (IS_INSN_TYPE (LD_STOR_INS))
1575     handle_LoadStor (operands);
1576
1577   if (operands != NULL)
1578     parse_operands (insn, operands);
1579 }
1580
1581 /* Cinv instruction requires special handling.  */
1582
1583 static int
1584 get_cinv_parameters (char * operand)
1585 {
1586   char *p = operand;
1587   int d_used = 0, i_used = 0, u_used = 0, b_used = 0;
1588
1589   while (*++p != ']')
1590     {
1591       if (*p == ',' || *p == ' ')
1592         continue;
1593
1594       if (*p == 'd')
1595         d_used = 1;
1596       else if (*p == 'i')
1597         i_used = 1;
1598       else if (*p == 'u')
1599         u_used = 1;
1600       else if (*p == 'b')
1601         b_used = 1;
1602       else
1603         as_bad (_("Illegal `cinv' parameter: `%c'"), *p);
1604     }
1605
1606   return ((b_used ? 8 : 0)
1607         + (d_used ? 4 : 0)
1608         + (i_used ? 2 : 0)
1609         + (u_used ? 1 : 0));
1610 }
1611
1612 /* Retrieve the opcode image of a given register.
1613    If the register is illegal for the current instruction,
1614    issue an error.  */
1615
1616 static int
1617 getreg_image (reg r)
1618 {
1619   const reg_entry *reg;
1620   char *reg_name;
1621   int is_procreg = 0; /* Nonzero means argument should be processor reg.  */
1622
1623   if (((IS_INSN_MNEMONIC ("mtpr")) && (cur_arg_num == 1))
1624       || ((IS_INSN_MNEMONIC ("mfpr")) && (cur_arg_num == 0)) )
1625     is_procreg = 1;
1626
1627   /* Check whether the register is in registers table.  */
1628   if (r < MAX_REG)
1629     reg = &crx_regtab[r];
1630   /* Check whether the register is in coprocessor registers table.  */
1631   else if (r < MAX_COPREG)
1632     reg = &crx_copregtab[r-MAX_REG];
1633   /* Register not found.  */
1634   else
1635     {
1636       as_bad (_("Unknown register: `%d'"), r);
1637       return 0;
1638     }
1639
1640   reg_name = reg->name;
1641
1642 /* Issue a error message when register is illegal.  */
1643 #define IMAGE_ERR \
1644   as_bad (_("Illegal register (`%s') in Instruction: `%s'"), \
1645             reg_name, ins_parse);                            \
1646   break;
1647
1648   switch (reg->type)
1649   {
1650     case CRX_U_REGTYPE:
1651       if (is_procreg || (instruction->flags & USER_REG))
1652         return reg->image;
1653       else
1654         IMAGE_ERR;
1655
1656     case CRX_CFG_REGTYPE:
1657       if (is_procreg)
1658         return reg->image;
1659       else
1660         IMAGE_ERR;
1661
1662     case CRX_R_REGTYPE:
1663       if (! is_procreg)
1664         return reg->image;
1665       else
1666         IMAGE_ERR;
1667
1668     case CRX_C_REGTYPE:
1669     case CRX_CS_REGTYPE:
1670       return reg->image;
1671       break;
1672
1673     default:
1674       IMAGE_ERR;
1675   }
1676
1677   return 0;
1678 }
1679
1680 /* Routine used to get the binary-string equivalent of a integer constant
1681    which currently require currbits to represent itself to be extended to
1682    nbits.  */
1683
1684 static unsigned long int
1685 getconstant (unsigned long int x, int nbits)
1686 {
1687   int cnt = 0;
1688   unsigned long int temp = x;
1689
1690   while (temp > 0)
1691     {
1692       temp >>= 1;
1693       cnt++;
1694     }
1695
1696   /* Escape sequence to next 16bit immediate.  */
1697   if (cnt > nbits)
1698     as_bad (_("Value `%ld' truncated to fit `%d' bits in instruction `%s'"),
1699             x, cnt, ins_parse);
1700   else
1701     {
1702       if (signflag)
1703         x |= SET_BITS_MASK (cnt, nbits - cnt);
1704       else
1705         x &= CLEAR_BITS_MASK (cnt, nbits - cnt);
1706     }
1707
1708   /* The following expression avoids overflow if
1709      'nbits' is the number of bits in 'bfd_vma'.  */
1710   return (x & ((((1 << (nbits - 1)) - 1) << 1) | 1));
1711 }
1712
1713 /* Print a constant value to 'output_opcode':
1714    ARG holds the operand's type and value.
1715    SHIFT represents the location of the operand to be print into.
1716    NBITS determines the size (in bits) of the constant.  */
1717
1718 static void
1719 print_constant (int nbits, int shift, argument *arg)
1720 {
1721   unsigned long mask = 0;
1722
1723   long constant = getconstant (arg->constant, nbits);
1724
1725   switch (nbits)
1726   {
1727     case 32:
1728     case 28:
1729     case 24:
1730     case 22:
1731       /* mask the upper part of the constant, that is, the bits
1732          going to the lowest byte of output_opcode[0].
1733          The upper part of output_opcode[1] is always filled,
1734          therefore it is always masked with 0xFFFF.  */
1735       mask = (1 << (nbits - 16)) - 1;
1736       /* Divide the constant between two consecutive words :
1737                  0         1         2         3
1738             +---------+---------+---------+---------+
1739             |         | X X X X | X X X X |         |
1740             +---------+---------+---------+---------+
1741               output_opcode[0]    output_opcode[1]     */
1742
1743       CRX_PRINT (0, (constant >> WORD_SHIFT) & mask, 0);
1744       CRX_PRINT (1, (constant & 0xFFFF), WORD_SHIFT);
1745       break;
1746
1747     case 16:
1748     case 12:
1749       /* Special case - in arg_cr, the SHIFT represents the location
1750          of the REGISTER, not the constant, which is itself not shifted.  */
1751       if (arg->type == arg_cr)
1752         {
1753           CRX_PRINT (0, constant,  0);
1754           break;
1755         }
1756
1757       /* When instruction size is 3 and 'shift' is 16, a 16-bit constant is 
1758          always filling the upper part of output_opcode[1]. If we mistakenly 
1759          write it to output_opcode[0], the constant prefix (that is, 'match')
1760          will be overriden.
1761                  0         1         2         3
1762             +---------+---------+---------+---------+
1763             | 'match' |         | X X X X |         |
1764             +---------+---------+---------+---------+
1765               output_opcode[0]    output_opcode[1]     */
1766
1767       if ((instruction->size > 2) && (shift == WORD_SHIFT))
1768         CRX_PRINT (1, constant, WORD_SHIFT);
1769       else
1770         CRX_PRINT (0, constant, shift);
1771       break;
1772
1773     default:
1774       CRX_PRINT (0, constant,  shift);
1775       break;
1776   }
1777 }
1778
1779 /* Print an operand to 'output_opcode', which later on will be
1780    printed to the object file:
1781    ARG holds the operand's type, size and value.
1782    SHIFT represents the printing location of operand.
1783    NBITS determines the size (in bits) of a constant operand.  */
1784
1785 static void
1786 print_operand (int nbits, int shift, argument *arg)
1787 {
1788   switch (arg->type)
1789     {
1790     case arg_r:
1791       CRX_PRINT (0, getreg_image (arg->r), shift);
1792       break;
1793
1794     case arg_copr:
1795       if (arg->cr < c0 || arg->cr > c15)
1796         as_bad (_("Illegal Co-processor register in Instruction `%s' "),
1797                 ins_parse);
1798       CRX_PRINT (0, getreg_image (arg->cr), shift);
1799       break;
1800
1801     case arg_copsr:
1802       if (arg->cr < cs0 || arg->cr > cs15)
1803         as_bad (_("Illegal Co-processor special register in Instruction `%s' "),
1804                 ins_parse);
1805       CRX_PRINT (0, getreg_image (arg->cr), shift);
1806       break;
1807
1808     case arg_idxr:
1809       /*    16      12        8    6         0
1810             +--------------------------------+
1811             | r_base | r_idx  | scl|  disp   |
1812             +--------------------------------+    */
1813       CRX_PRINT (0, getreg_image (arg->r), 12);
1814       CRX_PRINT (0, getreg_image (arg->i_r), 8);
1815       CRX_PRINT (0, arg->scale, 6);
1816     case arg_ic:
1817     case arg_c:
1818       print_constant (nbits, shift, arg);
1819       break;
1820
1821     case arg_rbase:
1822       CRX_PRINT (0, getreg_image (arg->r), shift);
1823       break;
1824
1825     case arg_cr:
1826       /* case base_cst4.  */
1827       if ((instruction->flags & DISPU4MAP) && cst4flag)
1828         output_opcode[0] |= (getconstant (arg->constant, nbits)
1829                              << (shift + REG_SIZE));
1830       else
1831         /* rbase_disps<NN> and other such cases.  */
1832         print_constant (nbits, shift, arg);
1833       /* Add the register argument to the output_opcode.  */
1834       CRX_PRINT (0, getreg_image (arg->r), shift);
1835       break;
1836
1837     default:
1838       break;
1839     }
1840 }
1841
1842 /* Retrieve the number of operands for the current assembled instruction.  */
1843
1844 static int
1845 get_number_of_operands (void)
1846 {
1847   int i;
1848
1849   for (i = 0; instruction->operands[i].op_type && i < MAX_OPERANDS; i++)
1850     ;
1851   return i;
1852 }
1853
1854 /* Assemble a single instruction :
1855    Instruction has been parsed and all operand values set appropriately.
1856    Algorithm for assembling -
1857    For instruction to be assembled:
1858     Step 1: Find instruction in the array crx_instruction with same mnemonic.
1859     Step 2: Find instruction with same operand types.
1860     Step 3: If (size_of_operands) match then done, else increment the
1861             array_index and goto Step3.
1862     Step 4: Cannot assemble
1863    Returns 1 upon success, 0 upon failure.  */
1864
1865 static int
1866 assemble_insn (char *mnemonic, ins *insn)
1867 {
1868   /* Argument type of each operand in the instruction we are looking for.  */
1869   argtype atyp[MAX_OPERANDS];
1870   /* Argument type of each operand in the current instruction.  */
1871   argtype atyp_act[MAX_OPERANDS];
1872   /* Size (in bits) of each operand in the instruction we are looking for.  */
1873   int bits[MAX_OPERANDS];
1874   /* Size (in bits) of each operand in the current instruction.  */
1875   int bits_act[MAX_OPERANDS];
1876   /* Location (in bits) of each operand in the current instruction.  */
1877   int shift_act[MAX_OPERANDS];
1878   /* Instruction type to match.  */
1879   unsigned int ins_type;
1880   int match = 0;
1881   int done_flag = 0;
1882   int dispu4map_type = 0;
1883   int changed_already = 0;
1884   unsigned int temp_value = 0;
1885   int instrtype, i;
1886   /* A pointer to the argument's constant value.  */
1887   unsigned long int *cons;
1888   /* Pointer to loop over all cst4_map entries.  */
1889   const cst4_entry *cst4_op;
1890
1891   /* Instruction has no operands -> copy only the constant opcode.   */
1892   if (insn->nargs == 0)
1893     {
1894       output_opcode[0] = BIN (instruction->match, instruction->match_bits);
1895       return 1;
1896     }
1897
1898   /* Find instruction with same number of operands.  */
1899   while (get_number_of_operands () != insn->nargs
1900          && IS_INSN_MNEMONIC (mnemonic))
1901     instruction++;
1902
1903   if (!IS_INSN_MNEMONIC (mnemonic))
1904     return 0;
1905
1906   /* Initialize argument type and size of each given operand.  */
1907   for (i = 0; i < insn->nargs; i++)
1908     {
1909       atyp[i] = insn->arg[i].type;
1910       bits[i] = insn->arg[i].size;
1911     }
1912
1913   /* Initialize argument type and size of each operand in current inst.  */
1914   GET_ACTUAL_TYPE;
1915   GET_ACTUAL_SIZE;
1916
1917   /* In some case, same mnemonic can appear with different instruction types.
1918      For example, 'storb' is supported with 3 different types :
1919      LD_STOR_INS, LD_STOR_INS_INC, STOR_IMM_INS.
1920      We assume that when reaching this point, the instruction type was 
1921      pre-determined. We need to make sure that the type stays the same
1922      during a search for matching instruction.  */
1923   ins_type = CRX_INS_TYPE(instruction->flags);
1924
1925   while (match != 1
1926          /* Check we didn't get to end of table.  */
1927          && instruction->mnemonic != NULL
1928          /* Check that the actual mnemonic is still available.  */
1929          && IS_INSN_MNEMONIC (mnemonic)
1930          /* Check that the instruction type wasn't changed.  */
1931          && IS_INSN_TYPE(ins_type))
1932     {
1933       /* Check for argement type compatibility.  */
1934       for (i = 0; i < insn->nargs; i++)
1935         {
1936           if (atyp_act[i] == atyp[i])
1937             done_flag = 1;
1938           else
1939             {
1940               done_flag = 0;
1941               break;
1942             }
1943         }
1944
1945       if (done_flag)
1946         {
1947           for (i = 0; i < insn->nargs; i++)
1948             {
1949               if ((get_flags (instruction->operands[i].op_type) & OPERAND_UNSIGNED)
1950                   && (insn->arg[i].signflag))
1951                     {
1952                       done_flag = 0;
1953                       break;
1954                     }
1955             }
1956         }
1957
1958       if (done_flag == 0)
1959         {
1960           /* Try again with next instruction.  */
1961           instruction++;
1962           GET_ACTUAL_TYPE;
1963           GET_ACTUAL_SIZE;
1964           continue;
1965         }
1966       else
1967         {
1968           /* Check for size compatibility.  */
1969           for (i = 0; i < insn->nargs; i++)
1970             {
1971               if (bits[i] > bits_act[i])
1972                 {
1973                   /* Actual size is too small - try again.  */
1974                   done_flag = 0;
1975                   instruction++;
1976                   GET_ACTUAL_TYPE;
1977                   GET_ACTUAL_SIZE;
1978                   break;
1979                 }
1980             }
1981
1982         }
1983
1984       if (done_flag == 1)
1985         {
1986           /* Full match is found.  */
1987           match = 1;
1988           break;
1989         }
1990     }
1991
1992   if (match == 0)
1993     /* We haven't found a match - instruction can't be assembled.  */
1994     return 0;
1995   else
1996     /* Full match - print the final image.  */
1997     {
1998       /* If the post-increment address mode is used and the load/store 
1999          source register is the same as rbase, the result of the 
2000          instruction is undefined.  */
2001       if (IS_INSN_TYPE (LD_STOR_INS_INC))
2002         {
2003           /* Enough to verify that one of the arguments is a simple reg.  */
2004           if ((insn->arg[0].type == arg_r) || (insn->arg[1].type == arg_r))
2005             if (insn->arg[0].r == insn->arg[1].r)
2006               as_bad (_("Same src/dest register is used (`r%d'), result is undefined"), 
2007                        insn->arg[0].r);
2008         }
2009
2010       /* Optimization: Omit a zero displacement in bit operations, 
2011          saving 2-byte encoding space (e.g., 'cbitw $8, 0(r1)').  */
2012       if (IS_INSN_TYPE (CSTBIT_INS) && !relocatable)
2013         {
2014           if ((instruction->operands[1].op_type == rbase_disps12)
2015                && (insn->arg[1].constant == 0))
2016                 {
2017                   instruction--;
2018                   GET_ACTUAL_SIZE;
2019                 }
2020         }
2021
2022       /* Some instruction assume the stack pointer as rptr operand.
2023          Issue an error when the register to be loaded is also SP.  */
2024       if (instruction->flags & NO_SP)
2025         {
2026           if (getreg_image (insn->arg[0].r) == getreg_image (sp))
2027             as_bad (_("`%s' has undefined result"), ins_parse);
2028         }
2029
2030       /* If the rptr register is specified as one of the registers to be loaded, 
2031          the final contents of rptr are undefined. Thus, we issue an error.  */
2032       if (instruction->flags & NO_RPTR)
2033         {
2034           if ((1 << getreg_image (insn->arg[0].r)) & insn->arg[1].constant)
2035             as_bad (_("Same src/dest register is used (`r%d'), result is undefined"), 
2036              getreg_image (insn->arg[0].r));
2037         }
2038
2039       /* Handle positive constants.  */
2040       if (!signflag)
2041         {
2042           if ((instruction->flags & DISPU4MAP) && !relocatable)
2043             {
2044               /* Get the map type of the instruction.  */
2045               instrtype = instruction->flags & REVERSE_MATCH ? 0 : 1;
2046               cons = &insn->arg[instrtype].constant;
2047               dispu4map_type = instruction->flags & DISPU4MAP;
2048
2049               switch (dispu4map_type)
2050                 {
2051                 case DISPUB4:
2052                   /* 14 and 15 are reserved escape sequences of dispub4.  */
2053                   if (*cons == 14 || *cons == 15)
2054                     {
2055                       instruction++;
2056                       GET_ACTUAL_SIZE;
2057                     }
2058                   break;
2059
2060                 case DISPUW4:
2061                   /* Mapping has to be done.  */
2062                   if (*cons <= 15 && *cons % 2 != 0)
2063                     {
2064                       instruction++;
2065                       GET_ACTUAL_SIZE;
2066                     }
2067                   else if (*cons > 15 && *cons < 27 && *cons % 2 == 0)
2068                     {
2069                       instruction--;
2070                       GET_ACTUAL_SIZE;
2071                     }
2072                   if (*cons < 27 && *cons % 2 == 0)
2073                     *cons /= 2;
2074                   break;
2075
2076                 case DISPUD4:
2077                   /* Mapping has to be done.  */
2078                   if (*cons <= 15 && *cons % 4 != 0)
2079                     {
2080                       instruction++;
2081                       GET_ACTUAL_SIZE;
2082                     }
2083                   else if (*cons > 15 && *cons < 53 && *cons % 4 == 0)
2084                     {
2085                       instruction--;
2086                       GET_ACTUAL_SIZE;
2087                     }
2088                   if (*cons < 53 && *cons % 4 == 0)
2089                     *cons /= 4;
2090                   break;
2091                 default:
2092                   as_bad (_("Invalid DISPU4 type"));
2093                   break;
2094               }
2095             }
2096
2097           /* Check whether a cst4 mapping has to be done.  */
2098           if ((instruction->flags & CST4MAP) && !relocatable)
2099             {
2100               /* 'const' equals reserved escape sequences -->>
2101                  represent as i16.  */
2102               if (insn->arg[0].constant == ESC_16
2103                   || insn->arg[0].constant == ESC_32)
2104                 {
2105                   instruction++;
2106                   GET_ACTUAL_SIZE;
2107                 }
2108               else
2109                 {
2110                   /* Loop over cst4_map entries.  */
2111                   for (cst4_op = cst4_map; cst4_op < (cst4_map + cst4_maps);
2112                        cst4_op++)
2113                     {
2114                       /* 'const' equals a binary, which is already mapped
2115                          by a different value -->> represent as i16.  */
2116                       if (insn->arg[0].constant == (unsigned int)cst4_op->binary
2117                           && cst4_op->binary != cst4_op->value)
2118                         {
2119                           instruction++;
2120                           GET_ACTUAL_SIZE;
2121                         }
2122                       /* 'const' equals a value bigger than 16 -->> map to
2123                          its binary and represent as cst4.  */
2124                       else if (insn->arg[0].constant == (unsigned int)cst4_op->value
2125                                && insn->arg[0].constant >= 16)
2126                         {
2127                           instruction--;
2128                           insn->arg[0].constant = cst4_op->binary;
2129                           GET_ACTUAL_SIZE;
2130                         }
2131                     }
2132                 }
2133             }
2134
2135           /* Special check for 'addub 0, r0' instruction -
2136              The opcode '0000 0000 0000 0000' is not allowed.  */
2137           if (IS_INSN_MNEMONIC ("addub"))
2138             {
2139               if ((instruction->operands[0].op_type == cst4)
2140                   && instruction->operands[1].op_type == regr)
2141                 {
2142                   if (insn->arg[0].constant == 0 && insn->arg[1].r == r0)
2143                     instruction++;
2144                 }
2145             }
2146           if ((IS_INSN_TYPE (LD_STOR_INS) || IS_INSN_TYPE (CSTBIT_INS)
2147                || IS_INSN_TYPE (STOR_IMM_INS)) & !relocatable)
2148             {
2149               instrtype = instruction->flags & REVERSE_MATCH ? 0 : 1;
2150               changed_already = 0;
2151               /* Convert 32 bits accesses to 16 bits accesses.  */
2152               if (instruction->operands[instrtype].op_type == abs32)
2153                 {
2154                   if ((insn->arg[instrtype].constant & 0xFFFF0000) == 0xFFFF0000)
2155                     {
2156                       instruction--;
2157                       insn->arg[instrtype].constant =
2158                         insn->arg[instrtype].constant & 0xFFFF;
2159                       insn->arg[instrtype].size = 16;
2160                       changed_already = 1;
2161                       GET_ACTUAL_SIZE;
2162                     }
2163                 }
2164               /* Convert 16 bits accesses to 32 bits accesses.  */
2165               if (instruction->operands[instrtype].op_type == abs16
2166                   && changed_already != 1)
2167                 {
2168                   instruction++;
2169                   insn->arg[instrtype].constant =
2170                     insn->arg[instrtype].constant & 0xFFFF;
2171                   insn->arg[instrtype].size = 32;
2172                   GET_ACTUAL_SIZE;
2173                 }
2174               changed_already = 0;
2175             }
2176         }
2177
2178       for (i = 0; i < insn->nargs; i++)
2179         {
2180           /* Mark a CST4 argument, if exists.  */
2181           if (get_flags (instruction->operands[i].op_type) & OPERAND_CST4)
2182             cst4flag = 1;
2183
2184           /* Handle reserved escape sequences.  */
2185           if ((get_flags (instruction->operands[i].op_type) & OPERAND_ESC)
2186               && !relocatable)
2187             {
2188               /* 0x7e and 0x7f are reserved escape sequences of dispe9.  */
2189               if (insn->arg[i].constant == 0x7e || insn->arg[i].constant == 0x7f)
2190                 {
2191                   /* Use a disps17 for these values.  */
2192                   instruction++;
2193                   GET_ACTUAL_SIZE;
2194                 }
2195             }
2196         }
2197
2198       /* First, copy the instruction's opcode.  */
2199       output_opcode[0] = BIN (instruction->match, instruction->match_bits);
2200
2201       /* Swap the argument values in case bcop instructions.  */
2202       if (IS_INSN_TYPE (COP_BRANCH_INS))
2203         {
2204           temp_value = insn->arg[0].constant;
2205           insn->arg[0].constant = insn->arg[1].constant;
2206           insn->arg[1].constant = temp_value;
2207         }
2208
2209       for (i = 0; i < insn->nargs; i++)
2210         {
2211           shift_act[i] = instruction->operands[i].shift;
2212           signflag = insn->arg[i].signflag;
2213           cur_arg_num = i;
2214           print_operand (bits_act[i], shift_act[i], &insn->arg[i]);
2215         }
2216     }
2217
2218   return 1;
2219 }
2220
2221 /* Set the appropriate bit for register 'r' in 'mask'.
2222    This indicates that this register is loaded or stored by
2223    the instruction.  */
2224
2225 static void
2226 mask_reg (int r, unsigned short int *mask)
2227 {
2228   if ((reg)r > (reg)sp)
2229     {
2230       as_bad (_("Invalid Register in Register List"));
2231       return;
2232     }
2233
2234   *mask |= (1 << r);
2235 }
2236
2237 /* Preprocess register list - create a 16-bit mask with one bit for each
2238    of the 16 general purpose registers. If a bit is set, it indicates
2239    that this register is loaded or stored by the instruction.  */
2240
2241 static char *
2242 preprocess_reglist (char *param, int *allocated)
2243 {
2244   char reg_name[MAX_REGNAME_LEN]; /* Current parsed register name.  */
2245   char *regP;                     /* Pointer to 'reg_name' string.  */
2246   int reg_counter = 0;            /* Count number of parsed registers.  */
2247   unsigned short int mask = 0;    /* Mask for 16 general purpose registers.  */
2248   char *new_param;                /* New created operands string.  */
2249   char *paramP = param;           /* Pointer to original opearands string.  */
2250   char maskstring[10];            /* Array to print the mask as a string.  */
2251   int hi_found = 0, lo_found = 0; /* Boolean flags for hi/lo registers.  */
2252   reg r;
2253   copreg cr;
2254
2255   /* If 'param' is already in form of a number, no need to preprocess.  */
2256   if (strchr (paramP, '{') == NULL)
2257     return param;
2258
2259   /* Verifying correct syntax of operand.  */
2260   if (strchr (paramP, '}') == NULL)
2261     as_fatal (_("Missing matching brackets : `%s'"), ins_parse);
2262
2263   while (*paramP++ != '{');
2264
2265   new_param = (char *)xcalloc (MAX_INST_LEN, sizeof (char));
2266   *allocated = 1;
2267   strncpy (new_param, param, paramP - param - 1);
2268
2269   while (*paramP != '}')
2270     {
2271       regP = paramP;
2272       memset (&reg_name, '\0', sizeof (reg_name));
2273
2274       while (ISALNUM (*paramP))
2275         paramP++;
2276
2277       strncpy (reg_name, regP, paramP - regP);
2278
2279       /* Coprocessor register c<N>.  */
2280       if (IS_INSN_TYPE (COP_REG_INS))
2281         {
2282           if (((cr = get_copregister (reg_name)) == nullcopregister)
2283               || (crx_copregtab[cr-MAX_REG].type != CRX_C_REGTYPE))
2284             as_fatal (_("Illegal register `%s' in cop-register list"), reg_name);
2285           mask_reg (getreg_image (cr - c0), &mask);
2286         }
2287       /* Coprocessor Special register cs<N>.  */
2288       else if (IS_INSN_TYPE (COPS_REG_INS))
2289         {
2290           if (((cr = get_copregister (reg_name)) == nullcopregister)
2291               || (crx_copregtab[cr-MAX_REG].type != CRX_CS_REGTYPE))
2292             as_fatal (_("Illegal register `%s' in cop-special-register list"), 
2293                       reg_name);
2294           mask_reg (getreg_image (cr - cs0), &mask);
2295         }
2296       /* User register u<N>.  */
2297       else if (instruction->flags & USER_REG)
2298         {
2299           if (streq(reg_name, "uhi"))
2300             {
2301               hi_found = 1;
2302               goto next_inst;
2303             }
2304           else if (streq(reg_name, "ulo"))
2305             {
2306               lo_found = 1;
2307               goto next_inst;
2308             }
2309           else if (((r = get_register (reg_name)) == nullregister)
2310               || (crx_regtab[r].type != CRX_U_REGTYPE))
2311             as_fatal (_("Illegal register `%s' in user register list"), reg_name);
2312           
2313           mask_reg (getreg_image (r - u0), &mask);        
2314         }
2315       /* General purpose register r<N>.  */
2316       else
2317         {
2318           if (streq(reg_name, "hi"))
2319             {
2320               hi_found = 1;
2321               goto next_inst;
2322             }
2323           else if (streq(reg_name, "lo"))
2324             {
2325               lo_found = 1;
2326               goto next_inst;
2327             }
2328           else if (((r = get_register (reg_name)) == nullregister)
2329               || (crx_regtab[r].type != CRX_R_REGTYPE))
2330             as_fatal (_("Illegal register `%s' in register list"), reg_name);
2331
2332           mask_reg (getreg_image (r - r0), &mask);
2333         }
2334
2335       if (++reg_counter > MAX_REGS_IN_MASK16)
2336         as_bad (_("Maximum %d bits may be set in `mask16' operand"),
2337                 MAX_REGS_IN_MASK16);
2338
2339 next_inst:
2340       while (!ISALNUM (*paramP) && *paramP != '}')
2341           paramP++;
2342     }
2343
2344   if (*++paramP != '\0')
2345     as_warn (_("rest of line ignored; first ignored character is `%c'"),
2346              *paramP);
2347
2348   switch (hi_found + lo_found)
2349     {
2350     case 0:
2351       /* At least one register should be specified.  */
2352       if (mask == 0)
2353         as_bad (_("Illegal `mask16' operand, operation is undefined - `%s'"),
2354                 ins_parse);
2355       break;
2356
2357     case 1:
2358       /* HI can't be specified without LO (and vise-versa).  */
2359       as_bad (_("HI/LO registers should be specified together"));
2360       break;
2361
2362     case 2:
2363       /* HI/LO registers mustn't be masked with additional registers.  */
2364       if (mask != 0)
2365         as_bad (_("HI/LO registers should be specified without additional registers"));
2366
2367     default:
2368       break;
2369     }
2370
2371   sprintf (maskstring, "$0x%x", mask);
2372   strcat (new_param, maskstring);
2373   return new_param;
2374 }
2375
2376 /* Print the instruction.
2377    Handle also cases where the instruction is relaxable/relocatable.  */
2378
2379 void
2380 print_insn (ins *insn)
2381 {
2382   unsigned int i, j, insn_size;
2383   char *this_frag;
2384   unsigned short words[4];
2385
2386   /* Arrange the insn encodings in a WORD size array.  */
2387   for (i = 0, j = 0; i < 2; i++)
2388     {
2389       words[j++] = (output_opcode[i] >> 16) & 0xFFFF;
2390       words[j++] = output_opcode[i] & 0xFFFF;
2391     }
2392
2393   /* Handle relaxtion.  */
2394   if ((instruction->flags & RELAXABLE) && relocatable)
2395     {
2396       int relax_subtype;
2397
2398       /* Write the maximal instruction size supported.  */
2399       insn_size = INSN_MAX_SIZE;
2400
2401       /* bCC  */
2402       if (IS_INSN_TYPE (BRANCH_INS))
2403         relax_subtype = 0;
2404       /* bal  */
2405       else if (IS_INSN_TYPE (DCR_BRANCH_INS) || IS_INSN_MNEMONIC ("bal"))
2406         relax_subtype = 3;
2407       /* cmpbr  */
2408       else if (IS_INSN_TYPE (CMPBR_INS))
2409         relax_subtype = 5;
2410       else
2411         abort ();
2412
2413       this_frag = frag_var (rs_machine_dependent, insn_size * 2,
2414                             4, relax_subtype,
2415                             insn->exp.X_add_symbol,
2416                             insn->exp.X_add_number,
2417                             0);
2418     }
2419   else
2420     {
2421       insn_size = instruction->size;
2422       this_frag = frag_more (insn_size * 2);
2423
2424       /* Handle relocation.  */
2425       if ((relocatable) && (insn->rtype != BFD_RELOC_NONE))
2426         {
2427           reloc_howto_type *reloc_howto;
2428           int size;
2429
2430           reloc_howto = bfd_reloc_type_lookup (stdoutput, insn->rtype);
2431
2432           if (!reloc_howto)
2433             abort ();
2434
2435           size = bfd_get_reloc_size (reloc_howto);
2436
2437           if (size < 1 || size > 4)
2438             abort ();
2439
2440           fix_new_exp (frag_now, this_frag - frag_now->fr_literal,
2441                        size, &insn->exp, reloc_howto->pc_relative,
2442                        insn->rtype);
2443         }
2444     }
2445
2446   /* Write the instruction encoding to frag.  */
2447   for (i = 0; i < insn_size; i++)
2448     {
2449       md_number_to_chars (this_frag, (valueT) words[i], 2);
2450       this_frag += 2;
2451     }
2452 }
2453
2454 /* This is the guts of the machine-dependent assembler.  OP points to a
2455    machine dependent instruction.  This function is supposed to emit
2456    the frags/bytes it assembles to.  */
2457
2458 void
2459 md_assemble (char *op)
2460 {
2461   ins crx_ins;
2462   char *param;
2463   char c;
2464
2465   /* Reset global variables for a new instruction.  */
2466   reset_vars (op, &crx_ins);
2467
2468   /* Strip the mnemonic.  */
2469   for (param = op; *param != 0 && !ISSPACE (*param); param++)
2470     ;
2471   c = *param;
2472   *param++ = '\0';
2473
2474   /* Find the instruction.  */
2475   instruction = (const inst *) hash_find (crx_inst_hash, op);
2476   if (instruction == NULL)
2477     {
2478       as_bad (_("Unknown opcode: `%s'"), op);
2479       return;
2480     }
2481
2482   /* Tie dwarf2 debug info to the address at the start of the insn.  */
2483   dwarf2_emit_insn (0);
2484
2485   if (NO_OPERANDS_INST (op))
2486     /* Handle instructions with no operands.  */
2487     crx_ins.nargs = 0;
2488   else
2489     /* Parse the instruction's operands.  */
2490     parse_insn (&crx_ins, param);
2491
2492   /* Assemble the instruction.  */
2493   if (assemble_insn (op, &crx_ins) == 0)
2494     {
2495       as_bad (_("Illegal operands in instruction : `%s'"), ins_parse);
2496       return;
2497     }
2498
2499   /* Print the instruction.  */
2500   print_insn (&crx_ins);
2501 }