1 /* tc-metag.c -- Assembler for the Imagination Technologies Meta.
2    Copyright (C) 2013-2014 Free Software Foundation, Inc.
3    Contributed by Imagination Technologies Ltd.
4 
5    This file is part of GAS, the GNU Assembler.
6 
7    GAS is free software; you can redistribute it and/or modify
8    it under the terms of the GNU General Public License as published by
9    the Free Software Foundation; either version 3, or (at your option)
10    any later version.
11 
12    GAS is distributed in the hope that it will be useful,
13    but WITHOUT ANY WARRANTY; without even the implied warranty of
14    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.	 See the
15    GNU General Public License for more details.
16 
17    You should have received a copy of the GNU General Public License
18    along with GAS; see the file COPYING.  If not, write to the Free
19    Software Foundation, 51 Franklin Street - Fifth Floor, Boston, MA
20    02110-1301, USA.  */
21 
22 #include "as.h"
23 #include "subsegs.h"
24 #include "symcat.h"
25 #include "safe-ctype.h"
26 #include "hashtab.h"
27 #include "libbfd.h"
28 
29 #include <stdio.h>
30 
31 #include "opcode/metag.h"
32 
33 const char comment_chars[]        = "!";
34 const char line_comment_chars[]   = "!#";
35 const char line_separator_chars[] = ";";
36 const char FLT_CHARS[]            = "rRsSfFdDxXpP";
37 const char EXP_CHARS[]            = "eE";
38 const char metag_symbol_chars[]   = "[";
39 
40 static char register_chars[256];
41 static char mnemonic_chars[256];
42 
43 #define is_register_char(x) (register_chars[(unsigned char) x])
44 #define is_mnemonic_char(x) (mnemonic_chars[(unsigned char) x])
45 #define is_whitespace_char(x) (((x) == ' ') || ((x) == '\t'))
46 #define is_space_char(x) ((x) == ' ')
47 
48 #define FPU_PREFIX_CHAR 'f'
49 #define DSP_PREFIX_CHAR 'd'
50 
51 /* Instruction mnemonics that need disambiguating with respect to prefixes.  */
52 #define FFB_INSN        "ffb"
53 #define DCACHE_INSN     "dcache"
54 #define DEFR_INSN       "defr"
55 
56 #define FPU_DOUBLE_CHAR 'd'
57 #define FPU_PAIR_CHAR   'l'
58 
59 #define DSP_DUAL_CHAR	'l'
60 
61 #define END_OF_INSN     '\0'
62 
63 /* Maximum length of a mnemonic including all suffixes.  */
64 #define MAX_MNEMONIC_LEN 16
65 /* Maximum length of a register name.  */
66 #define MAX_REG_LEN      17
67 
68 /* Addressing modes must be enclosed with square brackets.  */
69 #define ADDR_BEGIN_CHAR '['
70 #define ADDR_END_CHAR   ']'
71 /* Immediates must be prefixed with a hash.  */
72 #define IMM_CHAR        '#'
73 
74 #define COMMA           ','
75 #define PLUS            '+'
76 #define MINUS           '-'
77 
78 /* Short units are those that can be encoded with 2 bits.  */
79 #define SHORT_UNITS     "D0, D1, A0 or A1"
80 
81 static unsigned int mcpu_opt = CoreMeta12;
82 static unsigned int mfpu_opt = 0;
83 static unsigned int mdsp_opt = 0;
84 
85 const char * md_shortopts = "m:";
86 
87 struct option md_longopts[] =
88 {
89   {NULL, no_argument, NULL, 0}
90 };
91 size_t md_longopts_size = sizeof (md_longopts);
92 
93 /* Parser hash tables.  */
94 static htab_t mnemonic_htab;
95 static htab_t reg_htab;
96 static htab_t dsp_reg_htab;
97 static htab_t dsp_tmpl_reg_htab[2];
98 static htab_t scond_htab;
99 
100 #define GOT_NAME "__GLOBAL_OFFSET_TABLE__"
101 symbolS * GOT_symbol;
102 
103 enum fpu_insn_width {
104   FPU_WIDTH_SINGLE,
105   FPU_WIDTH_DOUBLE,
106   FPU_WIDTH_PAIR,
107 };
108 
109 #define FPU_ACTION_ABS_CHAR   'a'
110 #define FPU_ACTION_INV_CHAR   'i'
111 #define FPU_ACTION_QUIET_CHAR 'q'
112 #define FPU_ACTION_ZERO_CHAR  'z'
113 
114 #define FPU_ACTION_ABS        0x1
115 #define FPU_ACTION_INV        0x2
116 #define FPU_ACTION_QUIET      0x4
117 #define FPU_ACTION_ZERO       0x8
118 
119 enum dsp_insn_width {
120   DSP_WIDTH_SINGLE,
121   DSP_WIDTH_DUAL,
122 };
123 
124 #define DSP_ACTION_QR64_CHAR     'q'
125 #define DSP_ACTION_UMUL_CHAR     'u'
126 #define DSP_ACTION_ROUND_CHAR    'r'
127 #define DSP_ACTION_CLAMP9_CHAR   'g'
128 #define DSP_ACTION_CLAMP8_CHAR   'b'
129 #define DSP_ACTION_MOD_CHAR      'm'
130 #define DSP_ACTION_ACC_ZERO_CHAR 'z'
131 #define DSP_ACTION_ACC_ADD_CHAR  'p'
132 #define DSP_ACTION_ACC_SUB_CHAR  'n'
133 #define DSP_ACTION_OV_CHAR       'o'
134 
135 #define DSP_ACTION_QR64          0x001
136 #define DSP_ACTION_UMUL          0x002
137 #define DSP_ACTION_ROUND         0x004
138 #define DSP_ACTION_CLAMP9        0x008
139 #define DSP_ACTION_CLAMP8        0x010
140 #define DSP_ACTION_MOD           0x020
141 #define DSP_ACTION_ACC_ZERO      0x040
142 #define DSP_ACTION_ACC_ADD       0x080
143 #define DSP_ACTION_ACC_SUB       0x100
144 #define DSP_ACTION_OV            0x200
145 
146 #define DSP_DAOPPAME_8_CHAR    'b'
147 #define DSP_DAOPPAME_16_CHAR   'w'
148 #define DSP_DAOPPAME_TEMP_CHAR 't'
149 #define DSP_DAOPPAME_HIGH_CHAR 'h'
150 
151 #define DSP_DAOPPAME_8         0x1
152 #define DSP_DAOPPAME_16        0x2
153 #define DSP_DAOPPAME_TEMP      0x4
154 #define DSP_DAOPPAME_HIGH      0x8
155 
156 /* Structure holding information about a parsed instruction.  */
157 typedef struct {
158   /* Instruction type.  */
159   enum insn_type type;
160   /* Split condition code. */
161   enum scond_code scond;
162 
163   /* Instruction bits.  */
164   unsigned int bits;
165   /* Size of the instruction in bytes.  */
166   size_t len;
167 
168   /* FPU instruction encoding.  */
169   enum fpu_insn_width fpu_width;
170   unsigned int fpu_action_flags;
171 
172   /* DSP instruction encoding. */
173   enum dsp_insn_width dsp_width;
174   unsigned int dsp_action_flags;
175   unsigned int dsp_daoppame_flags;
176 
177   /* Reloc encoding information, maximum of one reloc per insn.  */
178   enum bfd_reloc_code_real reloc_type;
179   int reloc_pcrel;
180   expressionS reloc_exp;
181   unsigned int reloc_size;
182 } metag_insn;
183 
184 /* Structure holding information about a parsed addressing mode.  */
185 typedef struct {
186   const metag_reg *base_reg;
187   const metag_reg *offset_reg;
188 
189   expressionS exp;
190 
191   enum bfd_reloc_code_real reloc_type;
192 
193   /* Whether we have an immediate or not.  */
194   unsigned short immediate:1;
195   /* Whether or not the base register is updated.  */
196   unsigned short update:1;
197   /* Whether the operation uses the address pre or post increment.  */
198   unsigned short post_increment:1;
199   /* Whether the immediate should be negated.  */
200   unsigned short negate:1;
201 } metag_addr;
202 
203 /* Linked list of possible parsers for this instruction.  */
204 typedef struct _insn_templates {
205   const insn_template *template;
206   struct _insn_templates *next;
207 } insn_templates;
208 
209 /* Parse an instruction that takes no operands.  */
210 static const char *
parse_none(const char * line,metag_insn * insn,const insn_template * template)211 parse_none (const char *line, metag_insn *insn,
212 	    const insn_template *template)
213 {
214   insn->bits = template->meta_opcode;
215   insn->len = 4;
216   return line;
217 }
218 
219 /* Return the next non-whitespace character in LINE or NULL.  */
220 static const char *
skip_whitespace(const char * line)221 skip_whitespace (const char *line)
222 {
223   const char *l = line;
224 
225   if (is_whitespace_char (*l))
226     {
227       l++;
228     }
229 
230   return l;
231 }
232 
233 /* Return the next non-space character in LINE or NULL.  */
234 static const char *
skip_space(const char * line)235 skip_space (const char *line)
236 {
237   const char *l = line;
238 
239   if (is_space_char (*l))
240     {
241       l++;
242     }
243 
244   return l;
245 }
246 
247 /* Return the character after the current one in LINE if the current
248    character is a comma, otherwise NULL.  */
249 static const char *
skip_comma(const char * line)250 skip_comma (const char *line)
251 {
252   const char *l = line;
253 
254   if (l == NULL || *l != COMMA)
255     return NULL;
256 
257   l++;
258 
259   return l;
260 }
261 
262 /* Return the metag_reg struct corresponding to NAME or NULL if no such
263    register exists.  */
264 static const metag_reg *
parse_gp_reg(const char * name)265 parse_gp_reg (const char *name)
266 {
267   const metag_reg *reg;
268   metag_reg entry;
269 
270   entry.name = name;
271 
272   reg = (const metag_reg *) htab_find (reg_htab, &entry);
273 
274   return reg;
275 }
276 
277 /* Parse a list of up to COUNT GP registers from LINE, returning the
278    registers parsed in REGS and the number parsed in REGS_READ. Return
279    a pointer to the next character or NULL.  */
280 static const char *
parse_gp_regs_list(const char * line,const metag_reg ** regs,size_t count,size_t * regs_read)281 parse_gp_regs_list (const char *line, const metag_reg **regs, size_t count,
282 		    size_t *regs_read)
283 {
284   const char *l = line;
285   char reg_buf[MAX_REG_LEN];
286   int seen_regs = 0;
287   size_t i;
288 
289   for (i = 0; i < count; i++)
290     {
291       size_t len = 0;
292       const char *next;
293 
294       next = l;
295 
296       if (i > 0)
297 	{
298 	  l = skip_comma (l);
299 	  if (l == NULL)
300 	    {
301 	      *regs_read = seen_regs;
302 	      return next;
303 	    }
304 	}
305 
306       while (is_register_char (*l))
307 	{
308 	  reg_buf[len] = *l;
309 	  l++;
310 	  len++;
311 	  if (!(len < MAX_REG_LEN))
312 	    return NULL;
313 	}
314 
315       reg_buf[len] = '\0';
316 
317       if (len)
318 	{
319 	  const metag_reg *reg = parse_gp_reg (reg_buf);
320 
321 	  if (!reg)
322 	    {
323 	      *regs_read = seen_regs;
324 	      return next;
325 	    }
326 	  else
327 	    {
328 	      regs[i] = reg;
329 	      seen_regs++;
330 	    }
331 	}
332       else
333 	{
334 	  *regs_read = seen_regs;
335 	  return next;
336 	}
337     }
338 
339   *regs_read = seen_regs;
340   return l;
341 }
342 
343 /* Parse a list of exactly COUNT GP registers from LINE, returning the
344    registers parsed in REGS. Return a pointer to the next character or NULL.  */
345 static const char *
parse_gp_regs(const char * line,const metag_reg ** regs,size_t count)346 parse_gp_regs (const char *line, const metag_reg **regs, size_t count)
347 {
348   const char *l = line;
349   size_t regs_read = 0;
350 
351   l = parse_gp_regs_list (l, regs, count, &regs_read);
352 
353   if (regs_read != count)
354     return NULL;
355   else
356     return l;
357 }
358 
359 /* Parse a list of exactly COUNT FPU registers from LINE, returning the
360    registers parsed in REGS. Return a pointer to the next character or NULL.  */
361 static const char *
parse_fpu_regs(const char * line,const metag_reg ** regs,size_t count)362 parse_fpu_regs (const char *line, const metag_reg **regs, size_t count)
363 {
364   const char *l = line;
365   size_t regs_read = 0;
366 
367   l = parse_gp_regs_list (l, regs, count, &regs_read);
368 
369   if (regs_read != count)
370     return NULL;
371   else
372     {
373       size_t i;
374       for (i = 0; i < count; i++)
375 	{
376 	  if (regs[i]->unit != UNIT_FX)
377 	    return NULL;
378 	}
379       return l;
380     }
381 }
382 
383 /* Return TRUE if REG1 and REG2 are in paired units.  */
384 static bfd_boolean
is_unit_pair(const metag_reg * reg1,const metag_reg * reg2)385 is_unit_pair (const metag_reg *reg1, const metag_reg *reg2)
386 {
387   if ((reg1->unit == UNIT_A0 &&
388        (reg2->unit == UNIT_A1)) ||
389       (reg1->unit == UNIT_A1 &&
390        (reg2->unit == UNIT_A0)) ||
391       (reg1->unit == UNIT_D0 &&
392        (reg2->unit == UNIT_D1)) ||
393       (reg1->unit == UNIT_D1 &&
394        (reg2->unit == UNIT_D0)))
395     return TRUE;
396 
397   return FALSE;
398 }
399 
400 /* Return TRUE if REG1 and REG2 form a register pair.  */
401 static bfd_boolean
is_reg_pair(const metag_reg * reg1,const metag_reg * reg2)402 is_reg_pair (const metag_reg *reg1, const metag_reg *reg2)
403 {
404   if (reg1->unit == UNIT_FX &&
405       reg2->unit == UNIT_FX &&
406       reg2->no == reg1->no + 1)
407     return TRUE;
408 
409   if (reg1->no != reg2->no)
410     return FALSE;
411 
412   return is_unit_pair (reg1, reg2);
413 }
414 
415 /* Parse a pair of GP registers from LINE, returning the registers parsed
416    in REGS. Return a pointer to the next character or NULL.  */
417 static const char *
parse_pair_gp_regs(const char * line,const metag_reg ** regs)418 parse_pair_gp_regs (const char *line, const metag_reg **regs)
419 {
420   const char *l = line;
421 
422   l = parse_gp_regs (line, regs, 2);
423 
424   if (l == NULL)
425     {
426       l = parse_gp_regs (line, regs, 1);
427 
428       if (l == NULL)
429 	return NULL;
430 
431       if (regs[0]->unit == UNIT_RD)
432 	return l;
433       else
434 	return NULL;
435     }
436 
437   if (is_reg_pair (regs[0], regs[1]))
438     return l;
439 
440   return NULL;
441 }
442 
443 /* Parse a unit-to-unit MOV instruction.  */
444 static const char *
parse_mov_u2u(const char * line,metag_insn * insn,const insn_template * template)445 parse_mov_u2u (const char *line, metag_insn *insn,
446 	       const insn_template *template)
447 {
448   const metag_reg *regs[2];
449 
450   line = parse_gp_regs (line, regs, 2);
451 
452   if (line == NULL)
453     return NULL;
454 
455   if (!mfpu_opt && (regs[0]->unit == UNIT_FX || regs[1]->unit == UNIT_FX))
456     {
457       as_bad (_("no floating point unit specified"));
458       return NULL;
459     }
460 
461   insn->bits = (template->meta_opcode |
462 		(regs[1]->no << 19) |
463 		(regs[0]->no << 14) |
464 		(regs[1]->unit << 10) |
465 		(regs[0]->unit << 5));
466   insn->len = 4;
467   return line;
468 }
469 
470 /* Parse a MOV to port instruction.  */
471 static const char *
parse_mov_port(const char * line,metag_insn * insn,const insn_template * template)472 parse_mov_port (const char *line, metag_insn *insn,
473 		const insn_template *template)
474 {
475   const char *l = line;
476   unsigned int is_movl = MINOR_OPCODE (template->meta_opcode) == MOVL_MINOR;
477   const metag_reg *dest_regs[2];
478   const metag_reg *port_regs[1];
479 
480   if (is_movl)
481     l = parse_gp_regs (l, dest_regs, 2);
482   else
483     l = parse_gp_regs (l, dest_regs, 1);
484 
485   if (l == NULL)
486     return NULL;
487 
488   if (template->insn_type == INSN_FPU && dest_regs[0]->unit != UNIT_FX)
489     return NULL;
490 
491   l = skip_comma (l);
492 
493   if (l == NULL ||
494       *l == END_OF_INSN)
495     return NULL;
496 
497   l = parse_gp_regs (l, port_regs, 1);
498 
499   if (l == NULL)
500     return NULL;
501 
502   if (port_regs[0]->unit != UNIT_RD ||
503       port_regs[0]->no != 0)
504     return NULL;
505 
506   if (is_movl)
507     {
508       if (!is_unit_pair (dest_regs[0], dest_regs[1]))
509 	return NULL;
510 
511       insn->bits = (template->meta_opcode |
512 		    (dest_regs[0]->no << 14) |
513 		    (dest_regs[1]->no << 9) |
514 		    ((dest_regs[0]->unit & SHORT_UNIT_MASK) << 5));
515     }
516   else
517     insn->bits = (template->meta_opcode |
518 		  (dest_regs[0]->no << 14) |
519 		  (dest_regs[0]->unit << 5));
520 
521   insn->len = 4;
522   return l;
523 }
524 
525 /* Parse a MOVL to TTREC instruction.  */
526 static const char *
parse_movl_ttrec(const char * line,metag_insn * insn,const insn_template * template)527 parse_movl_ttrec (const char *line, metag_insn *insn,
528 		  const insn_template *template)
529 {
530   const char *l = line;
531   const metag_reg *src_regs[2];
532   const metag_reg *dest_regs[1];
533 
534   l = parse_gp_regs (l, dest_regs, 1);
535 
536   if (l == NULL)
537     return NULL;
538 
539   if (dest_regs[0]->unit != UNIT_TT ||
540       dest_regs[0]->no != 3)
541     return NULL;
542 
543   l = skip_comma (l);
544 
545   if (l == NULL ||
546       *l == END_OF_INSN)
547     return NULL;
548 
549   l = parse_gp_regs (l, src_regs, 2);
550 
551   if (l == NULL)
552     return NULL;
553 
554   if (!is_unit_pair (src_regs[0], src_regs[1]))
555     return NULL;
556 
557   insn->bits = (template->meta_opcode |
558 		(src_regs[0]->no << 19) |
559 		(src_regs[1]->no << 14) |
560 		((src_regs[0]->unit & SHORT_UNIT_MASK) << 7));
561 
562   insn->len = 4;
563   return l;
564 }
565 
566 /* Parse an incrementing or decrementing addressing mode.  */
567 static const char *
parse_addr_incr_op(const char * line,metag_addr * addr)568 parse_addr_incr_op (const char *line, metag_addr *addr)
569 {
570   const char *l = line;
571   const char *ll;
572 
573   ll = l + 1;
574 
575   if (*l == PLUS &&
576       *ll == PLUS)
577     {
578       addr->update = 1;
579       ll++;
580       return ll;
581     }
582   else if (*l == MINUS &&
583 	   *ll == MINUS)
584     {
585       addr->update = 1;
586       addr->negate = 1;
587       ll++;
588       return ll;
589     }
590   return NULL;
591 }
592 
593 /* Parse an pre-incrementing or pre-decrementing addressing mode.  */
594 static const char *
parse_addr_pre_incr_op(const char * line,metag_addr * addr)595 parse_addr_pre_incr_op (const char *line, metag_addr *addr)
596 {
597   return parse_addr_incr_op (line, addr);
598 }
599 
600 /* Parse an post-incrementing or post-decrementing addressing mode.  */
601 static const char *
parse_addr_post_incr_op(const char * line,metag_addr * addr)602 parse_addr_post_incr_op (const char *line, metag_addr *addr)
603 {
604   const char *l;
605 
606   l = parse_addr_incr_op (line, addr);
607 
608   if (l == NULL)
609     return NULL;
610 
611   addr->post_increment = 1;
612 
613   return l;
614 }
615 
616 /* Parse an infix addressing mode.  */
617 static const char *
parse_addr_op(const char * line,metag_addr * addr)618 parse_addr_op (const char *line, metag_addr *addr)
619 {
620   const char *l = line;
621   const char *ll;
622 
623   ll = l + 1;
624 
625   if (*l == PLUS)
626     {
627       if (*ll == PLUS)
628 	{
629 	  addr->update = 1;
630 	  ll++;
631 	  return ll;
632 	}
633       l++;
634       return l;
635     }
636   return NULL;
637 }
638 
639 /* Parse the immediate portion of an addrssing mode.  */
640 static const char *
parse_imm_addr(const char * line,metag_addr * addr)641 parse_imm_addr (const char *line, metag_addr *addr)
642 {
643   const char *l = line;
644   char *save_input_line_pointer;
645   expressionS *exp = &addr->exp;
646 
647   /* Skip #.  */
648   if (*l == '#')
649     l++;
650   else
651     return NULL;
652 
653   save_input_line_pointer = input_line_pointer;
654   input_line_pointer = (char *) l;
655 
656   expression (exp);
657 
658   l = input_line_pointer;
659   input_line_pointer = save_input_line_pointer;
660 
661   if (exp->X_op == O_absent || exp->X_op == O_big)
662     {
663       return NULL;
664     }
665   else if (exp->X_op == O_constant)
666     {
667       return l;
668     }
669   else
670     {
671       if (exp->X_op == O_PIC_reloc &&
672 	  exp->X_md == BFD_RELOC_METAG_GETSET_GOT)
673 	{
674 	  exp->X_op = O_symbol;
675 	  addr->reloc_type = BFD_RELOC_METAG_GETSET_GOT;
676 	}
677       else if (exp->X_op == O_PIC_reloc &&
678 	       exp->X_md == BFD_RELOC_METAG_TLS_IE)
679 	{
680 	  exp->X_op = O_symbol;
681 	  addr->reloc_type = BFD_RELOC_METAG_TLS_IE;
682 	}
683       else if (exp->X_op == O_PIC_reloc &&
684 	  exp->X_md == BFD_RELOC_METAG_GOTOFF)
685 	{
686 	  exp->X_op = O_symbol;
687 	  addr->reloc_type = BFD_RELOC_METAG_GETSET_GOTOFF;
688 	}
689       else
690 	addr->reloc_type = BFD_RELOC_METAG_GETSETOFF;
691       return l;
692     }
693 }
694 
695 /* Parse the offset portion of an addressing mode (register or immediate).  */
696 static const char *
parse_addr_offset(const char * line,metag_addr * addr,int size)697 parse_addr_offset (const char *line, metag_addr *addr, int size)
698 {
699   const char *l = line;
700   const metag_reg *regs[1];
701 
702   if (*l == IMM_CHAR)
703     {
704       /* ++ is a valid operator in our addressing but not in an expr. Make
705 	 sure that the expression parser never sees it.  */
706       char *ppp = strstr(l, "++");
707       char ppch = '+';
708 
709       if (ppp)
710 	*ppp = '\0';
711 
712       l = parse_imm_addr (l, addr);
713 
714       if (ppp)
715 	*ppp = ppch;
716 
717       if (l == NULL)
718 	return NULL;
719 
720       if (addr->exp.X_add_number % size)
721 	{
722 	  as_bad (_("offset must be a multiple of %d"), size);
723 	  return NULL;
724 	}
725 
726       addr->immediate = 1;
727       return l;
728     }
729   else
730     {
731       l = parse_gp_regs (l, regs, 1);
732 
733       if (l == NULL)
734 	return NULL;
735 
736       if (regs[0]->unit != addr->base_reg->unit)
737 	{
738 	  as_bad (_("offset and base must be from the same unit"));
739 	  return NULL;
740 	}
741 
742       addr->offset_reg = regs[0];
743       return l;
744     }
745 }
746 
747 /* Parse an addressing mode.  */
748 static const char *
parse_addr(const char * line,metag_addr * addr,unsigned int size)749 parse_addr (const char *line, metag_addr *addr, unsigned int size)
750 {
751   const char *l = line;
752   const char *ll;
753   const metag_reg *regs[1];
754 
755   /* Skip opening square bracket.  */
756   l++;
757 
758   ll = parse_addr_pre_incr_op (l, addr);
759 
760   if (ll != NULL)
761     l = ll;
762 
763   l = parse_gp_regs (l, regs, 1);
764 
765   if (l == NULL)
766     return NULL;
767 
768   addr->base_reg = regs[0];
769 
770   if (*l == ADDR_END_CHAR)
771     {
772       addr->exp.X_op = O_constant;
773       addr->exp.X_add_symbol = NULL;
774       addr->exp.X_op_symbol = NULL;
775       if (addr->update == 1)
776 	{
777 	  /* We have a pre increment/decrement.  */
778 	  addr->exp.X_add_number = size;
779 	}
780       else
781 	{
782 	  /* Simple register with no offset (0 immediate).  */
783 	  addr->exp.X_add_number = 0;
784 	}
785       addr->immediate = 1;
786       l++;
787       return l;
788     }
789 
790   /* We already had a pre increment/decrement.  */
791   if (addr->update == 1)
792     return NULL;
793 
794   ll = parse_addr_post_incr_op (l, addr);
795 
796   if (ll && *ll == ADDR_END_CHAR)
797     {
798       if (addr->update == 1)
799 	{
800 	  /* We have a post increment/decrement.  */
801 	  addr->exp.X_op = O_constant;
802 	  addr->exp.X_add_number = size;
803 	  addr->exp.X_add_symbol = NULL;
804 	  addr->exp.X_op_symbol = NULL;
805 	  addr->post_increment = 1;
806 	}
807       addr->immediate = 1;
808       ll++;
809       return ll;
810     }
811 
812   addr->post_increment = 0;
813 
814   l = parse_addr_op (l, addr);
815 
816   if (l == NULL)
817     return NULL;
818 
819   l = parse_addr_offset (l, addr, size);
820 
821   if (l == NULL)
822     return NULL;
823 
824   if (*l == ADDR_END_CHAR)
825     {
826       l++;
827       return l;
828     }
829 
830   /* We already had a pre increment/decrement. */
831   if (addr->update == 1)
832     return NULL;
833 
834   l = parse_addr_post_incr_op (l, addr);
835 
836   if (l == NULL)
837     return NULL;
838 
839   if (*l == ADDR_END_CHAR)
840     {
841       l++;
842       return l;
843     }
844 
845   return NULL;
846 }
847 
848 /* Parse a GET or pipeline MOV instruction.  */
849 static const char *
parse_get(const char * line,const metag_reg ** regs,metag_addr * addr,unsigned int size,bfd_boolean is_mov)850 parse_get (const char *line, const metag_reg **regs, metag_addr *addr,
851 	   unsigned int size, bfd_boolean is_mov)
852 {
853   const char *l = line;
854 
855   if (size == 8)
856     {
857       l = parse_pair_gp_regs (l, regs);
858 
859       if (l == NULL)
860 	  return NULL;
861     }
862   else
863     {
864       l = parse_gp_regs (l, regs, 1);
865 
866       if (l == NULL)
867 	{
868 	  if (!is_mov)
869 	    as_bad (_("invalid destination register"));
870 	  return NULL;
871 	}
872     }
873 
874   l = skip_comma (l);
875 
876   if (l == NULL ||
877       *l == END_OF_INSN)
878     return NULL;
879 
880   l = parse_addr (l, addr, size);
881 
882   if (l == NULL)
883     {
884       if (!is_mov)
885 	as_bad (_("invalid memory operand"));
886       return NULL;
887     }
888 
889   return l;
890 }
891 
892 /* Parse a SET instruction.  */
893 static const char *
parse_set(const char * line,const metag_reg ** regs,metag_addr * addr,unsigned int size)894 parse_set (const char *line, const metag_reg **regs, metag_addr *addr,
895 	   unsigned int size)
896 {
897   const char *l = line;
898 
899   l = parse_addr (l, addr, size);
900 
901   if (l == NULL)
902     {
903 	  as_bad (_("invalid memory operand"));
904 	  return NULL;
905     }
906 
907   l = skip_comma (l);
908 
909   if (l == NULL ||
910       *l == END_OF_INSN)
911     return NULL;
912 
913   if (size == 8)
914     {
915       const char *ll = l;
916 
917       ll = parse_pair_gp_regs (l, regs);
918 
919       if (ll == NULL)
920 	{
921 	  /* Maybe this is an RD register, which is 64 bits wide so needs no
922 	     pair.  */
923 	  l = parse_gp_regs (l, regs, 1);
924 
925 	  if (l == NULL ||
926 	      regs[0]->unit != UNIT_RD)
927 	    {
928 	      return NULL;
929 	    }
930 	}
931       else
932 	l = ll;
933     }
934   else
935     {
936       l = parse_gp_regs (l, regs, 1);
937 
938       if (l == NULL)
939 	{
940 	  as_bad (_("invalid source register"));
941 	  return NULL;
942 	}
943     }
944 
945   return l;
946 }
947 
948 /* Check a signed integer value can be represented in the given number
949    of bits.  */
950 static bfd_boolean
within_signed_range(int value,unsigned int bits)951 within_signed_range (int value, unsigned int bits)
952 {
953   int min_val = -(1 << (bits - 1));
954   int max_val = (1 << (bits - 1)) - 1;
955   return (value <= max_val) && (value >= min_val);
956 }
957 
958 /* Check an unsigned integer value can be represented in the given number
959    of bits.  */
960 static bfd_boolean
within_unsigned_range(unsigned int value,unsigned int bits)961 within_unsigned_range (unsigned int value, unsigned int bits)
962 {
963   return value < (unsigned int)(1 << bits);
964 }
965 
966 /* Return TRUE if UNIT can be expressed using a short code.  */
967 static bfd_boolean
is_short_unit(enum metag_unit unit)968 is_short_unit (enum metag_unit unit)
969 {
970   switch (unit)
971     {
972     case UNIT_A0:
973     case UNIT_A1:
974     case UNIT_D0:
975     case UNIT_D1:
976       return TRUE;
977     default:
978       return FALSE;
979     }
980 }
981 
982 /* Copy reloc data from ADDR to INSN.  */
983 static void
copy_addr_reloc(metag_insn * insn,metag_addr * addr)984 copy_addr_reloc (metag_insn *insn, metag_addr *addr)
985 {
986   memcpy (&insn->reloc_exp, &addr->exp, sizeof(insn->reloc_exp));
987   insn->reloc_type = addr->reloc_type;
988 }
989 
990 /* Parse a GET, SET or pipeline MOV instruction.  */
991 static const char *
parse_get_set(const char * line,metag_insn * insn,const insn_template * template)992 parse_get_set (const char *line, metag_insn *insn,
993 	       const insn_template *template)
994 {
995   const char *l = line;
996   const metag_reg *regs[2];
997   metag_addr addr;
998   unsigned int size = metag_get_set_size_bytes (template->meta_opcode);
999   bfd_boolean is_get = MAJOR_OPCODE (template->meta_opcode) == OPC_GET;
1000   unsigned int reg_no;
1001 
1002   memset(&addr, 0, sizeof(addr));
1003   addr.reloc_type = BFD_RELOC_UNUSED;
1004 
1005   if (is_get)
1006     {
1007       bfd_boolean is_mov = strncmp (template->name, "MOV", 3) == 0;
1008 
1009       l = parse_get (l, regs, &addr, size, is_mov);
1010 
1011       if (l == NULL)
1012 	return NULL;
1013 
1014       if (!(regs[0]->unit == UNIT_D0 ||
1015 	    regs[0]->unit == UNIT_D1 ||
1016 	    regs[0]->unit == UNIT_A0 ||
1017 	    regs[0]->unit == UNIT_A1 ||
1018 	    (regs[0]->unit == UNIT_RD && is_mov) ||
1019 	    (regs[0]->unit == UNIT_CT && size == 4) ||
1020 	    (regs[0]->unit == UNIT_PC && size == 4) ||
1021 	    (regs[0]->unit == UNIT_TR && size == 4) ||
1022 	    (regs[0]->unit == UNIT_TT && (size == 4 || size == 8)) ||
1023 	    regs[0]->unit == UNIT_FX))
1024 	{
1025 	  as_bad (_("invalid destination unit"));
1026 	  return NULL;
1027 	}
1028 
1029       if (regs[0]->unit == UNIT_RD)
1030 	{
1031 	  if (regs[0]->no == 0)
1032 	    {
1033 	      as_bad (_("mov cannot use RD port as destination"));
1034 	      return NULL;
1035 	    }
1036 	}
1037 
1038       reg_no = regs[0]->no;
1039     }
1040   else
1041     {
1042       l = parse_set (l, regs, &addr, size);
1043 
1044       if (l == NULL)
1045 	return NULL;
1046 
1047       if (!(regs[0]->unit == UNIT_D0 ||
1048 	    regs[0]->unit == UNIT_D1 ||
1049 	    regs[0]->unit == UNIT_A0 ||
1050 	    regs[0]->unit == UNIT_A1 ||
1051 	    regs[0]->unit == UNIT_RD ||
1052 	    (regs[0]->unit == UNIT_CT && size == 4) ||
1053 	    (regs[0]->unit == UNIT_PC && size == 4) ||
1054 	    (regs[0]->unit == UNIT_TR && size == 4) ||
1055 	    (regs[0]->unit == UNIT_TT && (size == 4 || size == 8)) ||
1056 	    regs[0]->unit == UNIT_FX))
1057 	{
1058 	  as_bad (_("invalid source unit"));
1059 	  return NULL;
1060 	}
1061 
1062       if (addr.immediate == 0 &&
1063 	  (regs[0]->unit == addr.base_reg->unit ||
1064 	   (size == 8 && is_unit_pair (regs[0], addr.base_reg))))
1065 	{
1066 	  as_bad (_("source and address units must not be shared for this addressing mode"));
1067 	  return NULL;
1068 	}
1069 
1070       if (regs[0]->unit == UNIT_RD)
1071 	{
1072 	  if (regs[0]->no != 0)
1073 	    {
1074 	      as_bad (_("set can only use RD port as source"));
1075 	      return NULL;
1076 	    }
1077 	  reg_no = 16;
1078 	}
1079       else
1080 	reg_no = regs[0]->no;
1081     }
1082 
1083   insn->bits = (template->meta_opcode |
1084 		(reg_no << 19) |
1085 		(regs[0]->unit << 1));
1086 
1087   if (!is_short_unit (addr.base_reg->unit))
1088     {
1089       as_bad (_("base unit must be one of %s"), SHORT_UNITS);
1090       return NULL;
1091     }
1092 
1093   insn->bits |= ((addr.base_reg->no << 14) |
1094 		 ((addr.base_reg->unit & SHORT_UNIT_MASK) << 5));
1095 
1096   if (addr.immediate)
1097     {
1098       int offset = addr.exp.X_add_number;
1099 
1100       copy_addr_reloc (insn, &addr);
1101 
1102       if (addr.negate)
1103 	offset = -offset;
1104 
1105       offset = offset / (int)size;
1106 
1107       if (!within_signed_range (offset, GET_SET_IMM_BITS))
1108 	{
1109 	  /* We already tried to encode as an extended GET/SET.  */
1110 	  as_bad (_("offset value out of range"));
1111 	  return NULL;
1112 	}
1113 
1114       offset = offset & GET_SET_IMM_MASK;
1115 
1116       insn->bits |= (0x1 << 25);
1117       insn->bits |= (offset << 8);
1118     }
1119   else
1120     {
1121       insn->bits |= (addr.offset_reg->no << 9);
1122     }
1123 
1124   if (addr.update)
1125     insn->bits |= (0x1 << 7);
1126 
1127   if (addr.post_increment)
1128     insn->bits |= 0x1;
1129 
1130   insn->len = 4;
1131   return l;
1132 }
1133 
1134 /* Parse an extended GET or SET instruction.  */
1135 static const char *
parse_get_set_ext(const char * line,metag_insn * insn,const insn_template * template)1136 parse_get_set_ext (const char *line, metag_insn *insn,
1137 		   const insn_template *template)
1138 {
1139   const char *l = line;
1140   const metag_reg *regs[2];
1141   metag_addr addr;
1142   unsigned int size = metag_get_set_ext_size_bytes (template->meta_opcode);
1143   bfd_boolean is_get = MINOR_OPCODE (template->meta_opcode) == GET_EXT_MINOR;
1144   bfd_boolean is_mov = MINOR_OPCODE (template->meta_opcode) == MOV_EXT_MINOR;
1145   unsigned int reg_unit;
1146 
1147   memset(&addr, 0, sizeof(addr));
1148   addr.reloc_type = BFD_RELOC_UNUSED;
1149 
1150   if (is_get || is_mov)
1151     {
1152       l = parse_get (l, regs, &addr, size, is_mov);
1153     }
1154   else
1155     {
1156       l = parse_set (l, regs, &addr, size);
1157     }
1158 
1159   if (l == NULL)
1160     return NULL;
1161 
1162   /* Extended GET/SET does not support incrementing addressing.  */
1163   if (addr.update)
1164     return NULL;
1165 
1166   if (is_mov)
1167     {
1168       if (regs[0]->unit != UNIT_RD)
1169 	{
1170 	  as_bad (_("destination unit must be RD"));
1171 	  return NULL;
1172 	}
1173       reg_unit = 0;
1174     }
1175   else
1176     {
1177       if (!is_short_unit (regs[0]->unit))
1178 	{
1179 	  return NULL;
1180 	}
1181       reg_unit = regs[0]->unit;
1182     }
1183 
1184   insn->bits = (template->meta_opcode |
1185 		(regs[0]->no << 19) |
1186 		((reg_unit & SHORT_UNIT_MASK) << 3));
1187 
1188   if (!is_short_unit (addr.base_reg->unit))
1189     {
1190       as_bad (_("base unit must be one of %s"), SHORT_UNITS);
1191       return NULL;
1192     }
1193 
1194   if (addr.base_reg->no > 1)
1195     {
1196       return NULL;
1197     }
1198 
1199   insn->bits |= ((addr.base_reg->no & EXT_BASE_REG_MASK) |
1200 		 ((addr.base_reg->unit & SHORT_UNIT_MASK) << 5));
1201 
1202   if (addr.immediate)
1203     {
1204       int offset = addr.exp.X_add_number;
1205 
1206       copy_addr_reloc (insn, &addr);
1207 
1208       if (addr.negate)
1209 	offset = -offset;
1210 
1211       offset = offset / (int)size;
1212 
1213       if (!within_signed_range (offset, GET_SET_EXT_IMM_BITS))
1214 	{
1215 	  /* Parsing as a standard GET/SET provides a smaller offset.  */
1216 	  as_bad (_("offset value out of range"));
1217 	  return NULL;
1218 	}
1219 
1220       offset = offset & GET_SET_EXT_IMM_MASK;
1221 
1222       insn->bits |= (offset << 7);
1223     }
1224   else
1225     {
1226       return NULL;
1227     }
1228 
1229   insn->len = 4;
1230   return l;
1231 }
1232 
1233 /* Parse an MGET or MSET instruction addressing mode.  */
1234 static const char *
parse_mget_mset_addr(const char * line,metag_addr * addr)1235 parse_mget_mset_addr (const char *line, metag_addr *addr)
1236 {
1237   const char *l = line;
1238   const char *ll;
1239   const metag_reg *regs[1];
1240 
1241   /* Skip opening square bracket.  */
1242   l++;
1243 
1244   l = parse_gp_regs (l, regs, 1);
1245 
1246   if (l == NULL)
1247     return NULL;
1248 
1249   addr->base_reg = regs[0];
1250 
1251   ll = parse_addr_post_incr_op (l, addr);
1252 
1253   if (ll != NULL)
1254     l = ll;
1255 
1256   if (addr->negate == 1)
1257     return NULL;
1258 
1259   if (*l == ADDR_END_CHAR)
1260     {
1261       l++;
1262       return l;
1263     }
1264 
1265   return NULL;
1266 }
1267 
1268 /* Parse an MGET instruction.  */
1269 static const char *
parse_mget(const char * line,const metag_reg ** regs,metag_addr * addr,size_t * regs_read)1270 parse_mget (const char *line, const metag_reg **regs, metag_addr *addr,
1271 	    size_t *regs_read)
1272 {
1273   const char *l = line;
1274 
1275   l = parse_gp_regs_list (l, regs, MGET_MSET_MAX_REGS, regs_read);
1276 
1277   if (l == NULL ||
1278       *regs_read == 0)
1279     {
1280       as_bad (_("invalid destination register list"));
1281       return NULL;
1282     }
1283 
1284   l = skip_comma (l);
1285 
1286   if (l == NULL ||
1287       *l == END_OF_INSN)
1288     return NULL;
1289 
1290   l = parse_mget_mset_addr (l, addr);
1291 
1292   if (l == NULL)
1293     {
1294 	  as_bad (_("invalid memory operand"));
1295 	  return NULL;
1296     }
1297 
1298   return l;
1299 }
1300 
1301 /* Parse an MSET instruction.  */
1302 static const char *
parse_mset(const char * line,const metag_reg ** regs,metag_addr * addr,size_t * regs_read)1303 parse_mset (const char *line, const metag_reg **regs, metag_addr *addr,
1304 	    size_t *regs_read)
1305 {
1306   const char *l = line;
1307 
1308   l = parse_mget_mset_addr (l, addr);
1309 
1310   if (l == NULL)
1311     {
1312 	  as_bad (_("invalid memory operand"));
1313 	  return NULL;
1314     }
1315 
1316   l = skip_comma (l);
1317 
1318   if (l == NULL ||
1319       *l == END_OF_INSN)
1320     return NULL;
1321 
1322   l = parse_gp_regs_list (l, regs, MGET_MSET_MAX_REGS, regs_read);
1323 
1324   if (l == NULL ||
1325       *regs_read == 0)
1326     {
1327       as_bad (_("invalid source register list"));
1328       return NULL;
1329     }
1330 
1331   return l;
1332 }
1333 
1334 /* Take a register list REGS of size REGS_READ and convert it into an
1335    rmask value if possible. Return the rmask value in RMASK and the
1336    lowest numbered register in LOWEST_REG. Return TRUE if the conversion
1337    was successful.  */
1338 static bfd_boolean
check_rmask(const metag_reg ** regs,size_t regs_read,bfd_boolean is_fpu,bfd_boolean is_64bit,unsigned int * lowest_reg,unsigned int * rmask)1339 check_rmask (const metag_reg **regs, size_t regs_read, bfd_boolean is_fpu,
1340 	     bfd_boolean is_64bit, unsigned int *lowest_reg,
1341 	     unsigned int *rmask)
1342 {
1343   unsigned int reg_unit = regs[0]->unit;
1344   size_t i;
1345 
1346   for (i = 0; i < regs_read; i++)
1347     {
1348       if (is_fpu)
1349 	{
1350 	  if (is_64bit && regs[i]->no % 2)
1351 	    {
1352 	      as_bad (_("register list must be even numbered"));
1353 	      return FALSE;
1354 	    }
1355 	}
1356       else if (regs[i]->unit != reg_unit)
1357 	{
1358 	  as_bad (_("register list must be from the same unit"));
1359 	  return FALSE;
1360 	}
1361 
1362       if (regs[i]->no < *lowest_reg)
1363 	*lowest_reg = regs[i]->no;
1364     }
1365 
1366   for (i = 0; i < regs_read; i++)
1367     {
1368       unsigned int next_bit, next_reg;
1369       if (regs[i]->no == *lowest_reg)
1370 	continue;
1371 
1372       if (is_fpu && is_64bit)
1373 	next_reg = ((regs[i]->no / 2) - ((*lowest_reg / 2) + 1));
1374       else
1375 	next_reg = (regs[i]->no - (*lowest_reg + 1));
1376 
1377       next_bit = (1 << next_reg);
1378 
1379       if (*rmask & next_bit)
1380 	{
1381 	  as_bad (_("register list must not contain duplicates"));
1382 	  return FALSE;
1383 	}
1384 
1385       *rmask |= next_bit;
1386     }
1387 
1388   return TRUE;
1389 }
1390 
1391 /* Parse an MGET or MSET instruction.  */
1392 static const char *
parse_mget_mset(const char * line,metag_insn * insn,const insn_template * template)1393 parse_mget_mset (const char *line, metag_insn *insn,
1394 		 const insn_template *template)
1395 {
1396   const char *l = line;
1397   const metag_reg *regs[MGET_MSET_MAX_REGS];
1398   metag_addr addr;
1399   bfd_boolean is_get = MAJOR_OPCODE (template->meta_opcode) == OPC_GET;
1400   bfd_boolean is_fpu = (MINOR_OPCODE (template->meta_opcode) & 0x6) == 0x6;
1401   bfd_boolean is_64bit = (MINOR_OPCODE (template->meta_opcode) & 0x1) == 0x1;
1402   size_t regs_read = 0;
1403   unsigned int rmask = 0, reg_unit = 0, lowest_reg = 0xffffffff;
1404 
1405   memset(&addr, 0, sizeof(addr));
1406   addr.reloc_type = BFD_RELOC_UNUSED;
1407 
1408   if (is_get)
1409     {
1410       l = parse_mget (l, regs, &addr, &regs_read);
1411     }
1412   else
1413     {
1414       l = parse_mset (l, regs, &addr, &regs_read);
1415     }
1416 
1417   if (l == NULL)
1418     return NULL;
1419 
1420   if (!check_rmask (regs, regs_read, is_fpu, is_64bit, &lowest_reg, &rmask))
1421     return NULL;
1422 
1423   reg_unit = regs[0]->unit;
1424 
1425   if (is_fpu)
1426     {
1427       if (reg_unit != UNIT_FX)
1428 	return NULL;
1429 
1430       reg_unit = 0;
1431     }
1432   else if (reg_unit == UNIT_FX)
1433     return NULL;
1434 
1435   insn->bits = (template->meta_opcode |
1436 		(lowest_reg << 19) |
1437 		((reg_unit & SHORT_UNIT_MASK) << 3));
1438 
1439   if (!is_short_unit (addr.base_reg->unit))
1440     {
1441       as_bad (_("base unit must be one of %s"), SHORT_UNITS);
1442       return NULL;
1443     }
1444 
1445   insn->bits |= ((addr.base_reg->no << 14) |
1446 		 ((addr.base_reg->unit & SHORT_UNIT_MASK) << 5));
1447 
1448   insn->bits |= (rmask & RMASK_MASK) << 7;
1449 
1450   insn->len = 4;
1451   return l;
1452 }
1453 
1454 /* Parse a list of registers for MMOV pipeline prime.  */
1455 static const char *
parse_mmov_prime_list(const char * line,const metag_reg ** regs,unsigned int * rmask)1456 parse_mmov_prime_list (const char *line, const metag_reg **regs,
1457 		       unsigned int *rmask)
1458 {
1459   const char *l = line;
1460   const metag_reg *ra_regs[MMOV_MAX_REGS];
1461   size_t regs_read = 0, i;
1462   unsigned int mask = 0;
1463 
1464   l = parse_gp_regs_list (l, regs, 1, &regs_read);
1465 
1466   /* First register must be a port. */
1467   if (l == NULL || regs[0]->unit != UNIT_RD)
1468     return NULL;
1469 
1470   l = skip_comma (l);
1471 
1472   if (l == NULL)
1473     return NULL;
1474 
1475   l = parse_gp_regs_list (l, ra_regs, MMOV_MAX_REGS, &regs_read);
1476 
1477   if (l == NULL)
1478     return NULL;
1479 
1480   /* Check remaining registers match the first.
1481 
1482      Note that we also accept RA (0x10) as input for the remaining registers.
1483      Whilst this doesn't represent the instruction in any way we're stuck
1484      with it because the embedded assembler accepts it.  */
1485   for (i = 0; i < regs_read; i++)
1486     {
1487       if (ra_regs[i]->unit != UNIT_RD ||
1488 	  (ra_regs[i]->no != 0x10 && ra_regs[i]->no != regs[0]->no))
1489 	return NULL;
1490 
1491       mask = (mask << 1) | 0x1;
1492     }
1493 
1494   *rmask = mask;
1495 
1496   return l;
1497 }
1498 
1499 /* Parse a MMOV instruction.  */
1500 static const char *
parse_mmov(const char * line,metag_insn * insn,const insn_template * template)1501 parse_mmov (const char *line, metag_insn *insn,
1502 	    const insn_template *template)
1503 {
1504   const char *l = line;
1505   unsigned int is_fpu = template->insn_type == INSN_FPU;
1506   unsigned int is_prime = ((MINOR_OPCODE (template->meta_opcode) & 0x2) &&
1507 			   !is_fpu);
1508   unsigned int is_64bit = MINOR_OPCODE (template->meta_opcode) & 0x1;
1509   unsigned int rmask = 0;
1510 
1511   if (is_prime)
1512     {
1513       const metag_reg *reg;
1514       metag_addr addr;
1515 
1516       memset (&addr, 0, sizeof(addr));
1517 
1518       l = parse_mmov_prime_list (l, &reg, &rmask);
1519 
1520       if (l == NULL)
1521 	return NULL;
1522 
1523       l = skip_comma (l);
1524 
1525       if (l == NULL)
1526 	return NULL;
1527 
1528       l = parse_mget_mset_addr (l, &addr);
1529 
1530       if (l == NULL)
1531 	{
1532 	  as_bad (_("invalid memory operand"));
1533 	  return NULL;
1534 	}
1535 
1536       insn->bits = (template->meta_opcode |
1537 		    (reg->no << 19) |
1538 		    (addr.base_reg->no << 14) |
1539 		    ((rmask & RMASK_MASK) << 7) |
1540 		    ((addr.base_reg->unit & SHORT_UNIT_MASK) << 5));
1541     }
1542   else
1543     {
1544       const metag_reg *regs[MMOV_MAX_REGS + 1];
1545       unsigned int lowest_reg = 0xffffffff;
1546       size_t regs_read = 0;
1547 
1548       l = parse_gp_regs_list (l, regs, MMOV_MAX_REGS + 1, &regs_read);
1549 
1550       if (l == NULL || regs_read == 0)
1551 	return NULL;
1552 
1553       if (!is_short_unit (regs[0]->unit) &&
1554 	  !(is_fpu && regs[0]->unit == UNIT_FX))
1555 	{
1556 	  return NULL;
1557 	}
1558 
1559       if (!(regs[regs_read-1]->unit == UNIT_RD &&
1560 	    regs[regs_read-1]->no == 0))
1561 	{
1562 	  return NULL;
1563 	}
1564 
1565       if (!check_rmask (regs, regs_read - 1, is_fpu, is_64bit, &lowest_reg,
1566 			&rmask))
1567 	return NULL;
1568 
1569       if (is_fpu)
1570 	{
1571 	  insn->bits = (template->meta_opcode |
1572 			(regs[0]->no << 14) |
1573 			((rmask & RMASK_MASK) << 7));
1574 	}
1575       else
1576 	{
1577 	  insn->bits = (template->meta_opcode |
1578 			(regs[0]->no << 19) |
1579 			((rmask & RMASK_MASK) << 7) |
1580 			((regs[0]->unit & SHORT_UNIT_MASK) << 3));
1581 	}
1582     }
1583 
1584   insn->len = 4;
1585   return l;
1586 }
1587 
1588 /* Parse an immediate constant.  */
1589 static const char *
parse_imm_constant(const char * line,metag_insn * insn,int * value)1590 parse_imm_constant (const char *line, metag_insn *insn, int *value)
1591 {
1592   const char *l = line;
1593   char *save_input_line_pointer;
1594   expressionS *exp = &insn->reloc_exp;
1595 
1596   /* Skip #. */
1597   if (*l == '#')
1598     l++;
1599   else
1600     return NULL;
1601 
1602   save_input_line_pointer = input_line_pointer;
1603   input_line_pointer = (char *) l;
1604 
1605   expression (exp);
1606 
1607   l = input_line_pointer;
1608   input_line_pointer = save_input_line_pointer;
1609 
1610   if (exp->X_op == O_constant)
1611     {
1612       *value = exp->X_add_number;
1613 
1614       return l;
1615     }
1616   else
1617     {
1618       return NULL;
1619     }
1620 }
1621 
1622 /* Parse an MDRD instruction.  */
1623 static const char *
parse_mdrd(const char * line,metag_insn * insn,const insn_template * template)1624 parse_mdrd (const char *line, metag_insn *insn,
1625 	    const insn_template *template)
1626 {
1627   const char *l = line;
1628   unsigned int rmask = 0;
1629   int value = 0, i;
1630 
1631   l = parse_imm_constant (l, insn, &value);
1632 
1633   if (l == NULL)
1634     return NULL;
1635 
1636   if (value < 1 || value > 8)
1637     {
1638       as_bad (_("MDRD value must be between 1 and 8"));
1639       return NULL;
1640     }
1641 
1642   for (i = 1; i < value; i++)
1643     {
1644       rmask <<= 1;
1645       rmask |= 1;
1646     }
1647 
1648   insn->bits = (template->meta_opcode |
1649 		(rmask << 7));
1650 
1651   insn->len = 4;
1652   return l;
1653 }
1654 
1655 /* Parse a conditional SET instruction.  */
1656 static const char *
parse_cond_set(const char * line,metag_insn * insn,const insn_template * template)1657 parse_cond_set (const char *line, metag_insn *insn,
1658 		const insn_template *template)
1659 {
1660   const char *l = line;
1661   const metag_reg *regs[2];
1662   metag_addr addr;
1663   unsigned int size = metag_cond_set_size_bytes (template->meta_opcode);
1664   unsigned int reg_no;
1665 
1666   memset(&addr, 0, sizeof(addr));
1667   addr.reloc_type = BFD_RELOC_UNUSED;
1668 
1669   l = parse_set (l, regs, &addr, size);
1670 
1671   if (l == NULL)
1672     return NULL;
1673 
1674   if (regs[0]->unit == UNIT_RD)
1675     {
1676       if (regs[0]->no != 0)
1677 	{
1678 	  as_bad (_("set can only use RD port as source"));
1679 	  return NULL;
1680 	}
1681       reg_no = 16;
1682     }
1683   else
1684     reg_no = regs[0]->no;
1685 
1686   if (addr.update)
1687     return NULL;
1688 
1689   if (!(addr.immediate &&
1690 	addr.exp.X_add_number == 0))
1691     return NULL;
1692 
1693   insn->bits = (template->meta_opcode |
1694 		(reg_no << 19) |
1695 		(regs[0]->unit << 10));
1696 
1697   if (!is_short_unit (addr.base_reg->unit))
1698     {
1699       as_bad (_("base unit must be one of %s"), SHORT_UNITS);
1700       return NULL;
1701     }
1702 
1703   insn->bits |= ((addr.base_reg->no << 14) |
1704 		 ((addr.base_reg->unit & SHORT_UNIT_MASK) << 5));
1705 
1706   insn->len = 4;
1707   return l;
1708 }
1709 
1710 /* Parse an XFR instruction.  */
1711 static const char *
parse_xfr(const char * line,metag_insn * insn,const insn_template * template)1712 parse_xfr (const char *line, metag_insn *insn,
1713 	   const insn_template *template)
1714 {
1715   const char *l = line;
1716   metag_addr dest_addr, src_addr;
1717   unsigned int size = 4;
1718 
1719   memset(&dest_addr, 0, sizeof(dest_addr));
1720   memset(&src_addr, 0, sizeof(src_addr));
1721   dest_addr.reloc_type = BFD_RELOC_UNUSED;
1722   src_addr.reloc_type = BFD_RELOC_UNUSED;
1723 
1724   l = parse_addr (l, &dest_addr, size);
1725 
1726   if (l == NULL ||
1727       dest_addr.immediate == 1)
1728     {
1729 	  as_bad (_("invalid destination memory operand"));
1730 	  return NULL;
1731     }
1732 
1733   l = skip_comma (l);
1734 
1735   if (l == NULL ||
1736       *l == END_OF_INSN)
1737     return NULL;
1738 
1739   l = parse_addr (l, &src_addr, size);
1740 
1741   if (l == NULL ||
1742       src_addr.immediate == 1)
1743     {
1744 	  as_bad (_("invalid source memory operand"));
1745 	  return NULL;
1746     }
1747 
1748   if (!is_short_unit (dest_addr.base_reg->unit) ||
1749       !is_short_unit (src_addr.base_reg->unit))
1750     {
1751       as_bad (_("address units must be one of %s"), SHORT_UNITS);
1752       return NULL;
1753     }
1754 
1755   if ((dest_addr.base_reg->unit != dest_addr.offset_reg->unit) ||
1756       (src_addr.base_reg->unit != src_addr.offset_reg->unit))
1757     {
1758       as_bad (_("base and offset must be from the same unit"));
1759       return NULL;
1760     }
1761 
1762   if (dest_addr.update == 1 &&
1763       src_addr.update == 1 &&
1764       dest_addr.post_increment != src_addr.post_increment)
1765     {
1766       as_bad (_("source and destination increment mode must agree"));
1767       return NULL;
1768     }
1769 
1770   insn->bits = (template->meta_opcode |
1771 		(src_addr.base_reg->no << 19) |
1772 		(src_addr.offset_reg->no << 14) |
1773 		((src_addr.base_reg->unit & SHORT_UNIT_MASK) << 2));
1774 
1775   insn->bits |= ((dest_addr.base_reg->no << 9) |
1776 		 (dest_addr.offset_reg->no << 4) |
1777 		 ((dest_addr.base_reg->unit & SHORT_UNIT_MASK)));
1778 
1779   if (dest_addr.update == 1)
1780     insn->bits |= (1 << 26);
1781 
1782   if (src_addr.update == 1)
1783     insn->bits |= (1 << 27);
1784 
1785   if (dest_addr.post_increment == 1 ||
1786       src_addr.post_increment == 1)
1787     insn->bits |= (1 << 24);
1788 
1789   insn->len = 4;
1790   return l;
1791 }
1792 
1793 /* Parse an 8bit immediate value.  */
1794 static const char *
parse_imm8(const char * line,metag_insn * insn,int * value)1795 parse_imm8 (const char *line, metag_insn *insn, int *value)
1796 {
1797   const char *l = line;
1798   char *save_input_line_pointer;
1799   expressionS *exp = &insn->reloc_exp;
1800 
1801   /* Skip #. */
1802   if (*l == '#')
1803     l++;
1804   else
1805     return NULL;
1806 
1807   save_input_line_pointer = input_line_pointer;
1808   input_line_pointer = (char *) l;
1809 
1810   expression (exp);
1811 
1812   l = input_line_pointer;
1813   input_line_pointer = save_input_line_pointer;
1814 
1815   if (exp->X_op == O_absent || exp->X_op == O_big)
1816     {
1817       return NULL;
1818     }
1819   else if (exp->X_op == O_constant)
1820     {
1821       *value = exp->X_add_number;
1822     }
1823   else
1824     {
1825       insn->reloc_type = BFD_RELOC_METAG_REL8;
1826       insn->reloc_pcrel = 0;
1827     }
1828 
1829   return l;
1830 }
1831 
1832 /* Parse a 16bit immediate value.  */
1833 static const char *
parse_imm16(const char * line,metag_insn * insn,int * value)1834 parse_imm16 (const char *line, metag_insn *insn, int *value)
1835 {
1836   const char *l = line;
1837   char *save_input_line_pointer;
1838   expressionS *exp = &insn->reloc_exp;
1839   bfd_boolean is_hi = FALSE;
1840   bfd_boolean is_lo = FALSE;
1841 
1842   /* Skip #. */
1843   if (*l == '#')
1844     l++;
1845   else
1846     return NULL;
1847 
1848   if (strncasecmp (l, "HI", 2) == 0)
1849     {
1850       is_hi = TRUE;
1851       l += 2;
1852     }
1853   else if (strncasecmp (l, "LO", 2) == 0)
1854     {
1855       is_lo = TRUE;
1856       l += 2;
1857     }
1858 
1859   save_input_line_pointer = input_line_pointer;
1860   input_line_pointer = (char *) l;
1861 
1862   expression (exp);
1863 
1864   l = input_line_pointer;
1865   input_line_pointer = save_input_line_pointer;
1866 
1867   if (exp->X_op == O_absent || exp->X_op == O_big)
1868     {
1869       return NULL;
1870     }
1871   else if (exp->X_op == O_constant)
1872     {
1873       if (is_hi)
1874 	*value = (exp->X_add_number >> 16) & IMM16_MASK;
1875       else if (is_lo)
1876 	*value = exp->X_add_number & IMM16_MASK;
1877       else
1878 	*value = exp->X_add_number;
1879     }
1880   else
1881     {
1882       if (exp->X_op == O_PIC_reloc)
1883 	{
1884 	  exp->X_op = O_symbol;
1885 
1886 	  if (exp->X_md == BFD_RELOC_METAG_GOTOFF)
1887 	    {
1888 	      if (is_hi)
1889 		insn->reloc_type = BFD_RELOC_METAG_HI16_GOTOFF;
1890 	      else if (is_lo)
1891 		insn->reloc_type = BFD_RELOC_METAG_LO16_GOTOFF;
1892 	      else
1893 		return NULL;
1894 	    }
1895 	  else if (exp->X_md == BFD_RELOC_METAG_PLT)
1896 	    {
1897 	      if (is_hi)
1898 		insn->reloc_type = BFD_RELOC_METAG_HI16_PLT;
1899 	      else if (is_lo)
1900 		insn->reloc_type = BFD_RELOC_METAG_LO16_PLT;
1901 	      else
1902 		return NULL;
1903 	    }
1904 	  else if (exp->X_md == BFD_RELOC_METAG_TLS_LDO)
1905 	    {
1906 	      if (is_hi)
1907 		insn->reloc_type = BFD_RELOC_METAG_TLS_LDO_HI16;
1908 	      else if (is_lo)
1909 		insn->reloc_type = BFD_RELOC_METAG_TLS_LDO_LO16;
1910 	      else
1911 		return NULL;
1912 	    }
1913 	  else if (exp->X_md == BFD_RELOC_METAG_TLS_IENONPIC)
1914 	    {
1915 	      if (is_hi)
1916 		insn->reloc_type = BFD_RELOC_METAG_TLS_IENONPIC_HI16;
1917 	      else if (is_lo)
1918 		insn->reloc_type = BFD_RELOC_METAG_TLS_IENONPIC_LO16;
1919 	      else
1920 		return NULL;
1921 	    }
1922 	  else if (exp->X_md == BFD_RELOC_METAG_TLS_LE)
1923 	    {
1924 	      if (is_hi)
1925 		insn->reloc_type = BFD_RELOC_METAG_TLS_LE_HI16;
1926 	      else if (is_lo)
1927 		insn->reloc_type = BFD_RELOC_METAG_TLS_LE_LO16;
1928 	      else
1929 		return NULL;
1930 	    }
1931 	  else if (exp->X_md == BFD_RELOC_METAG_TLS_GD ||
1932 		   exp->X_md == BFD_RELOC_METAG_TLS_LDM)
1933 	    insn->reloc_type = exp->X_md;
1934 	}
1935       else
1936 	{
1937 	  if (exp->X_op == O_symbol && exp->X_add_symbol == GOT_symbol)
1938 	    {
1939 	      if (is_hi)
1940 		insn->reloc_type = BFD_RELOC_METAG_HI16_GOTPC;
1941 	      else if (is_lo)
1942 		insn->reloc_type = BFD_RELOC_METAG_LO16_GOTPC;
1943 	      else
1944 		return NULL;
1945 	    }
1946 	  else
1947 	    {
1948 	      if (is_hi)
1949 		insn->reloc_type = BFD_RELOC_METAG_HIADDR16;
1950 	      else if (is_lo)
1951 		insn->reloc_type = BFD_RELOC_METAG_LOADDR16;
1952 	      else
1953 		insn->reloc_type = BFD_RELOC_METAG_REL16;
1954 	    }
1955 	}
1956 
1957       insn->reloc_pcrel = 0;
1958     }
1959 
1960   return l;
1961 }
1962 
1963 /* Parse a MOV to control unit instruction.  */
1964 static const char *
parse_mov_ct(const char * line,metag_insn * insn,const insn_template * template)1965 parse_mov_ct (const char *line, metag_insn *insn,
1966 	      const insn_template *template)
1967 {
1968   const char *l = line;
1969   const metag_reg *regs[1];
1970   unsigned int top = template->meta_opcode & 0x1;
1971   unsigned int is_trace = (template->meta_opcode >> 2) & 0x1;
1972   unsigned int sign_extend = 0;
1973   int value = 0;
1974 
1975   l = parse_gp_regs (l, regs, 1);
1976 
1977   if (l == NULL)
1978     return NULL;
1979 
1980   if (is_trace)
1981     {
1982       if (regs[0]->unit != UNIT_TT)
1983 	return NULL;
1984     }
1985   else
1986     {
1987       if (regs[0]->unit != UNIT_CT)
1988 	return NULL;
1989     }
1990 
1991   l = skip_comma (l);
1992 
1993   if (l == NULL ||
1994       *l == END_OF_INSN)
1995     return NULL;
1996 
1997   l = parse_imm16 (l, insn, &value);
1998 
1999   if (l == NULL)
2000     return NULL;
2001 
2002   if (value < 0)
2003     sign_extend = 1;
2004 
2005   insn->bits = (template->meta_opcode |
2006 		(regs[0]->no << 19) |
2007 		((value & IMM16_MASK) << 3));
2008 
2009   if (sign_extend == 1 && top == 0)
2010     insn->bits |= (1 << 1);
2011 
2012   insn->len = 4;
2013   return l;
2014 }
2015 
2016 /* Parse a SWAP instruction.  */
2017 static const char *
parse_swap(const char * line,metag_insn * insn,const insn_template * template)2018 parse_swap (const char *line, metag_insn *insn,
2019 	    const insn_template *template)
2020 {
2021   const char *l = line;
2022   const metag_reg *regs[2];
2023 
2024   l = parse_gp_regs (l, regs, 2);
2025 
2026   if (l == NULL)
2027     return NULL;
2028 
2029   /* PC.r | CT.r | TR.r | TT.r are treated as if they are a single unit.  */
2030   switch (regs[0]->unit)
2031     {
2032     case UNIT_PC:
2033     case UNIT_CT:
2034     case UNIT_TR:
2035     case UNIT_TT:
2036       if (regs[1]->unit == UNIT_PC
2037 	  || regs[1]->unit == UNIT_CT
2038 	  || regs[1]->unit == UNIT_TR
2039 	  || regs[1]->unit == UNIT_TT)
2040 	{
2041 	  as_bad (_("PC, CT, TR and TT are treated as if they are a single unit but operands must be in different units"));
2042 	  return NULL;
2043 	}
2044 
2045     default:
2046       /* Registers must be in different units.  */
2047       if (regs[0]->unit == regs[1]->unit)
2048 	{
2049 	  as_bad (_("source and destination register must be in different units"));
2050 	  return NULL;
2051 	}
2052       break;
2053     }
2054 
2055   insn->bits = (template->meta_opcode
2056 		| (regs[1]->no << 19)
2057 		| (regs[0]->no << 14)
2058 		| (regs[1]->unit << 10)
2059 		| (regs[0]->unit << 5));
2060 
2061   insn->len = 4;
2062   return l;
2063 }
2064 
2065 /* Parse a JUMP instruction.  */
2066 static const char *
parse_jump(const char * line,metag_insn * insn,const insn_template * template)2067 parse_jump (const char *line, metag_insn *insn,
2068 	    const insn_template *template)
2069 {
2070   const char *l = line;
2071   const metag_reg *regs[1];
2072   int value = 0;
2073 
2074   l = parse_gp_regs (l, regs, 1);
2075 
2076   if (l == NULL)
2077     return NULL;
2078 
2079   if (!is_short_unit (regs[0]->unit))
2080     {
2081       as_bad (_("register unit must be one of %s"), SHORT_UNITS);
2082       return FALSE;
2083     }
2084 
2085   l = skip_comma (l);
2086 
2087   if (l == NULL ||
2088       *l == END_OF_INSN)
2089     return NULL;
2090 
2091   l = parse_imm16 (l, insn, &value);
2092 
2093   if (l == NULL)
2094     return NULL;
2095 
2096   insn->bits = (template->meta_opcode |
2097 		(regs[0]->no << 19) |
2098 		(regs[0]->unit & SHORT_UNIT_MASK) |
2099 		((value & IMM16_MASK) << 3));
2100 
2101   insn->len = 4;
2102   return l;
2103 }
2104 
2105 /* Parse a 19bit immediate value.  */
2106 static const char *
parse_imm19(const char * line,metag_insn * insn,int * value)2107 parse_imm19 (const char *line, metag_insn *insn, int *value)
2108 {
2109   const char *l = line;
2110   char *save_input_line_pointer;
2111   expressionS *exp = &insn->reloc_exp;
2112 
2113   /* Skip #.  */
2114   if (*l == '#')
2115     l++;
2116 
2117   save_input_line_pointer = input_line_pointer;
2118   input_line_pointer = (char *) l;
2119 
2120   expression (exp);
2121 
2122   l = input_line_pointer;
2123   input_line_pointer = save_input_line_pointer;
2124 
2125   if (exp->X_op == O_absent || exp->X_op == O_big)
2126     {
2127       return NULL;
2128     }
2129   else if (exp->X_op == O_constant)
2130     {
2131       *value = exp->X_add_number;
2132     }
2133   else
2134     {
2135       if (exp->X_op == O_PIC_reloc)
2136 	{
2137 	  exp->X_op = O_symbol;
2138 
2139 	  if (exp->X_md == BFD_RELOC_METAG_PLT)
2140 	    insn->reloc_type = BFD_RELOC_METAG_RELBRANCH_PLT;
2141 	  else
2142 	    return NULL;
2143 	}
2144       else
2145 	insn->reloc_type = BFD_RELOC_METAG_RELBRANCH;
2146       insn->reloc_pcrel = 1;
2147     }
2148 
2149   return l;
2150 }
2151 
2152 /* Parse a CALLR instruction.  */
2153 static const char *
parse_callr(const char * line,metag_insn * insn,const insn_template * template)2154 parse_callr (const char *line, metag_insn *insn,
2155 	     const insn_template *template)
2156 {
2157   const char *l = line;
2158   const metag_reg *regs[1];
2159   int value = 0;
2160 
2161   l = parse_gp_regs (l, regs, 1);
2162 
2163   if (l == NULL)
2164     return NULL;
2165 
2166   if (!is_short_unit (regs[0]->unit))
2167     {
2168       as_bad (_("link register unit must be one of %s"), SHORT_UNITS);
2169       return NULL;
2170     }
2171 
2172   if (regs[0]->no & ~CALLR_REG_MASK)
2173     {
2174       as_bad (_("link register must be in a low numbered register"));
2175       return NULL;
2176     }
2177 
2178   l = skip_comma (l);
2179 
2180   if (l == NULL ||
2181       *l == END_OF_INSN)
2182     return NULL;
2183 
2184   l = parse_imm19 (l, insn, &value);
2185 
2186   if (l == NULL)
2187     return NULL;
2188 
2189   if (!within_signed_range (value / 4, IMM19_BITS))
2190     {
2191       as_bad (_("target out of range"));
2192       return NULL;
2193     }
2194 
2195   insn->bits = (template->meta_opcode |
2196 		(regs[0]->no & CALLR_REG_MASK) |
2197 		((regs[0]->unit & SHORT_UNIT_MASK) << 3) |
2198 		((value & IMM19_MASK) << 5));
2199 
2200   insn->len = 4;
2201   return l;
2202 }
2203 
2204 /* Return the value for the register field if we apply the O2R modifier
2205    to operand 2 REG, combined with UNIT_BIT derived from the destination
2206    register or source1. Uses address unit O2R if IS_ADDR is set.  */
2207 static int
lookup_o2r(unsigned int is_addr,unsigned int unit_bit,const metag_reg * reg)2208 lookup_o2r (unsigned int is_addr, unsigned int unit_bit, const metag_reg *reg)
2209 {
2210   if (reg->no & ~O2R_REG_MASK)
2211     return -1;
2212 
2213   if (is_addr)
2214     {
2215       if (unit_bit)
2216 	{
2217 	  switch (reg->unit)
2218 	    {
2219 	    case UNIT_D1:
2220 	      return reg->no;
2221 	    case UNIT_D0:
2222 	      return (1 << 3) | reg->no;
2223 	    case UNIT_RD:
2224 	      return (2 << 3) | reg->no;
2225 	    case UNIT_A0:
2226 	      return (3 << 3) | reg->no;
2227 	    default:
2228 	      return -1;
2229 	    }
2230 	}
2231       else
2232 	{
2233 	  switch (reg->unit)
2234 	    {
2235 	    case UNIT_A1:
2236 	      return reg->no;
2237 	    case UNIT_D0:
2238 	      return (1 << 3) | reg->no;
2239 	    case UNIT_RD:
2240 	      return (2 << 3) | reg->no;
2241 	    case UNIT_D1:
2242 	      return (3 << 3) | reg->no;
2243 	    default:
2244 	      return -1;
2245 	    }
2246 	}
2247     }
2248   else
2249     {
2250       if (unit_bit)
2251 	{
2252 	  switch (reg->unit)
2253 	    {
2254 	    case UNIT_A1:
2255 	      return reg->no;
2256 	    case UNIT_D0:
2257 	      return (1 << 3) | reg->no;
2258 	    case UNIT_RD:
2259 	      return (2 << 3) | reg->no;
2260 	    case UNIT_A0:
2261 	      return (3 << 3) | reg->no;
2262 	    default:
2263 	      return -1;
2264 	    }
2265 	}
2266       else
2267 	{
2268 	  switch (reg->unit)
2269 	    {
2270 	    case UNIT_A1:
2271 	      return reg->no;
2272 	    case UNIT_D1:
2273 	      return (1 << 3) | reg->no;
2274 	    case UNIT_RD:
2275 	      return (2 << 3) | reg->no;
2276 	    case UNIT_A0:
2277 	      return (3 << 3) | reg->no;
2278 	    default:
2279 	      return -1;
2280 	    }
2281 	}
2282     }
2283 }
2284 
2285 /* Parse GP ALU instruction.  */
2286 static const char *
parse_alu(const char * line,metag_insn * insn,const insn_template * template)2287 parse_alu (const char *line, metag_insn *insn,
2288 	   const insn_template *template)
2289 {
2290   const char *l = line;
2291   const metag_reg *dest_regs[1];
2292   const metag_reg *src_regs[2];
2293   int value = 0;
2294   unsigned int o1z = 0;
2295   unsigned int imm = (template->meta_opcode >> 25) & 0x1;
2296   unsigned int cond = (template->meta_opcode >> 26) & 0x1;
2297   unsigned int ca = (template->meta_opcode >> 5) & 0x1;
2298   unsigned int top = template->meta_opcode & 0x1;
2299   unsigned int sign_extend = 0;
2300   unsigned int is_addr_op = MAJOR_OPCODE (template->meta_opcode) == OPC_ADDR;
2301   unsigned int is_mul = MAJOR_OPCODE (template->meta_opcode) == OPC_MUL;
2302   unsigned int unit_bit = 0;
2303   bfd_boolean is_quickrot = template->arg_type & GP_ARGS_QR;
2304 
2305   l = parse_gp_regs (l, dest_regs, 1);
2306 
2307   if (l == NULL)
2308     return NULL;
2309 
2310   l = skip_comma (l);
2311 
2312   if (l == NULL ||
2313       *l == END_OF_INSN)
2314     return NULL;
2315 
2316   if (is_addr_op)
2317     {
2318       if (dest_regs[0]->unit == UNIT_A0)
2319 	unit_bit = 0;
2320       else if (dest_regs[0]->unit == UNIT_A1)
2321 	unit_bit = 1;
2322     }
2323   else
2324     {
2325       if (dest_regs[0]->unit == UNIT_D0)
2326 	unit_bit = 0;
2327       else if (dest_regs[0]->unit == UNIT_D1)
2328 	unit_bit = 1;
2329     }
2330 
2331   if ((MAJOR_OPCODE (template->meta_opcode) == OPC_ADDR ||
2332       MAJOR_OPCODE (template->meta_opcode) == OPC_ADD ||
2333        MAJOR_OPCODE (template->meta_opcode) == OPC_SUB) &&
2334       ((template->meta_opcode >> 2) & 0x1))
2335     o1z = 1;
2336 
2337   if (imm)
2338     {
2339       if (!cond)
2340 	{
2341 	  if (is_addr_op)
2342 	    {
2343 	      if (dest_regs[0]->unit == UNIT_A0)
2344 		unit_bit = 0;
2345 	      else if (dest_regs[0]->unit == UNIT_A1)
2346 		unit_bit = 1;
2347 	      else
2348 		return NULL;
2349 	    }
2350 	  else
2351 	    {
2352 	      if (dest_regs[0]->unit == UNIT_D0)
2353 		unit_bit = 0;
2354 	      else if (dest_regs[0]->unit == UNIT_D1)
2355 		unit_bit = 1;
2356 	      else
2357 		return NULL;
2358 	    }
2359 	}
2360 
2361       if (cond)
2362 	{
2363 	  l = parse_gp_regs (l, src_regs, 1);
2364 
2365 	  if (l == NULL)
2366 	    return NULL;
2367 
2368 	  l = skip_comma (l);
2369 
2370 	  if (l == NULL ||
2371 	      *l == END_OF_INSN)
2372 	    return NULL;
2373 
2374 	  if (is_addr_op)
2375 	    {
2376 	      if (src_regs[0]->unit == UNIT_A0)
2377 		unit_bit = 0;
2378 	      else if (src_regs[0]->unit == UNIT_A1)
2379 		unit_bit = 1;
2380 	      else
2381 		return NULL;
2382 	    }
2383 	  else
2384 	    {
2385 	      if (src_regs[0]->unit == UNIT_D0)
2386 		unit_bit = 0;
2387 	      else if (src_regs[0]->unit == UNIT_D1)
2388 		unit_bit = 1;
2389 	      else
2390 		return NULL;
2391 	    }
2392 
2393 	  if (src_regs[0]->unit != dest_regs[0]->unit && !ca)
2394 	    return NULL;
2395 
2396 	  l = parse_imm8 (l, insn, &value);
2397 
2398 	  if (l == NULL)
2399 	    return NULL;
2400 
2401 	  if (!within_unsigned_range (value, IMM8_BITS))
2402 	    return NULL;
2403 
2404 	  insn->bits = (template->meta_opcode |
2405 			(dest_regs[0]->no << 19) |
2406 			(src_regs[0]->no << 14) |
2407 			((value & IMM8_MASK) << 6));
2408 
2409 	  if (ca)
2410 	    {
2411 	      if (is_addr_op)
2412 		{
2413 		  if (src_regs[0]->unit == UNIT_A0)
2414 		    unit_bit = 0;
2415 		  else if (src_regs[0]->unit == UNIT_A1)
2416 		    unit_bit = 1;
2417 		  else
2418 		    return NULL;
2419 		}
2420 	      else
2421 		{
2422 		  if (src_regs[0]->unit == UNIT_D0)
2423 		    unit_bit = 0;
2424 		  else if (src_regs[0]->unit == UNIT_D1)
2425 		    unit_bit = 1;
2426 		  else
2427 		    return NULL;
2428 		}
2429 
2430 	      insn->bits |= dest_regs[0]->unit << 1;
2431 	    }
2432 	}
2433       else if (o1z)
2434 	{
2435 	  l = parse_imm16 (l, insn, &value);
2436 
2437 	  if (l == NULL)
2438 	    return NULL;
2439 
2440 	  if (value < 0)
2441 	    {
2442 	      if (!within_signed_range (value, IMM16_BITS))
2443 		{
2444 		  as_bad (_("immediate out of range"));
2445 		  return NULL;
2446 		}
2447 	      sign_extend = 1;
2448 	    }
2449 	  else
2450 	    {
2451 	      if (!within_unsigned_range (value, IMM16_BITS))
2452 		{
2453 		  as_bad (_("immediate out of range"));
2454 		  return NULL;
2455 		}
2456 	    }
2457 
2458 	  insn->bits = (template->meta_opcode |
2459 			(dest_regs[0]->no << 19) |
2460 			((value & IMM16_MASK) << 3));
2461 	}
2462       else
2463 	{
2464 	  l = parse_gp_regs (l, src_regs, 1);
2465 
2466 	  if (l == NULL)
2467 	    return NULL;
2468 
2469 	  if (!(src_regs[0]->unit == dest_regs[0]->unit))
2470 	    return NULL;
2471 
2472 	  /* CPC is valid for address ops. */
2473 	  if (src_regs[0]->no != dest_regs[0]->no &&
2474 	      !(is_addr_op && src_regs[0]->no == 0x10))
2475 	    return NULL;
2476 
2477 	  l = skip_comma (l);
2478 
2479 	  if (l == NULL ||
2480 	      *l == END_OF_INSN)
2481 	    return NULL;
2482 
2483 	  l = parse_imm16 (l, insn, &value);
2484 
2485 	  if (l == NULL)
2486 	    return NULL;
2487 
2488 	  if (value < 0)
2489 	    {
2490 	      if (!within_signed_range (value, IMM16_BITS))
2491 		{
2492 		  as_bad (_("immediate out of range"));
2493 		  return NULL;
2494 		}
2495 	      sign_extend = 1;
2496 	    }
2497 	  else
2498 	    {
2499 	      if (!within_unsigned_range (value, IMM16_BITS))
2500 		{
2501 		  as_bad (_("immediate out of range"));
2502 		  return NULL;
2503 		}
2504 	    }
2505 
2506 	  insn->bits = (template->meta_opcode |
2507 			(dest_regs[0]->no << 19) |
2508 			(src_regs[0]->no << 19) |
2509 			((value & IMM16_MASK) << 3));
2510 	}
2511     }
2512   else
2513     {
2514       unsigned int o2r = 0;
2515       int rs2;
2516 
2517       if (cond || !o1z)
2518 	l = parse_gp_regs (l, src_regs, 2);
2519       else
2520 	l = parse_gp_regs (l, src_regs, 1);
2521 
2522       if (l == NULL)
2523 	return NULL;
2524 
2525       if (cond || !o1z)
2526 	{
2527 	  if (is_addr_op)
2528 	    {
2529 	      if (src_regs[0]->unit == UNIT_A0)
2530 		unit_bit = 0;
2531 	      else if (src_regs[0]->unit == UNIT_A1)
2532 		unit_bit = 1;
2533 	      else
2534 		return NULL;
2535 	    }
2536 	  else
2537 	    {
2538 	      if (src_regs[0]->unit == UNIT_D0)
2539 		unit_bit = 0;
2540 	      else if (src_regs[0]->unit == UNIT_D1)
2541 		unit_bit = 1;
2542 	      else
2543 		return NULL;
2544 	    }
2545 	}
2546       else
2547 	{
2548 	  if (is_addr_op)
2549 	    {
2550 	      if (dest_regs[0]->unit == UNIT_A0)
2551 		unit_bit = 0;
2552 	      else if (dest_regs[0]->unit == UNIT_A1)
2553 		unit_bit = 1;
2554 	      else
2555 		return NULL;
2556 	    }
2557 	  else
2558 	    {
2559 	      if (dest_regs[0]->unit == UNIT_D0)
2560 		unit_bit = 0;
2561 	      else if (dest_regs[0]->unit == UNIT_D1)
2562 		unit_bit = 1;
2563 	      else
2564 		return NULL;
2565 	    }
2566 	}
2567 
2568       if (cond)
2569 	{
2570 	  if (src_regs[0]->unit != src_regs[1]->unit)
2571 	    {
2572 	      rs2 = lookup_o2r (is_addr_op, unit_bit, src_regs[1]);
2573 
2574 	      if (rs2 < 0)
2575 		return NULL;
2576 
2577 	      o2r = 1;
2578 	    }
2579 	  else
2580 	    {
2581 	      rs2 = src_regs[1]->no;
2582 	    }
2583 
2584 	  insn->bits = (template->meta_opcode |
2585 			(dest_regs[0]->no << 19) |
2586 			(src_regs[0]->no << 14) |
2587 			(rs2 << 9));
2588 
2589 	  if (is_mul)
2590 	    {
2591 	      if (dest_regs[0]->unit != src_regs[0]->unit && is_mul)
2592 		{
2593 		  if (ca)
2594 		    {
2595 		      insn->bits |= dest_regs[0]->unit << 1;
2596 		    }
2597 		  else
2598 		    return NULL;
2599 		}
2600 	    }
2601 	  else
2602 	    insn->bits |= dest_regs[0]->unit << 5;
2603 	}
2604       else if (o1z)
2605 	{
2606 	  if (dest_regs[0]->unit != src_regs[0]->unit)
2607 	    {
2608 	      rs2 = lookup_o2r (is_addr_op, unit_bit, src_regs[0]);
2609 
2610 	      if (rs2 < 0)
2611 		return NULL;
2612 
2613 	      o2r = 1;
2614 	    }
2615 	  else
2616 	    {
2617 	      rs2 = src_regs[0]->no;
2618 	    }
2619 
2620 	  insn->bits = (template->meta_opcode |
2621 			(dest_regs[0]->no << 19) |
2622 			(rs2 << 9));
2623 	}
2624       else
2625 	{
2626 	  if (dest_regs[0]->unit != src_regs[0]->unit)
2627 	    return NULL;
2628 
2629 	  if (dest_regs[0]->unit != src_regs[1]->unit)
2630 	    {
2631 	      rs2 = lookup_o2r (is_addr_op, unit_bit, src_regs[1]);
2632 
2633 	      if (rs2 < 0)
2634 		return NULL;
2635 
2636 	      o2r = 1;
2637 	    }
2638 	  else
2639 	    {
2640 	      rs2 = src_regs[1]->no;
2641 	    }
2642 
2643 	  insn->bits = (template->meta_opcode |
2644 			(dest_regs[0]->no << 19) |
2645 			(src_regs[0]->no << 14) |
2646 			(rs2 << 9));
2647 	}
2648 
2649       if (o2r)
2650 	insn->bits |= 1;
2651     }
2652 
2653   if (is_quickrot)
2654     {
2655       const metag_reg *qr_regs[1];
2656       bfd_boolean limit_regs = imm && cond;
2657 
2658       l = skip_comma (l);
2659 
2660       if (l == NULL ||
2661 	  *l == END_OF_INSN)
2662 	return NULL;
2663 
2664       l = parse_gp_regs (l, qr_regs, 1);
2665 
2666       if (l == NULL)
2667 	return NULL;
2668 
2669       if (!((unit_bit == 0 && qr_regs[0]->unit != UNIT_A0) ||
2670 	    !(unit_bit == 1 && qr_regs[0]->unit != UNIT_A1)))
2671 	{
2672 	  as_bad (_("invalid quickrot unit specified"));
2673 	  return NULL;
2674 	}
2675 
2676       switch (qr_regs[0]->no)
2677 	{
2678 	case 2:
2679 	  break;
2680 	case 3:
2681 	  if (!limit_regs)
2682 	    {
2683 	      insn->bits |= (1 << 7);
2684 	      break;
2685 	    }
2686 	default:
2687 	  as_bad (_("invalid quickrot register specified"));
2688 	  return NULL;
2689 	}
2690     }
2691 
2692   if (sign_extend == 1 && top == 0)
2693     insn->bits |= (1 << 1);
2694 
2695   insn->bits |= unit_bit << 24;
2696   insn->len = 4;
2697   return l;
2698 }
2699 
2700 /* Parse a B instruction.  */
2701 static const char *
parse_branch(const char * line,metag_insn * insn,const insn_template * template)2702 parse_branch (const char *line, metag_insn *insn,
2703 	      const insn_template *template)
2704 {
2705   const char *l = line;
2706   int value = 0;
2707 
2708   l = parse_imm19 (l, insn, &value);
2709 
2710   if (l == NULL)
2711     return NULL;
2712 
2713   if (!within_signed_range (value / 4, IMM19_BITS))
2714     {
2715       as_bad (_("target out of range"));
2716       return NULL;
2717     }
2718 
2719   insn->bits = (template->meta_opcode |
2720 		((value & IMM19_MASK) << 5));
2721 
2722   insn->len = 4;
2723   return l;
2724 }
2725 
2726 /* Parse a KICK instruction.  */
2727 static const char *
parse_kick(const char * line,metag_insn * insn,const insn_template * template)2728 parse_kick (const char *line, metag_insn *insn,
2729 	    const insn_template *template)
2730 {
2731   const char *l = line;
2732   const metag_reg *regs[2];
2733 
2734   l = parse_gp_regs (l, regs, 2);
2735 
2736   if (l == NULL)
2737     return NULL;
2738 
2739   if (regs[1]->unit != UNIT_TR)
2740     {
2741       as_bad (_("source register must be in the trigger unit"));
2742       return NULL;
2743     }
2744 
2745   insn->bits = (template->meta_opcode |
2746 		(regs[1]->no << 19) |
2747 		(regs[0]->no << 14) |
2748 		(regs[0]->unit << 5));
2749 
2750   insn->len = 4;
2751   return l;
2752 }
2753 
2754 /* Parse a SWITCH instruction.  */
2755 static const char *
parse_switch(const char * line,metag_insn * insn,const insn_template * template)2756 parse_switch (const char *line, metag_insn *insn,
2757 	      const insn_template *template)
2758 {
2759   const char *l = line;
2760   int value = 0;
2761 
2762   l = parse_imm_constant (l, insn, &value);
2763 
2764   if (l == NULL)
2765     return NULL;
2766 
2767   if (!within_unsigned_range (value, IMM24_BITS))
2768     {
2769       as_bad (_("target out of range"));
2770       return NULL;
2771     }
2772 
2773   insn->bits = (template->meta_opcode |
2774 		(value & IMM24_MASK));
2775 
2776   insn->len = 4;
2777   return l;
2778 }
2779 
2780 /* Parse a shift instruction.  */
2781 static const char *
parse_shift(const char * line,metag_insn * insn,const insn_template * template)2782 parse_shift (const char *line, metag_insn *insn,
2783 	     const insn_template *template)
2784 {
2785   const char *l = line;
2786   const metag_reg *regs[2];
2787   const metag_reg *src2_regs[1];
2788   int value = 0;
2789   unsigned int cond = (template->meta_opcode >> 26) & 0x1;
2790   unsigned int ca = (template->meta_opcode >> 5) & 0x1;
2791   unsigned int unit_bit = 0;
2792 
2793   l = parse_gp_regs (l, regs, 2);
2794 
2795   if (l == NULL)
2796     return NULL;
2797 
2798   l = skip_comma (l);
2799 
2800   if (l == NULL ||
2801       *l == END_OF_INSN)
2802     return NULL;
2803 
2804   if (regs[1]->unit == UNIT_D0)
2805     unit_bit = 0;
2806   else if (regs[1]->unit == UNIT_D1)
2807     unit_bit = 1;
2808   else
2809     return NULL;
2810 
2811   if (regs[0]->unit != regs[1]->unit && !(cond && ca))
2812     return NULL;
2813 
2814   if (*l == '#')
2815     {
2816       l = parse_imm_constant (l, insn, &value);
2817 
2818       if (l == NULL)
2819 	return NULL;
2820 
2821       if (!within_unsigned_range (value, IMM5_BITS))
2822 	return NULL;
2823 
2824       insn->bits = (template->meta_opcode |
2825 		    (1 << 25) |
2826 		    (regs[0]->no << 19) |
2827 		    (regs[1]->no << 14) |
2828 		    ((value & IMM5_MASK) << 9));
2829     }
2830   else
2831     {
2832       l = parse_gp_regs (l, src2_regs, 1);
2833 
2834       if (l == NULL)
2835 	return NULL;
2836 
2837       insn->bits = (template->meta_opcode |
2838 		    (regs[0]->no << 19) |
2839 		    (regs[1]->no << 14) |
2840 		    (src2_regs[0]->no << 9));
2841 
2842       if (src2_regs[0]->unit != regs[1]->unit)
2843 	{
2844 	  as_bad(_("Source registers must be in the same unit"));
2845 	  return NULL;
2846 	}
2847     }
2848 
2849   if (regs[0]->unit != regs[1]->unit)
2850     {
2851       if (cond && ca)
2852 	{
2853 	  if (regs[1]->unit == UNIT_D0)
2854 	    unit_bit = 0;
2855 	  else if (regs[1]->unit == UNIT_D1)
2856 	    unit_bit = 1;
2857 	  else
2858 	    return NULL;
2859 
2860 	  insn->bits |= ((1 << 5) |
2861 			 (regs[0]->unit << 1));
2862 	}
2863       else
2864 	return NULL;
2865     }
2866 
2867   insn->bits |= unit_bit << 24;
2868   insn->len = 4;
2869   return l;
2870 }
2871 
2872 /* Parse a MIN or MAX instruction.  */
2873 static const char *
parse_min_max(const char * line,metag_insn * insn,const insn_template * template)2874 parse_min_max (const char *line, metag_insn *insn,
2875 	       const insn_template *template)
2876 {
2877   const char *l = line;
2878   const metag_reg *regs[3];
2879 
2880   l = parse_gp_regs (l, regs, 3);
2881 
2882   if (l == NULL)
2883     return NULL;
2884 
2885   if (!(regs[0]->unit == UNIT_D0 ||
2886 	regs[0]->unit == UNIT_D1))
2887       return NULL;
2888 
2889   if (!(regs[0]->unit == regs[1]->unit &&
2890 	regs[1]->unit == regs[2]->unit))
2891       return NULL;
2892 
2893   insn->bits = (template->meta_opcode |
2894 		(regs[0]->no << 19) |
2895 		(regs[1]->no << 14) |
2896 		(regs[2]->no << 9));
2897 
2898   if (regs[0]->unit == UNIT_D1)
2899     insn->bits |= (1 << 24);
2900 
2901   insn->len = 4;
2902   return l;
2903 }
2904 
2905 /* Parse a bit operation instruction.  */
2906 static const char *
parse_bitop(const char * line,metag_insn * insn,const insn_template * template)2907 parse_bitop (const char *line, metag_insn *insn,
2908 	     const insn_template *template)
2909 {
2910   const char *l = line;
2911   const metag_reg *regs[2];
2912   unsigned int swap_inst = MAJOR_OPCODE (template->meta_opcode) == OPC_MISC;
2913   unsigned int is_bexl = 0;
2914 
2915   if (swap_inst &&
2916       ((template->meta_opcode >> 1) & 0xb) == 0xa)
2917     is_bexl = 1;
2918 
2919   l = parse_gp_regs (l, regs, 2);
2920 
2921   if (l == NULL)
2922     return NULL;
2923 
2924   if (!(regs[0]->unit == UNIT_D0 ||
2925 	regs[0]->unit == UNIT_D1))
2926       return NULL;
2927 
2928   if (is_bexl)
2929     {
2930       if (regs[0]->unit == UNIT_D0 &&
2931 	  regs[1]->unit != UNIT_D1)
2932 	return NULL;
2933       else if (regs[0]->unit == UNIT_D1 &&
2934 	       regs[1]->unit != UNIT_D0)
2935 	return NULL;
2936     }
2937   else if (!(regs[0]->unit == regs[1]->unit))
2938       return NULL;
2939 
2940   insn->bits = (template->meta_opcode |
2941 		(regs[0]->no << 19) |
2942 		(regs[1]->no << 14));
2943 
2944   if (swap_inst)
2945     {
2946       if (regs[1]->unit == UNIT_D1)
2947 	insn->bits |= 1;
2948     }
2949   else
2950     {
2951       if (regs[1]->unit == UNIT_D1)
2952 	insn->bits |= (1 << 24);
2953     }
2954 
2955   insn->len = 4;
2956   return l;
2957 }
2958 
2959 /* Parse a CMP or TST instruction.  */
2960 static const char *
parse_cmp(const char * line,metag_insn * insn,const insn_template * template)2961 parse_cmp (const char *line, metag_insn *insn,
2962 	   const insn_template *template)
2963 {
2964   const char *l = line;
2965   const metag_reg *dest_regs[1];
2966   const metag_reg *src_regs[1];
2967   int value = 0;
2968   unsigned int imm = (template->meta_opcode >> 25) & 0x1;
2969   unsigned int cond = (template->meta_opcode >> 26) & 0x1;
2970   unsigned int top = template->meta_opcode & 0x1;
2971   unsigned int sign_extend = 0;
2972   unsigned int unit_bit = 0;
2973 
2974   l = parse_gp_regs (l, dest_regs, 1);
2975 
2976   if (l == NULL)
2977     return NULL;
2978 
2979   l = skip_comma (l);
2980 
2981   if (l == NULL ||
2982       *l == END_OF_INSN)
2983     return NULL;
2984 
2985   if (dest_regs[0]->unit == UNIT_D0)
2986     unit_bit = 0;
2987   else if (dest_regs[0]->unit == UNIT_D1)
2988     unit_bit = 1;
2989   else
2990     return NULL;
2991 
2992   if (imm)
2993     {
2994       if (cond)
2995 	{
2996 	  l = parse_imm_constant (l, insn, &value);
2997 
2998 	  if (l == NULL)
2999 	    return NULL;
3000 
3001 	  if (!within_unsigned_range (value, IMM8_BITS))
3002 	    return NULL;
3003 
3004 	  insn->bits = (template->meta_opcode |
3005 			(dest_regs[0]->no << 14) |
3006 			((value & IMM8_MASK) << 6));
3007 
3008 	}
3009       else
3010 	{
3011 	  l = parse_imm16 (l, insn, &value);
3012 
3013 	  if (l == NULL)
3014 	    return NULL;
3015 
3016 	  if (value < 0)
3017 	    {
3018 	      if (!within_signed_range (value, IMM16_BITS))
3019 		{
3020 		  as_bad (_("immediate out of range"));
3021 		  return NULL;
3022 		}
3023 	      sign_extend = 1;
3024 	    }
3025 	  else
3026 	    {
3027 	      if (!within_unsigned_range (value, IMM16_BITS))
3028 		{
3029 		  as_bad (_("immediate out of range"));
3030 		  return NULL;
3031 		}
3032 	    }
3033 
3034 	  insn->bits = (template->meta_opcode |
3035 			(dest_regs[0]->no << 19) |
3036 			((value & IMM16_MASK) << 3));
3037 	}
3038     }
3039   else
3040     {
3041       unsigned int o2r = 0;
3042       int rs2;
3043 
3044       l = parse_gp_regs (l, src_regs, 1);
3045 
3046       if (l == NULL)
3047 	return NULL;
3048 
3049       if (dest_regs[0]->unit != src_regs[0]->unit)
3050 	{
3051 	  rs2 = lookup_o2r (0, unit_bit, src_regs[0]);
3052 
3053 	  if (rs2 < 0)
3054 	    return NULL;
3055 
3056 	  o2r = 1;
3057 	}
3058       else
3059 	{
3060 	  rs2 = src_regs[0]->no;
3061 	}
3062 
3063       insn->bits = (template->meta_opcode |
3064 		    (dest_regs[0]->no << 14) |
3065 		    (rs2 << 9));
3066 
3067       if (o2r)
3068 	insn->bits |= 1;
3069     }
3070 
3071   if (sign_extend == 1 && top == 0)
3072     insn->bits |= (1 << 1);
3073 
3074   insn->bits |= unit_bit << 24;
3075   insn->len = 4;
3076   return l;
3077 }
3078 
3079 /* Parse a CACHEW instruction.  */
3080 static const char *
parse_cachew(const char * line,metag_insn * insn,const insn_template * template)3081 parse_cachew (const char *line, metag_insn *insn,
3082 	      const insn_template *template)
3083 {
3084   const char *l = line;
3085   const metag_reg *src_regs[2];
3086   unsigned int size = ((template->meta_opcode >> 1) & 0x1) ? 8 : 4;
3087   metag_addr addr;
3088   int offset;
3089 
3090   memset(&addr, 0, sizeof(addr));
3091   addr.reloc_type = BFD_RELOC_UNUSED;
3092 
3093   l = parse_addr (l, &addr, size);
3094 
3095   if (l == NULL ||
3096       !is_short_unit (addr.base_reg->unit) ||
3097       addr.update ||
3098       !addr.immediate)
3099     {
3100 	  as_bad (_("invalid memory operand"));
3101 	  return NULL;
3102     }
3103 
3104   l = skip_comma (l);
3105 
3106   if (l == NULL ||
3107       *l == END_OF_INSN)
3108     return NULL;
3109 
3110   if (size == 4)
3111     l = parse_gp_regs (l, src_regs, 1);
3112   else
3113     l = parse_pair_gp_regs (l, src_regs);
3114 
3115   if (l == NULL ||
3116       !is_short_unit (src_regs[0]->unit))
3117     {
3118       as_bad (_("invalid source register"));
3119       return NULL;
3120     }
3121 
3122   offset = addr.exp.X_add_number;
3123 
3124   if (addr.negate)
3125     offset = -offset;
3126 
3127   offset = offset / 64;
3128 
3129   if (!within_signed_range (offset, GET_SET_IMM_BITS))
3130     {
3131       as_bad (_("offset value out of range"));
3132       return NULL;
3133     }
3134 
3135   insn->bits = (template->meta_opcode |
3136 		(src_regs[0]->no << 19) |
3137 		(addr.base_reg->no << 14) |
3138 		((offset & GET_SET_IMM_MASK) << 8) |
3139 		((addr.base_reg->unit & SHORT_UNIT_MASK) << 5) |
3140 		((src_regs[0]->unit & SHORT_UNIT_MASK) << 3));
3141 
3142   insn->len = 4;
3143   return l;
3144 }
3145 
3146 /* Parse a CACHEW instruction.  */
3147 static const char *
parse_cacher(const char * line,metag_insn * insn,const insn_template * template)3148 parse_cacher (const char *line, metag_insn *insn,
3149 	      const insn_template *template)
3150 {
3151   const char *l = line;
3152   const metag_reg *dest_regs[2];
3153   unsigned int size = ((template->meta_opcode >> 1) & 0x1) ? 8 : 4;
3154   metag_addr addr;
3155   int offset;
3156 
3157   memset(&addr, 0, sizeof(addr));
3158   addr.reloc_type = BFD_RELOC_UNUSED;
3159 
3160   if (size == 4)
3161     l = parse_gp_regs (l, dest_regs, 1);
3162   else
3163     l = parse_pair_gp_regs (l, dest_regs);
3164 
3165   if (l == NULL ||
3166       !is_short_unit (dest_regs[0]->unit))
3167     {
3168       as_bad (_("invalid destination register"));
3169       return NULL;
3170     }
3171 
3172   l = skip_comma (l);
3173 
3174   if (l == NULL ||
3175       *l == END_OF_INSN)
3176     return NULL;
3177 
3178   l = parse_addr (l, &addr, size);
3179 
3180   if (l == NULL ||
3181       !is_short_unit (addr.base_reg->unit) ||
3182       addr.update ||
3183       !addr.immediate)
3184     {
3185 	  as_bad (_("invalid memory operand"));
3186 	  return NULL;
3187     }
3188 
3189   offset = addr.exp.X_add_number;
3190 
3191   if (addr.negate)
3192     offset = -offset;
3193 
3194   offset = offset / (int)size;
3195 
3196   if (!within_signed_range (offset, GET_SET_IMM_BITS))
3197     {
3198       as_bad (_("offset value out of range"));
3199       return NULL;
3200     }
3201 
3202   insn->bits = (template->meta_opcode |
3203 		(dest_regs[0]->no << 19) |
3204 		(addr.base_reg->no << 14) |
3205 		((offset & GET_SET_IMM_MASK) << 8) |
3206 		((addr.base_reg->unit & SHORT_UNIT_MASK) << 5) |
3207 		((dest_regs[0]->unit & SHORT_UNIT_MASK) << 3));
3208 
3209   insn->len = 4;
3210   return l;
3211 }
3212 
3213 /* Parse an ICACHE instruction.  */
3214 static const char *
parse_icache(const char * line,metag_insn * insn,const insn_template * template)3215 parse_icache (const char *line, metag_insn *insn,
3216 	      const insn_template *template)
3217 {
3218   const char *l = line;
3219   int offset;
3220   int pfcount;
3221 
3222   l = parse_imm_constant (l, insn, &offset);
3223 
3224   if (l == NULL)
3225     return NULL;
3226 
3227   if (!within_signed_range (offset, IMM15_BITS))
3228     return NULL;
3229 
3230   l = skip_comma (l);
3231 
3232   l = parse_imm_constant (l, insn, &pfcount);
3233 
3234   if (l == NULL)
3235     return NULL;
3236 
3237   if (!within_unsigned_range (pfcount, IMM4_BITS))
3238     return NULL;
3239 
3240   insn->bits = (template->meta_opcode |
3241 		((offset & IMM15_MASK) << 9) |
3242 		((pfcount & IMM4_MASK) << 1));
3243 
3244   insn->len = 4;
3245   return l;
3246 }
3247 
3248 /* Parse a LNKGET instruction.  */
3249 static const char *
parse_lnkget(const char * line,metag_insn * insn,const insn_template * template)3250 parse_lnkget (const char *line, metag_insn *insn,
3251 	      const insn_template *template)
3252 {
3253   const char *l = line;
3254   const metag_reg *dest_regs[2];
3255   unsigned int size = metag_get_set_ext_size_bytes (template->meta_opcode);
3256   metag_addr addr;
3257   int offset;
3258 
3259   memset(&addr, 0, sizeof(addr));
3260   addr.reloc_type = BFD_RELOC_UNUSED;
3261 
3262   if (size == 8)
3263     l = parse_pair_gp_regs (l, dest_regs);
3264   else
3265     l = parse_gp_regs (l, dest_regs, 1);
3266 
3267   if (l == NULL ||
3268       !is_short_unit (dest_regs[0]->unit))
3269     {
3270       as_bad (_("invalid destination register"));
3271       return NULL;
3272     }
3273 
3274   l = skip_comma (l);
3275 
3276   if (l == NULL ||
3277       *l == END_OF_INSN)
3278     return NULL;
3279 
3280   l = parse_addr (l, &addr, size);
3281 
3282   if (l == NULL ||
3283       !is_short_unit (addr.base_reg->unit) ||
3284       addr.update ||
3285       !addr.immediate)
3286     {
3287 	  as_bad (_("invalid memory operand"));
3288 	  return NULL;
3289     }
3290 
3291   offset = addr.exp.X_add_number;
3292 
3293   if (addr.negate)
3294     offset = -offset;
3295 
3296   offset = offset / size;
3297 
3298   if (!within_signed_range (offset, GET_SET_IMM_BITS))
3299     {
3300       as_bad (_("offset value out of range"));
3301       return NULL;
3302     }
3303 
3304   insn->bits = (template->meta_opcode |
3305 		(dest_regs[0]->no << 19) |
3306 		(addr.base_reg->no << 14) |
3307 		((offset & GET_SET_IMM_MASK) << 8) |
3308 		((addr.base_reg->unit & SHORT_UNIT_MASK) << 5) |
3309 		((dest_regs[0]->unit & SHORT_UNIT_MASK) << 3));
3310 
3311   insn->len = 4;
3312   return l;
3313 }
3314 
3315 /* Parse an FPU MOV instruction.  */
3316 static const char *
parse_fmov(const char * line,metag_insn * insn,const insn_template * template)3317 parse_fmov (const char *line, metag_insn *insn,
3318 	    const insn_template *template)
3319 {
3320   const char *l = line;
3321   const metag_reg *regs[2];
3322 
3323   l = parse_fpu_regs (l, regs, 2);
3324 
3325   if (l == NULL)
3326     return NULL;
3327 
3328   insn->bits = (template->meta_opcode |
3329 		(regs[0]->no << 19) |
3330 		(regs[1]->no << 14));
3331 
3332   if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3333     insn->bits |= (1 << 5);
3334   else if (insn->fpu_width == FPU_WIDTH_PAIR)
3335     insn->bits |= (1 << 6);
3336 
3337   insn->len = 4;
3338   return l;
3339 }
3340 
3341 /* Parse an FPU MMOV instruction.  */
3342 static const char *
parse_fmmov(const char * line,metag_insn * insn,const insn_template * template)3343 parse_fmmov (const char *line, metag_insn *insn,
3344 	     const insn_template *template)
3345 {
3346   const char *l = line;
3347   bfd_boolean to_fpu = MAJOR_OPCODE (template->meta_opcode) == OPC_GET;
3348   bfd_boolean is_mmovl = MINOR_OPCODE (template->meta_opcode) & 0x1;
3349   size_t regs_read = 0;
3350   const metag_reg *regs[16];
3351   unsigned int lowest_data_reg = 0xffffffff;
3352   unsigned int lowest_fpu_reg = 0xffffffff;
3353   unsigned int rmask = 0, data_unit;
3354   size_t i;
3355   int last_reg = -1;
3356 
3357   if (insn->fpu_width != FPU_WIDTH_SINGLE)
3358     return NULL;
3359 
3360   l = parse_gp_regs_list (l, regs, 16, &regs_read);
3361 
3362   if (l == NULL)
3363     return NULL;
3364 
3365   if (regs_read % 2)
3366     return NULL;
3367 
3368   if (to_fpu)
3369     {
3370       for (i = 0; i < regs_read / 2; i++)
3371 	{
3372 	  if (regs[i]->unit != UNIT_FX)
3373 	    return NULL;
3374 
3375 	  if (last_reg == -1)
3376 	    {
3377 	      last_reg = regs[i]->no;
3378 	      lowest_fpu_reg = last_reg;
3379 	    }
3380 	  else
3381 	    {
3382 	      if (is_mmovl)
3383 		{
3384 		  if (regs[i]->no != (unsigned int)(last_reg + 2))
3385 		    return NULL;
3386 		}
3387 	      else if (regs[i]->no != (unsigned int)(last_reg + 1))
3388 		return NULL;
3389 
3390 	      last_reg = regs[i]->no;
3391 	    }
3392 	}
3393 
3394       if (regs[i]->unit == UNIT_D0)
3395 	data_unit = 0;
3396       else if (regs[i]->unit == UNIT_D1)
3397 	data_unit = 1;
3398       else
3399 	return NULL;
3400 
3401       if (!check_rmask (&regs[i], regs_read / 2, TRUE, FALSE, &lowest_data_reg,
3402 			&rmask))
3403 	return NULL;
3404     }
3405   else
3406     {
3407       if (regs[0]->unit == UNIT_D0)
3408 	data_unit = 0;
3409       else if (regs[0]->unit == UNIT_D1)
3410 	data_unit = 1;
3411       else
3412 	return NULL;
3413 
3414       if (!check_rmask (regs, regs_read / 2, TRUE, FALSE, &lowest_data_reg,
3415 			&rmask))
3416 	return NULL;
3417 
3418       for (i = regs_read / 2; i < regs_read; i++)
3419 	{
3420 	  if (regs[i]->unit != UNIT_FX)
3421 	    return NULL;
3422 
3423 	  if (last_reg == -1)
3424 	    {
3425 	      last_reg = regs[i]->no;
3426 	      lowest_fpu_reg = last_reg;
3427 	    }
3428 	  else
3429 	    {
3430 	      if (is_mmovl)
3431 		{
3432 		  if (regs[i]->no != (unsigned int)(last_reg + 2))
3433 		    return NULL;
3434 		}
3435 	      else if (regs[i]->no != (unsigned int)(last_reg + 1))
3436 		return NULL;
3437 
3438 	      last_reg = regs[i]->no;
3439 	    }
3440 	}
3441     }
3442 
3443   insn->bits = (template->meta_opcode |
3444 		((lowest_data_reg & REG_MASK) << 19) |
3445 		((lowest_fpu_reg & REG_MASK) << 14) |
3446 		((rmask & RMASK_MASK) << 7) |
3447 		data_unit);
3448 
3449   insn->len = 4;
3450   return l;
3451 }
3452 
3453 /* Parse an FPU data unit MOV instruction.  */
3454 static const char *
parse_fmov_data(const char * line,metag_insn * insn,const insn_template * template)3455 parse_fmov_data (const char *line, metag_insn *insn,
3456 	    const insn_template *template)
3457 {
3458   const char *l = line;
3459   unsigned int to_fpu = ((template->meta_opcode >> 7) & 0x1);
3460   const metag_reg *regs[2];
3461   unsigned int base_unit;
3462 
3463   if (insn->fpu_width == FPU_WIDTH_PAIR)
3464     return NULL;
3465 
3466   l = parse_gp_regs (l, regs, 2);
3467 
3468   if (l == NULL)
3469     return NULL;
3470 
3471   if (to_fpu)
3472     {
3473       if (regs[0]->unit != UNIT_FX)
3474 	return NULL;
3475 
3476       if (regs[1]->unit == UNIT_D0)
3477 	base_unit = 0;
3478       else if (regs[1]->unit == UNIT_D1)
3479 	base_unit = 1;
3480       else
3481 	return NULL;
3482     }
3483   else
3484     {
3485       if (regs[0]->unit == UNIT_D0)
3486 	base_unit = 0;
3487       else if (regs[0]->unit == UNIT_D1)
3488 	base_unit = 1;
3489       else
3490 	return NULL;
3491 
3492       if (regs[1]->unit != UNIT_FX)
3493 	return NULL;
3494     }
3495 
3496   insn->bits = (template->meta_opcode |
3497 		(base_unit << 24) |
3498 		(regs[0]->no << 19) |
3499 		(regs[1]->no << 9));
3500 
3501   insn->len = 4;
3502   return l;
3503 }
3504 
3505 /* Parse an FPU immediate MOV instruction.  */
3506 static const char *
parse_fmov_i(const char * line,metag_insn * insn,const insn_template * template)3507 parse_fmov_i (const char *line, metag_insn *insn,
3508 	      const insn_template *template)
3509 {
3510   const char *l = line;
3511   const metag_reg *regs[1];
3512   int value = 0;
3513 
3514   l = parse_fpu_regs (l, regs, 1);
3515 
3516   l = skip_comma (l);
3517 
3518   if (l == NULL ||
3519       *l == END_OF_INSN)
3520     return NULL;
3521 
3522   l = parse_imm16 (l, insn, &value);
3523 
3524   if (l == NULL)
3525     return NULL;
3526 
3527   insn->bits = (template->meta_opcode |
3528 		(regs[0]->no << 19) |
3529 		((value & IMM16_MASK) << 3));
3530 
3531   if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3532     insn->bits |= (1 << 1);
3533   else if (insn->fpu_width == FPU_WIDTH_PAIR)
3534     insn->bits |= (1 << 2);
3535 
3536   insn->len = 4;
3537   return l;
3538 }
3539 
3540 /* Parse an FPU PACK instruction.  */
3541 static const char *
parse_fpack(const char * line,metag_insn * insn,const insn_template * template)3542 parse_fpack (const char *line, metag_insn *insn,
3543 	     const insn_template *template)
3544 {
3545   const char *l = line;
3546   const metag_reg *regs[3];
3547 
3548   l = parse_fpu_regs (l, regs, 3);
3549 
3550   if (l == NULL)
3551     return NULL;
3552 
3553   if (regs[0]->no % 2)
3554     {
3555       as_bad (_("destination register should be even numbered"));
3556       return NULL;
3557     }
3558 
3559   insn->bits = (template->meta_opcode |
3560 		(regs[0]->no << 19) |
3561 		(regs[1]->no << 14) |
3562 		(regs[2]->no << 9));
3563 
3564   insn->len = 4;
3565   return l;
3566 }
3567 
3568 /* Parse an FPU SWAP instruction.  */
3569 static const char *
parse_fswap(const char * line,metag_insn * insn,const insn_template * template)3570 parse_fswap (const char *line, metag_insn *insn,
3571 	     const insn_template *template)
3572 {
3573   const char *l = line;
3574   const metag_reg *regs[2];
3575 
3576   if (insn->fpu_width != FPU_WIDTH_PAIR)
3577     return NULL;
3578 
3579   l = parse_fpu_regs (l, regs, 2);
3580 
3581   if (l == NULL)
3582     return NULL;
3583 
3584   if (regs[0]->no % 2)
3585     return NULL;
3586 
3587   if (regs[1]->no % 2)
3588     return NULL;
3589 
3590   insn->bits = (template->meta_opcode |
3591 		(regs[0]->no << 19) |
3592 		(regs[1]->no << 14));
3593 
3594   insn->len = 4;
3595   return l;
3596 }
3597 
3598 /* Parse an FPU CMP instruction.  */
3599 static const char *
parse_fcmp(const char * line,metag_insn * insn,const insn_template * template)3600 parse_fcmp (const char *line, metag_insn *insn,
3601 	    const insn_template *template)
3602 {
3603   const char *l = line, *l2;
3604   const metag_reg *regs1[1];
3605   const metag_reg *regs2[1];
3606 
3607   l = parse_fpu_regs (l, regs1, 1);
3608 
3609   l = skip_comma (l);
3610 
3611   if (l == NULL ||
3612       *l == END_OF_INSN)
3613     return NULL;
3614 
3615   l2 = parse_fpu_regs (l, regs2, 1);
3616 
3617   if (l2 != NULL)
3618     {
3619       insn->bits = (regs2[0]->no << 9);
3620     }
3621   else
3622     {
3623       int constant = 0;
3624       l2 = parse_imm_constant (l, insn, &constant);
3625       if (!l2 || constant != 0)
3626 	{
3627 	  as_bad (_("comparison must be with register or #0"));
3628 	  return NULL;
3629 	}
3630       insn->bits = (1 << 8);
3631     }
3632 
3633   insn->bits |= (template->meta_opcode |
3634 		 (regs1[0]->no << 14));
3635 
3636   if (insn->fpu_action_flags & FPU_ACTION_ABS)
3637     insn->bits |= (1 << 19);
3638 
3639   if (insn->fpu_action_flags & FPU_ACTION_QUIET)
3640     insn->bits |= (1 << 7);
3641 
3642   if (insn->fpu_width == FPU_WIDTH_PAIR)
3643     insn->bits |= (1 << 6);
3644   else if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3645     insn->bits |= (1 << 5);
3646 
3647   insn->len = 4;
3648   return l2;
3649 }
3650 
3651 /* Parse an FPU MIN or MAX instruction.  */
3652 static const char *
parse_fminmax(const char * line,metag_insn * insn,const insn_template * template)3653 parse_fminmax (const char *line, metag_insn *insn,
3654 	       const insn_template *template)
3655 {
3656   const char *l = line;
3657   const metag_reg *regs[3];
3658 
3659   l = parse_fpu_regs (l, regs, 3);
3660 
3661   if (l == NULL)
3662     return NULL;
3663 
3664   insn->bits = (template->meta_opcode |
3665 		(regs[0]->no << 19) |
3666 		(regs[1]->no << 14) |
3667 		(regs[2]->no << 9));
3668 
3669   if (insn->fpu_width == FPU_WIDTH_PAIR)
3670     insn->bits |= (1 << 6);
3671   else if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3672     insn->bits |= (1 << 5);
3673 
3674   insn->len = 4;
3675   return l;
3676 }
3677 
3678 /* Parse an FPU data conversion instruction.  */
3679 static const char *
parse_fconv(const char * line,metag_insn * insn,const insn_template * template)3680 parse_fconv (const char *line, metag_insn *insn,
3681 	     const insn_template *template)
3682 {
3683   const char *l = line;
3684   const metag_reg *regs[2];
3685 
3686   if (insn->fpu_width == FPU_WIDTH_PAIR)
3687     {
3688       if (strncasecmp (template->name, "FTOH", 4) &&
3689 	  strncasecmp (template->name, "HTOF", 4) &&
3690 	  strncasecmp (template->name, "FTOI", 4) &&
3691 	  strncasecmp (template->name, "ITOF", 4))
3692 	{
3693 	  as_bad (_("instruction cannot operate on pair values"));
3694 	  return NULL;
3695 	}
3696     }
3697 
3698   if (insn->fpu_action_flags & FPU_ACTION_ZERO)
3699     {
3700       if (strncasecmp (template->name, "FTOI", 4) &&
3701 	  strncasecmp (template->name, "DTOI", 4) &&
3702 	  strncasecmp (template->name, "DTOL", 4))
3703 	{
3704 	  as_bad (_("zero flag is not valid for this instruction"));
3705 	  return NULL;
3706 	}
3707     }
3708 
3709   l = parse_fpu_regs (l, regs, 2);
3710 
3711   if (l == NULL)
3712     return NULL;
3713 
3714   if (!strncasecmp (template->name, "DTOL", 4) ||
3715       !strncasecmp (template->name, "LTOD", 4))
3716     {
3717       if (regs[0]->no % 2)
3718 	{
3719 	  as_bad (_("destination register should be even numbered"));
3720 	  return NULL;
3721 	}
3722 
3723       if (regs[1]->no % 2)
3724 	{
3725 	  as_bad (_("source register should be even numbered"));
3726 	  return NULL;
3727 	}
3728     }
3729 
3730   insn->bits = (template->meta_opcode |
3731 		(regs[0]->no << 19) |
3732 		(regs[1]->no << 14));
3733 
3734   if (insn->fpu_width == FPU_WIDTH_PAIR)
3735     insn->bits |= (1 << 6);
3736 
3737   if (insn->fpu_action_flags & FPU_ACTION_ZERO)
3738     insn->bits |= (1 << 12);
3739 
3740   insn->len = 4;
3741   return l;
3742 }
3743 
3744 /* Parse an FPU extended data conversion instruction.  */
3745 static const char *
parse_fconvx(const char * line,metag_insn * insn,const insn_template * template)3746 parse_fconvx (const char *line, metag_insn *insn,
3747 	      const insn_template *template)
3748 {
3749   const char *l = line;
3750   const metag_reg *regs[2];
3751   int fraction_bits = 0;
3752 
3753   if (insn->fpu_width == FPU_WIDTH_PAIR)
3754     {
3755       if (strncasecmp (template->name, "FTOX", 4) &&
3756 	  strncasecmp (template->name, "XTOF", 4))
3757 	{
3758 	  as_bad (_("instruction cannot operate on pair values"));
3759 	  return NULL;
3760 	}
3761     }
3762 
3763   l = parse_fpu_regs (l, regs, 2);
3764 
3765   l = skip_comma (l);
3766 
3767   if (l == NULL ||
3768       *l == END_OF_INSN)
3769     return NULL;
3770 
3771   l = parse_imm_constant (l, insn, &fraction_bits);
3772 
3773   if (l == NULL)
3774     return NULL;
3775 
3776   insn->bits = (template->meta_opcode |
3777 		(regs[0]->no << 19) |
3778 		(regs[1]->no << 14));
3779 
3780   if (strncasecmp (template->name, "DTOXL", 5) &&
3781       strncasecmp (template->name, "XLTOD", 5))
3782     {
3783       if (!within_unsigned_range (fraction_bits, IMM5_BITS))
3784 	{
3785 	  as_bad (_("fraction bits value out of range"));
3786 	  return NULL;
3787 	}
3788       insn->bits |= ((fraction_bits & IMM5_MASK) << 9);
3789     }
3790   else
3791     {
3792       if (!within_unsigned_range (fraction_bits, IMM6_BITS))
3793 	{
3794 	  as_bad (_("fraction bits value out of range"));
3795 	  return NULL;
3796 	}
3797       insn->bits |= ((fraction_bits & IMM6_MASK) << 8);
3798     }
3799 
3800   if (insn->fpu_width == FPU_WIDTH_PAIR)
3801     insn->bits |= (1 << 6);
3802 
3803   insn->len = 4;
3804   return l;
3805 }
3806 
3807 /* Parse an FPU basic arithmetic instruction.  */
3808 static const char *
parse_fbarith(const char * line,metag_insn * insn,const insn_template * template)3809 parse_fbarith (const char *line, metag_insn *insn,
3810 	       const insn_template *template)
3811 {
3812   const char *l = line;
3813   const metag_reg *regs[3];
3814 
3815   l = parse_fpu_regs (l, regs, 3);
3816 
3817   if (l == NULL)
3818     return NULL;
3819 
3820   insn->bits = (template->meta_opcode |
3821 		(regs[0]->no << 19) |
3822 		(regs[1]->no << 14) |
3823 		(regs[2]->no << 9));
3824 
3825   if (insn->fpu_width == FPU_WIDTH_PAIR)
3826     insn->bits |= (1 << 6);
3827   else if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3828     insn->bits |= (1 << 5);
3829 
3830   if (insn->fpu_action_flags & FPU_ACTION_INV)
3831     insn->bits |= (1 << 7);
3832 
3833   insn->len = 4;
3834   return l;
3835 }
3836 
3837 /* Parse a floating point accumulator name.  */
3838 static const char *
parse_acf(const char * line,int * part)3839 parse_acf (const char *line, int *part)
3840 {
3841   const char *l = line;
3842   size_t i;
3843 
3844   for (i = 0; i < sizeof(metag_acftab)/sizeof(metag_acftab[0]); i++)
3845     {
3846       const metag_acf *acf = &metag_acftab[i];
3847       size_t name_len = strlen (acf->name);
3848 
3849       if (strncasecmp (l, acf->name, name_len) == 0)
3850 	{
3851 	  l += name_len;
3852 	  *part = acf->part;
3853 	  return l;
3854 	}
3855     }
3856   return NULL;
3857 }
3858 
3859 /* Parse an FPU extended arithmetic instruction.  */
3860 static const char *
parse_fearith(const char * line,metag_insn * insn,const insn_template * template)3861 parse_fearith (const char *line, metag_insn *insn,
3862 	       const insn_template *template)
3863 {
3864   const char *l = line;
3865   const metag_reg *regs[3];
3866   bfd_boolean is_muz = (MINOR_OPCODE (template->meta_opcode) == 0x6 &&
3867 			((template->meta_opcode >> 4) & 0x1));
3868   unsigned int is_o3o = template->meta_opcode & 0x1;
3869   unsigned int is_mac = 0;
3870   unsigned int is_maw = 0;
3871 
3872   if (!strncasecmp (template->name, "MAW", 3))
3873     is_maw = 1;
3874 
3875   if (!strncasecmp (template->name, "MAC", 3))
3876     {
3877       int part;
3878       l = parse_acf (l, &part);
3879 
3880       if (l == NULL || part != 0)
3881 	return NULL;
3882 
3883       l = skip_comma (l);
3884 
3885       l = parse_fpu_regs (l, &regs[1], 2);
3886 
3887       is_mac = 1;
3888     }
3889   else
3890     {
3891       if (is_o3o && is_maw)
3892 	l = parse_fpu_regs (l, regs, 2);
3893       else
3894 	l = parse_fpu_regs (l, regs, 3);
3895     }
3896 
3897   if (l == NULL)
3898     return NULL;
3899 
3900   if (is_o3o && is_maw)
3901     insn->bits = (template->meta_opcode |
3902 		  (regs[1]->no << 9));
3903   else
3904     insn->bits = (template->meta_opcode |
3905 		  (regs[1]->no << 14));
3906 
3907   if (!(is_o3o && is_maw))
3908     insn->bits |= (regs[2]->no << 9);
3909 
3910   if (is_o3o && is_maw)
3911     insn->bits |= (regs[0]->no << 14);
3912   else if (!is_mac)
3913     insn->bits |= (regs[0]->no << 19);
3914 
3915   if (insn->fpu_width == FPU_WIDTH_PAIR)
3916     insn->bits |= (1 << 6);
3917   else if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3918     insn->bits |= (1 << 5);
3919 
3920   if (!is_mac && !is_maw)
3921     if (insn->fpu_action_flags & FPU_ACTION_INV)
3922       insn->bits |= (1 << 7);
3923 
3924   if (is_muz)
3925     if (insn->fpu_action_flags & FPU_ACTION_QUIET)
3926       insn->bits |= (1 << 1);
3927 
3928   insn->len = 4;
3929   return l;
3930 }
3931 
3932 /* Parse an FPU RCP or RSQ instruction.  */
3933 static const char *
parse_frec(const char * line,metag_insn * insn,const insn_template * template)3934 parse_frec (const char *line, metag_insn *insn,
3935 	    const insn_template *template)
3936 {
3937   const char *l = line;
3938   const metag_reg *regs[2];
3939 
3940   l = parse_fpu_regs (l, regs, 2);
3941 
3942   if (l == NULL)
3943     return NULL;
3944 
3945   insn->bits = (template->meta_opcode |
3946 		(regs[0]->no << 19) |
3947 		(regs[1]->no << 14));
3948 
3949   if (insn->fpu_width == FPU_WIDTH_PAIR)
3950     insn->bits |= (1 << 6);
3951   else if (insn->fpu_width == FPU_WIDTH_DOUBLE)
3952     insn->bits |= (1 << 5);
3953 
3954   if (insn->fpu_action_flags & FPU_ACTION_ZERO)
3955     insn->bits |= (1 << 10);
3956   else if (insn->fpu_action_flags & FPU_ACTION_QUIET)
3957     insn->bits |= (1 << 9);
3958 
3959   if (insn->fpu_action_flags & FPU_ACTION_INV)
3960     insn->bits |= (1 << 7);
3961 
3962   insn->len = 4;
3963   return l;
3964 }
3965 
3966 /* Parse an FPU vector arithmetic instruction.  */
3967 static const char *
parse_fsimd(const char * line,metag_insn * insn,const insn_template * template)3968 parse_fsimd (const char *line, metag_insn *insn,
3969 	     const insn_template *template)
3970 {
3971   const char *l = line;
3972   const metag_reg *regs[3];
3973 
3974   if (insn->fpu_width != FPU_WIDTH_PAIR)
3975     {
3976       as_bad (_("simd instructions operate on pair values (L prefix)"));
3977       return NULL;
3978     }
3979 
3980   l = parse_fpu_regs (l, regs, 3);
3981 
3982   if (l == NULL)
3983     return NULL;
3984 
3985   if (regs[0]->no % 2)
3986     {
3987       as_bad (_("destination register should be even numbered"));
3988       return NULL;
3989     }
3990 
3991   if ((regs[1]->no % 2) ||
3992       (regs[2]->no % 2))
3993     {
3994       as_bad (_("source registers should be even numbered"));
3995       return NULL;
3996     }
3997 
3998   insn->bits = (template->meta_opcode |
3999 		(regs[0]->no << 19) |
4000 		(regs[1]->no << 14) |
4001 		(regs[2]->no << 9));
4002 
4003   if (insn->fpu_action_flags & FPU_ACTION_INV)
4004     insn->bits |= (1 << 7);
4005 
4006   insn->len = 4;
4007   return l;
4008 }
4009 
4010 /* Parse an FPU accumulator GET or SET instruction. */
4011 static const char *
parse_fget_set_acf(const char * line,metag_insn * insn,const insn_template * template)4012 parse_fget_set_acf (const char *line, metag_insn *insn,
4013 		    const insn_template *template)
4014 {
4015   const char *l = line;
4016   int part;
4017   metag_addr addr;
4018   bfd_boolean is_get = MAJOR_OPCODE (template->meta_opcode) == OPC_GET;
4019 
4020   memset(&addr, 0, sizeof(addr));
4021   addr.reloc_type = BFD_RELOC_UNUSED;
4022 
4023   if (is_get)
4024     {
4025       l = parse_acf (l, &part);
4026 
4027       l = skip_comma (l);
4028 
4029       if (l == NULL)
4030 	return NULL;
4031 
4032       l = parse_mget_mset_addr (l, &addr);
4033     }
4034   else
4035     {
4036       l = parse_mget_mset_addr (l, &addr);
4037 
4038       l = skip_comma (l);
4039 
4040       if (l == NULL)
4041 	return NULL;
4042 
4043       l = parse_acf (l, &part);
4044     }
4045 
4046   if (l == NULL)
4047     return NULL;
4048 
4049   insn->bits = (template->meta_opcode |
4050 		(part << 19));
4051 
4052   if (!is_short_unit (addr.base_reg->unit))
4053     {
4054       as_bad (_("base unit must be one of %s"), SHORT_UNITS);
4055       return NULL;
4056     }
4057 
4058   insn->bits |= ((addr.base_reg->no << 14) |
4059 		 ((addr.base_reg->unit & SHORT_UNIT_MASK) << 5));
4060 
4061   insn->len = 4;
4062   return l;
4063 }
4064 
4065 /* Copy the name of the next register in LINE to REG_BUF.  */
4066 static size_t
strip_reg_name(const char * line,char * reg_buf)4067 strip_reg_name(const char *line, char *reg_buf)
4068 {
4069   const char *l = line;
4070   size_t len = 0;
4071 
4072   while (is_register_char (*l))
4073     {
4074       reg_buf[len] = *l;
4075       l++;
4076       len++;
4077       if (!(len < MAX_REG_LEN))
4078 	return 0;
4079     }
4080 
4081   if (len)
4082     reg_buf[len] = '\0';
4083 
4084   return len;
4085 }
4086 
4087 /* Parse a DSP register from LINE into REG using only the registers
4088    from DSP_REGTAB. Return the next character or NULL.  */
4089 static const char *
__parse_dsp_reg(const char * line,const metag_reg ** reg,htab_t dsp_regtab)4090 __parse_dsp_reg (const char *line, const metag_reg **reg, htab_t dsp_regtab)
4091 {
4092   const char *l = line;
4093   char name[MAX_REG_LEN];
4094   size_t len = 0;
4095   metag_reg entry;
4096   const metag_reg *_reg;
4097 
4098   /* We don't entirely strip the register name because we might
4099      actually want to match whole string in the register table,
4100      e.g. "D0AW.1++" not just "D0AW.1". The string length of the table
4101      entry limits our comaprison to a reasonable bound anyway.  */
4102   while (is_register_char (*l) || *l == PLUS)
4103     {
4104       name[len] = *l;
4105       l++;
4106       len++;
4107       if (!(len < MAX_REG_LEN))
4108 	return NULL;
4109     }
4110 
4111   if (!len)
4112     return NULL;
4113 
4114   name[len] = '\0';
4115   entry.name = name;
4116 
4117   _reg = (const metag_reg *) htab_find (dsp_regtab, &entry);
4118   if (!_reg)
4119     return NULL;
4120 
4121   *reg = _reg;
4122 
4123   return l;
4124 }
4125 
4126 /* Parse a DSP register and setup "reg" with a metag_reg whose "no"
4127    member is suitable for encoding into a DSP insn register field.  */
4128 static const char *
parse_dsp_insn_reg(const char * line,const metag_reg ** reg)4129 parse_dsp_insn_reg (const char *line, const metag_reg **reg)
4130 {
4131   return __parse_dsp_reg (line, reg, dsp_reg_htab);
4132 }
4133 
4134 /* Parse a DSP register and setup "reg" with a metag_reg whose "no"
4135    member is suitable for encoding into a DSP template definition insn
4136    register field.
4137 
4138    There is a separate table for whether we're doing a load or a store
4139    definition. "load" specifies which table to look at.  */
4140 static const char *
parse_dsp_template_reg(const char * line,const metag_reg ** reg,bfd_boolean load)4141 parse_dsp_template_reg (const char *line, const metag_reg **reg,
4142 			bfd_boolean load)
4143 {
4144   return __parse_dsp_reg (line, reg, dsp_tmpl_reg_htab[load]);
4145 }
4146 
4147 /* Parse a single DSP register from LINE.  */
4148 static const char *
parse_dsp_reg(const char * line,const metag_reg ** reg,bfd_boolean tmpl,bfd_boolean load)4149 parse_dsp_reg (const char *line, const metag_reg **reg,
4150 	       bfd_boolean tmpl, bfd_boolean load)
4151 {
4152   if (tmpl)
4153     return parse_dsp_template_reg (line, reg, load);
4154   else
4155     return parse_dsp_insn_reg (line, reg);
4156 }
4157 
4158 /* Return TRUE if UNIT is an address unit.  */
4159 static bfd_boolean
is_addr_unit(enum metag_unit unit)4160 is_addr_unit (enum metag_unit unit)
4161 {
4162   switch (unit)
4163     {
4164     case UNIT_A0:
4165     case UNIT_A1:
4166       return TRUE;
4167     default:
4168       return FALSE;
4169     }
4170 }
4171 
4172 /* Return TRUE if UNIT1 and UNIT2 are equivalent units.  */
4173 static bfd_boolean
is_same_data_unit(enum metag_unit unit1,enum metag_unit unit2)4174 is_same_data_unit (enum metag_unit unit1, enum metag_unit unit2)
4175 {
4176   if (unit1 == unit2)
4177     return TRUE;
4178 
4179   switch (unit1)
4180     {
4181     case UNIT_D0:
4182       if (unit2 == UNIT_ACC_D0 || unit2 == UNIT_RAM_D0)
4183 	return TRUE;
4184       break;
4185     case UNIT_D1:
4186       if (unit2 == UNIT_ACC_D1 || unit2 == UNIT_RAM_D1)
4187 	return TRUE;
4188       break;
4189     case UNIT_ACC_D0:
4190       if (unit2 == UNIT_D0 || unit2 == UNIT_RAM_D0)
4191 	return TRUE;
4192       break;
4193     case UNIT_ACC_D1:
4194       if (unit2 == UNIT_D1 || unit2 == UNIT_RAM_D1)
4195 	return TRUE;
4196       break;
4197     case UNIT_RAM_D0:
4198       if (unit2 == UNIT_ACC_D0 || unit2 == UNIT_D0)
4199 	return TRUE;
4200       break;
4201     case UNIT_RAM_D1:
4202       if (unit2 == UNIT_ACC_D1 || unit2 == UNIT_D1)
4203 	return TRUE;
4204       break;
4205     default:
4206       return FALSE;
4207     }
4208 
4209   return FALSE;
4210 }
4211 
4212 /* Return TRUE if the register NUM is a quickrot control register.  */
4213 static bfd_boolean
is_quickrot_reg(unsigned int num)4214 is_quickrot_reg (unsigned int num)
4215 {
4216   switch (num)
4217     {
4218     case 2:
4219     case 3:
4220       return TRUE;
4221     }
4222 
4223   return FALSE;
4224 }
4225 
4226 /* Return TRUE if REG is an accumulator register.  */
4227 static bfd_boolean
is_accumulator_reg(const metag_reg * reg)4228 is_accumulator_reg (const metag_reg *reg)
4229 {
4230   if (reg->unit == UNIT_ACC_D0 || reg->unit == UNIT_ACC_D1)
4231     return TRUE;
4232 
4233   return FALSE;
4234 }
4235 
4236 /* Return TRUE if REG is a DSP RAM register.  */
4237 static bfd_boolean
is_dspram_reg(const metag_reg * reg)4238 is_dspram_reg (const metag_reg *reg)
4239 {
4240   if (reg->unit == UNIT_RAM_D0 || reg->unit == UNIT_RAM_D1)
4241       return TRUE;
4242 
4243   return FALSE;
4244 }
4245 
4246 static const char *
__parse_gp_reg(const char * line,const metag_reg ** reg,bfd_boolean load)4247 __parse_gp_reg (const char *line, const metag_reg **reg, bfd_boolean load)
4248 {
4249   const char *l = line;
4250   char reg_buf[MAX_REG_LEN];
4251   size_t len = 0;
4252 
4253   if (l == NULL)
4254     return NULL;
4255 
4256   /* Parse [DSPRAM.x].  */
4257   if (*l == ADDR_BEGIN_CHAR)
4258     {
4259       l++;
4260 
4261       if (l == NULL)
4262 	return NULL;
4263 
4264       l = parse_dsp_reg (l, reg, TRUE, load);
4265       if (l == NULL)
4266 	return NULL;
4267 
4268       if (*l == ADDR_END_CHAR)
4269 	l++;
4270       else
4271 	{
4272 	  as_bad (_("expected ']', not %c in %s"), *l, l);
4273 	  return NULL;
4274 	}
4275 
4276       return l;
4277     }
4278   else
4279     {
4280 
4281       len = strip_reg_name (l, reg_buf);
4282       if (!len)
4283 	return NULL;
4284 
4285       l += len;
4286       *reg = parse_gp_reg (reg_buf);
4287       if (*reg == NULL)
4288 	return NULL;
4289     }
4290 
4291   return l;
4292 }
4293 
4294 /* Parse a list of DSP/GP registers. TRY_GP indicates whether we
4295    should try to parse the register as a general-purpose register if
4296    we fail to parse it as a DSP one. TMPL indicates whether the
4297    registers are part of a template definition instruction. If this is
4298    a template definition instruction LOAD says whether it's a load
4299    template insn. FIRST_DST indicates whether the first register is
4300    a destination operand.  */
4301 static const char *
parse_dsp_regs_list(const char * line,const metag_reg ** regs,size_t count,size_t * regs_read,bfd_boolean try_gp,bfd_boolean tmpl,bfd_boolean load,bfd_boolean first_dst)4302 parse_dsp_regs_list (const char *line, const metag_reg **regs, size_t count,
4303 		     size_t *regs_read, bfd_boolean try_gp, bfd_boolean tmpl,
4304 		     bfd_boolean load, bfd_boolean first_dst)
4305 {
4306   const char *l = line;
4307   int seen_regs = 0;
4308   size_t i;
4309   const metag_reg *reg;
4310 
4311   for (i = 0; i < count; i++)
4312     {
4313       const char *next, *ll;
4314 
4315       next = l;
4316 
4317       if (i > 0)
4318 	{
4319 	  l = skip_comma (l);
4320 	  if (l == NULL)
4321 	    {
4322 	      *regs_read = seen_regs;
4323 	      return next;
4324 	    }
4325 	}
4326 
4327       ll = parse_dsp_reg (l, &reg, tmpl, load);
4328 
4329       if (!ll)
4330 	{
4331 	  if (try_gp)
4332 	    {
4333 	      l = __parse_gp_reg (l, &reg, !(first_dst && i == 0));
4334 	      if (l == NULL)
4335 		{
4336 		  *regs_read = seen_regs;
4337 		  return next;
4338 		}
4339 	      regs[i] = reg;
4340 	      seen_regs++;
4341 	    }
4342 	  else
4343 	    {
4344 	      *regs_read = seen_regs;
4345 	      return l;
4346 	    }
4347 	}
4348       else
4349 	{
4350 	  regs[i] = reg;
4351 	  seen_regs++;
4352 	  l = ll;
4353 	}
4354     }
4355 
4356   *regs_read = seen_regs;
4357   return l;
4358 }
4359 
4360 /* Parse the following memory references:
4361 
4362      - [Ax.r]
4363      - [Ax.r++]
4364      - [Ax.r--]
4365      - [Ax.r+Ax.r++]
4366      - [Ax.r-Ax.r--]
4367 
4368      - [DSPRam]
4369      - [DSPRam++]
4370      - [DSPRam+DSPRam++]
4371      - [DSPRam-DSPRam--]  */
4372 static const char *
parse_dsp_addr(const char * line,metag_addr * addr,unsigned int size,bfd_boolean load)4373 parse_dsp_addr (const char *line, metag_addr *addr, unsigned int size,
4374 		bfd_boolean load)
4375 {
4376   const char *l = line, *ll;
4377   const metag_reg *regs[1];
4378   size_t regs_read;
4379 
4380   /* Skip opening square bracket.  */
4381   l++;
4382 
4383   l = parse_dsp_regs_list (l, regs, 1, &regs_read, TRUE, TRUE, load, FALSE);
4384 
4385   if (l == NULL)
4386     return NULL;
4387 
4388   if (!is_addr_unit (regs[0]->unit) &&
4389       !is_dspram_reg (regs[0]))
4390     {
4391       as_bad (_("invalid register for memory access"));
4392       return NULL;
4393     }
4394 
4395   addr->base_reg = regs[0];
4396 
4397   if (*l == ADDR_END_CHAR)
4398     {
4399       addr->exp.X_op = O_constant;
4400       addr->exp.X_add_symbol = NULL;
4401       addr->exp.X_op_symbol = NULL;
4402 
4403       /* Simple register with no offset (0 immediate).  */
4404       addr->exp.X_add_number = 0;
4405 
4406       addr->immediate = 1;
4407       l++;
4408 
4409       return l;
4410     }
4411 
4412   ll = parse_addr_post_incr_op (l, addr);
4413 
4414   if (ll && *ll == ADDR_END_CHAR)
4415     {
4416       if (addr->update == 1)
4417 	{
4418 	  /* We have a post increment/decrement.  */
4419 	  addr->exp.X_op = O_constant;
4420 	  addr->exp.X_add_number = size;
4421 	  addr->exp.X_add_symbol = NULL;
4422 	  addr->exp.X_op_symbol = NULL;
4423 	  addr->post_increment = 1;
4424 	}
4425       addr->immediate = 1;
4426       ll++;
4427       return ll;
4428     }
4429 
4430   addr->post_increment = 0;
4431 
4432   l = parse_addr_op (l, addr);
4433 
4434   if (l == NULL)
4435     return NULL;
4436 
4437   l = parse_dsp_regs_list (l, regs, 1, &regs_read, TRUE, TRUE, load, FALSE);
4438 
4439   if (l == NULL)
4440     return NULL;
4441 
4442   if (regs[0]->unit != addr->base_reg->unit)
4443     {
4444       as_bad (_("offset and base must be from the same unit"));
4445       return NULL;
4446     }
4447 
4448   addr->offset_reg = regs[0];
4449 
4450   if (*l == ADDR_END_CHAR)
4451     {
4452       l++;
4453       return l;
4454     }
4455 
4456   l = parse_addr_post_incr_op (l, addr);
4457 
4458   if (l == NULL)
4459     return NULL;
4460 
4461   if (*l == ADDR_END_CHAR)
4462     {
4463       l++;
4464       return l;
4465     }
4466 
4467   return NULL;
4468 }
4469 
4470 /* Parse a DSP GET or SET instruction.  */
4471 static const char *
parse_dget_set(const char * line,metag_insn * insn,const insn_template * template)4472 parse_dget_set (const char *line, metag_insn *insn,
4473 		const insn_template *template)
4474 {
4475   const char *l = line;
4476   metag_addr addr;
4477   int unit = 0;
4478   int rd_reg = 0;
4479   bfd_boolean is_get = (template->meta_opcode & 0x100);
4480   bfd_boolean is_dual = (template->meta_opcode & 0x4);
4481   bfd_boolean is_template = FALSE;
4482   const metag_reg *regs[2];
4483   unsigned int size;
4484   size_t count, regs_read;
4485 
4486   memset(&addr, 0, sizeof(addr));
4487   addr.reloc_type = BFD_RELOC_UNUSED;
4488 
4489   size = is_dual ? 8 : 4;
4490   count = is_dual ? 2 : 1;
4491 
4492   if (is_get)
4493     {
4494       /* GETL can be used on one template table entry.  */
4495       if (*l == 'T')
4496 	count = 1;
4497 
4498       l = parse_dsp_regs_list (l, regs, count, &regs_read, FALSE,
4499 			       FALSE, FALSE, FALSE);
4500       l = skip_comma (l);
4501 
4502       if (l == NULL)
4503 	{
4504 	  as_bad (_("unexpected end of line"));
4505 	  return NULL;
4506 	}
4507 
4508       l = parse_addr (l, &addr, size);
4509     }
4510   else
4511     {
4512       l = parse_addr (l, &addr, size);
4513 
4514       l = skip_comma (l);
4515 
4516       if (l == NULL)
4517 	return NULL;
4518 
4519       /* GETL can be used on one template table entry.  */
4520       if (*l == 'T')
4521 	count = 1;
4522 
4523       l = parse_dsp_regs_list (l, regs, count, &regs_read, FALSE, FALSE,
4524 			       FALSE, FALSE);
4525     }
4526 
4527   if (l == NULL)
4528     return NULL;
4529 
4530   /* The first register dictates the unit.  */
4531   if (regs[0]->unit == UNIT_DT)
4532       is_template = TRUE;
4533   else
4534     {
4535       if (regs[0]->unit == UNIT_D0 || regs[0]->unit == UNIT_RAM_D0 ||
4536 	  regs[0]->unit == UNIT_ACC_D0)
4537 	unit = 0;
4538       else
4539 	unit = 1;
4540     }
4541 
4542   rd_reg = regs[0]->no;
4543 
4544   /* The 'H' modifier allows a DSP GET/SET instruction to target the
4545      upper 8-bits of an accumulator. It is _only_ valid for the
4546      accumulators.  */
4547   if (insn->dsp_daoppame_flags & DSP_DAOPPAME_HIGH)
4548     {
4549       if (is_template || !(rd_reg >= 16 && rd_reg < 20))
4550 	{
4551 	  as_bad (_("'H' modifier only valid for accumulator registers"));
4552 	  return NULL;
4553 	}
4554 
4555       /* Top 8-bits of the accumulator.  */
4556       rd_reg |= 8;
4557     }
4558 
4559   if (is_template)
4560     {
4561       insn->bits = (template->meta_opcode | (1 << 1));
4562     }
4563   else
4564     {
4565       insn->bits = (template->meta_opcode | unit);
4566     }
4567 
4568   insn->bits |= (rd_reg << 19);
4569 
4570   if (addr.immediate)
4571     {
4572       int offset = addr.exp.X_add_number;
4573 
4574       if (addr.negate)
4575 	offset = -offset;
4576 
4577       offset = offset / (int)size;
4578 
4579       if (!within_signed_range (offset, DGET_SET_IMM_BITS))
4580 	{
4581 	  as_bad (_("offset value out of range"));
4582 	  return NULL;
4583 	}
4584 
4585       offset = offset & DGET_SET_IMM_MASK;
4586 
4587       insn->bits |= (1 << 13);
4588       insn->bits |= (offset << 9);
4589     }
4590   else
4591     {
4592       int au = (addr.base_reg->unit == UNIT_A1);
4593 
4594       insn->bits |= (au << 18);
4595       insn->bits |= ((addr.base_reg->no & REG_MASK) << 14);
4596       insn->bits |= ((addr.offset_reg->no & REG_MASK) << 9);
4597     }
4598 
4599   if (is_dual)
4600       insn->bits |= (1 << 2);
4601 
4602   if (!is_addr_unit (addr.base_reg->unit))
4603     {
4604       as_bad (_("base unit must be either A0 or A1"));
4605       return NULL;
4606     }
4607 
4608   unit = (addr.base_reg->unit == UNIT_A0) ? 0 : 1;
4609   insn->bits |= ((addr.base_reg->no << 14) | (unit << 18));
4610 
4611   insn->len = 4;
4612 
4613   return l;
4614 }
4615 
4616 /* Parse a DSP template instruction.  */
4617 static const char *
parse_dtemplate(const char * line,metag_insn * insn,const insn_template * template)4618 parse_dtemplate (const char *line, metag_insn *insn,
4619 		 const insn_template *template)
4620 {
4621   const char *l = line;
4622   const metag_reg *regs[TEMPLATE_NUM_REGS];
4623   bfd_boolean daop_only = FALSE;
4624   int regs_val[4];
4625   int regs_which[4] = { -1, -1, -1, -1};	/* Register or immediate?  */
4626   int i;
4627 
4628   for (i = 0; i < TEMPLATE_NUM_REGS; i++)
4629     {
4630       if (l == NULL)
4631 	{
4632 	  as_bad (_("unexpected end of line"));
4633 	  return NULL;
4634 	}
4635 
4636       /* We may only have 3 register operands.  */
4637       if (*l == END_OF_INSN && i == 3)
4638 	{
4639 	  daop_only = TRUE;
4640 	  break;
4641 	}
4642 
4643       if (i != 0)
4644 	{
4645 	  l = skip_comma (l);
4646 	  if (l == NULL)
4647 	    return NULL;
4648 	}
4649 
4650       if (*l == IMM_CHAR)
4651 	{
4652 	  l = parse_imm_constant (l, insn, &regs_val[i]);
4653 	  if (l == NULL)
4654 	    {
4655 	      as_bad (_("invalid immediate"));
4656 	      return NULL;
4657 	    }
4658 	  regs_which[i] = 0;
4659 	}
4660       else
4661 	{
4662 	  /* We can't tell from the template instantiation whether
4663 	     this is a load or store. So we have to try looking up the
4664 	     register name in both the load and store tables.  */
4665 	  const char *l2 = l;
4666 	  l = __parse_gp_reg (l, &regs[i], TRUE);
4667 	  if (l == NULL)
4668 	    {
4669 	      /* Try the store table too.  */
4670 	      l = __parse_gp_reg (l2, &regs[i], FALSE);
4671 	      if (l == NULL)
4672 		{
4673 		  /* Then try a DSP register.  */
4674 		  l = parse_dsp_insn_reg (l2, &regs[i]);
4675 		  if (l == NULL || regs[i]->unit == UNIT_DT)
4676 		    {
4677 		      as_bad (_("invalid register"));
4678 		      return NULL;
4679 		    }
4680 		}
4681 	    }
4682 	  regs_which[i] = 1;
4683 	}
4684     }
4685 
4686   insn->bits = template->meta_opcode;
4687 
4688   if (regs_which[0] == 0)
4689     insn->bits |= (regs_val[0] << 19);
4690   else if (regs_which[0] == 1)
4691     insn->bits |= (regs[0]->no << 19);
4692 
4693   if (regs_which[1] == 0)
4694     insn->bits |= (regs_val[1] << 14);
4695   else if (regs_which[1] == 1)
4696     insn->bits |= (regs[1]->no << 14);
4697 
4698   if (regs_which[2] == 0)
4699     insn->bits |= (regs_val[2] << 9);
4700   else if (regs_which[2] == 1)
4701     insn->bits |= (regs[2]->no << 9);
4702 
4703   if (regs_which[3] == 0)
4704     insn->bits |= (regs_val[3] << 4);
4705   else if (regs_which[3] == 1)
4706     insn->bits |= (regs[3]->no << 4);
4707 
4708   /* DaOp only.  */
4709   if (daop_only)
4710     insn->bits |= (0x3 << 24); /* Set the minor opcode.  */
4711   else if (insn->dsp_daoppame_flags & DSP_DAOPPAME_HIGH) /* Half Load/Store.  */
4712     insn->bits |= (0x5 << 24); /* Set the minor opcode.  */
4713 
4714   insn->len = 4;
4715 
4716   return l;
4717 }
4718 
4719 /* Parse a DSP Template definiton memory reference, e.g
4720    [A0.7+A0.5++]. DSPRAM is set to true by this function if this
4721    template definition is a DSP RAM template definition.  */
4722 static const char *
template_mem_ref(const char * line,metag_addr * addr,bfd_boolean * dspram,int size,bfd_boolean load)4723 template_mem_ref(const char *line, metag_addr *addr,
4724 		 bfd_boolean *dspram, int size, bfd_boolean load)
4725 {
4726   const char *l = line;
4727 
4728   l = parse_dsp_addr (l, addr, size, load);
4729 
4730   if (l != NULL)
4731     {
4732       if (is_addr_unit(addr->base_reg->unit))
4733 	*dspram = FALSE;
4734       else
4735 	*dspram = TRUE;
4736     }
4737 
4738   return l;
4739 }
4740 
4741 /* Sets LOAD to TRUE if this is a Template load definiton (otherwise
4742    it's a store). Fills out ADDR, TEMPLATE_REG and ADDR_UNIT.  */
4743 static const char *
parse_template_regs(const char * line,bfd_boolean * load,unsigned int * addr_unit,const metag_reg ** template_reg,metag_addr * addr,bfd_boolean * dspram,int size)4744 parse_template_regs (const char *line, bfd_boolean *load,
4745 		     unsigned int *addr_unit,
4746 		     const metag_reg **template_reg, metag_addr *addr,
4747 		     bfd_boolean *dspram, int size)
4748 {
4749   const char *l = line;
4750 
4751   if (l == NULL)
4752     return NULL;
4753 
4754   /* DSP Template load definition (Tx, [Ax]) */
4755   if (*l == 'T')
4756     {
4757       *load = TRUE;
4758       l = parse_dsp_reg (l, &template_reg[0], FALSE, FALSE);
4759       if (l == NULL)
4760 	return NULL;
4761 
4762       l = skip_comma (l);
4763 
4764       l = template_mem_ref (l, addr, dspram, size, *load);
4765 
4766       if (addr->base_reg->unit == UNIT_A1)
4767 	*addr_unit = 1;
4768 
4769     }
4770   else if (*l == ADDR_BEGIN_CHAR) /* DSP Template store ([Ax], Tx) */
4771     {
4772       *load = FALSE;
4773       l = template_mem_ref (l, addr, dspram, size, *load);
4774       l = skip_comma(l);
4775 
4776       if (l == NULL)
4777 	return NULL;
4778 
4779       l = parse_dsp_reg (l, &template_reg[0], FALSE, FALSE);
4780       if (l == NULL)
4781 	return NULL;
4782 
4783       if (addr->base_reg->unit == UNIT_A1)
4784 	*addr_unit = 1;
4785     }
4786   else
4787     {
4788       as_bad (_("invalid register operand"));
4789       return NULL;
4790     }
4791 
4792   return l;
4793 }
4794 
4795 #define INVALID_SHIFT (-1)
4796 
4797 static metag_reg _reg;
4798 
4799 /* Parse a template instruction definition.  */
4800 static const char *
interpret_template_regs(const char * line,metag_insn * insn,const metag_reg ** regs,int * regs_shift,bfd_boolean * load,bfd_boolean * dspram,int size,int * ls_shift,int * au_shift,unsigned int * au,int * imm,int * imm_shift,unsigned int * imm_mask)4801 interpret_template_regs(const char *line, metag_insn *insn,
4802 			const metag_reg **regs,
4803 			int *regs_shift, bfd_boolean *load, bfd_boolean *dspram,
4804 			int size, int *ls_shift, int *au_shift,
4805 			unsigned int *au, int *imm, int *imm_shift,
4806 			unsigned int *imm_mask)
4807 {
4808   const char *l = line;
4809   metag_addr addr;
4810   const metag_reg *template_reg[1];
4811 
4812   memset (&addr, 0, sizeof(addr));
4813 
4814   regs_shift[0] = 19;
4815   regs_shift[1] = INVALID_SHIFT;
4816 
4817   insn->bits |= (1 << 1);
4818 
4819   l = skip_whitespace (l);
4820 
4821   l = parse_template_regs (l, load, au, template_reg,
4822 			   &addr, dspram, size);
4823   if (l == NULL)
4824     {
4825       as_bad (_("could not parse template definition"));
4826       return NULL;
4827     }
4828 
4829   regs[2] = template_reg[0];
4830   regs_shift[2] = 9;
4831 
4832   /* DSPRAM definition.  */
4833   if (*dspram)
4834     {
4835 
4836       _reg = *addr.base_reg;
4837 
4838       if (addr.immediate)
4839 	{
4840 	  /* Set the post-increment bit in the register field.  */
4841 	  if (addr.update)
4842 	    _reg.no |= 0x1;
4843 	}
4844       else
4845 	{
4846 	  /* The bottom bit of the increment register tells us
4847 	     whether it's increment register 0 or 1.  */
4848 	  if (addr.offset_reg->no & 0x1)
4849 	    _reg.no |= 0x3;
4850 	  else
4851 	    _reg.no |= 0x2;
4852 	}
4853 
4854       regs[0] = &_reg;
4855 
4856       insn->bits |= (0x3 << 17); /* This signifies a DSPRAM definition.  */
4857     }
4858   else /* DaOpPaMe definition.  */
4859     {
4860       regs[0] = addr.base_reg;
4861       if (addr.immediate)
4862 	{
4863 	  /* Set the I bit.  */
4864 	  insn->bits |= (1 << 18);
4865 
4866 	  if (addr.update == 1)
4867 	    {
4868 	      if (addr.negate == 1)
4869 		*imm = 0x3;
4870 	      else
4871 		*imm = 0x1;
4872 	    }
4873 
4874 	  *imm_shift = 14;
4875 	  *imm_mask = 0x3;
4876 	}
4877       else
4878 	{
4879 	  /* Setup the offset register.  */
4880 	  regs[1] = addr.offset_reg;
4881 	  regs_shift[1] = 14;
4882 	}
4883       *au_shift = 23;
4884     }
4885 
4886   *ls_shift = 13;
4887 
4888   return l;
4889 }
4890 
4891 /* Does this combination of units need the O2R bit and can it be encoded?  */
4892 static bfd_boolean
units_need_o2r(enum metag_unit unit1,enum metag_unit unit2)4893 units_need_o2r (enum metag_unit unit1, enum metag_unit unit2)
4894 {
4895   if (unit1 == unit2)
4896     return FALSE;
4897 
4898   if (unit1 == UNIT_D0 || unit1 == UNIT_ACC_D0 || unit1 == UNIT_RAM_D0)
4899     {
4900       if (unit2 == UNIT_ACC_D0 || unit2 == UNIT_RAM_D0 || unit2 == UNIT_D0)
4901 	return FALSE;
4902 
4903       switch (unit2)
4904 	{
4905 	case UNIT_A1:
4906 	case UNIT_D1:
4907 	case UNIT_RD:
4908 	case UNIT_A0:
4909 	  return TRUE;
4910 	default:
4911 	  return FALSE;
4912 	}
4913     }
4914 
4915   if (unit1 == UNIT_D1 || unit1 == UNIT_ACC_D1 || unit1 == UNIT_RAM_D1)
4916     {
4917       if (unit2 == UNIT_ACC_D1 || unit2 == UNIT_RAM_D1 || unit2 == UNIT_D1)
4918 	return FALSE;
4919 
4920       switch (unit2)
4921 	{
4922 	case UNIT_A1:
4923 	case UNIT_D0:
4924 	case UNIT_RD:
4925 	case UNIT_A0:
4926 	  return TRUE;
4927 	default:
4928 	  return FALSE;
4929 	}
4930     }
4931 
4932   return FALSE;
4933 }
4934 
4935 /* Return TRUE if this is a DSP data unit.  */
4936 static bfd_boolean
is_dsp_data_unit(const metag_reg * reg)4937 is_dsp_data_unit (const metag_reg *reg)
4938 {
4939   switch (reg->unit)
4940     {
4941     case UNIT_D0:
4942     case UNIT_D1:
4943     case UNIT_ACC_D0:
4944     case UNIT_ACC_D1:
4945     case UNIT_RAM_D0:
4946     case UNIT_RAM_D1:
4947       return TRUE;
4948     default:
4949       return FALSE;
4950     }
4951 }
4952 
4953 static metag_reg o2r_reg;
4954 
4955 /* Parse a DaOpPaMe load template definition.  */
4956 static const char *
parse_dalu(const char * line,metag_insn * insn,const insn_template * template)4957 parse_dalu (const char *line, metag_insn *insn,
4958 	    const insn_template *template)
4959 {
4960   const char *l = line;
4961   const char *ll;
4962   const metag_reg *regs[4];
4963   metag_addr addr;
4964   size_t regs_read;
4965   bfd_boolean is_mov = MAJOR_OPCODE (template->meta_opcode) == OPC_ADD;
4966   bfd_boolean is_cmp = ((MAJOR_OPCODE (template->meta_opcode) == OPC_CMP) &&
4967 			((template->meta_opcode & 0xee) == 0));
4968   bfd_boolean is_dual = (insn->dsp_width == DSP_WIDTH_DUAL);
4969   bfd_boolean is_quickrot64 = ((insn->dsp_action_flags & DSP_ACTION_QR64) != 0);
4970   int l1_shift = INVALID_SHIFT;
4971   bfd_boolean load = FALSE;
4972   int ls_shift = INVALID_SHIFT;
4973   bfd_boolean ar = FALSE;
4974   int ar_shift = INVALID_SHIFT;
4975   int regs_shift[3] = { INVALID_SHIFT, INVALID_SHIFT, INVALID_SHIFT };
4976   int imm = 0;
4977   int imm_shift = INVALID_SHIFT;
4978   unsigned int imm_mask = 0;
4979   unsigned int au = 0;
4980   int au_shift = INVALID_SHIFT;
4981   unsigned int du = 0;
4982   int du_shift = INVALID_SHIFT;
4983   unsigned int sc = ((insn->dsp_action_flags & DSP_ACTION_OV) != 0);
4984   int sc_shift = INVALID_SHIFT;
4985   unsigned int om = ((insn->dsp_action_flags & DSP_ACTION_MOD) != 0);
4986   int om_shift = INVALID_SHIFT;
4987   unsigned int o2r = 0;
4988   int o2r_shift = INVALID_SHIFT;
4989   unsigned int qr = 0;
4990   int qr_shift = INVALID_SHIFT;
4991   int qd_shift = INVALID_SHIFT;
4992   unsigned int qn = 0;
4993   int qn_shift = INVALID_SHIFT;
4994   unsigned int a1 = ((insn->dsp_action_flags & (DSP_ACTION_ACC_SUB|DSP_ACTION_ACC_ZERO)) != 0);
4995   int a1_shift = INVALID_SHIFT;
4996   unsigned int a2 = ((insn->dsp_action_flags & (DSP_ACTION_ACC_SUB|DSP_ACTION_ACC_ADD)) != 0);
4997   int a2_shift = INVALID_SHIFT;
4998   unsigned su = ((insn->dsp_action_flags & DSP_ACTION_UMUL) != 0);
4999   int su_shift = INVALID_SHIFT;
5000   unsigned int ac;
5001   int ac_shift = INVALID_SHIFT;
5002   unsigned int mx = (((insn->dsp_daoppame_flags & DSP_DAOPPAME_8) != 0) ||
5003 		     (insn->dsp_daoppame_flags & DSP_DAOPPAME_16) != 0);
5004   int mx_shift = INVALID_SHIFT;
5005   int size = is_dual ? 8 : 4;
5006   bfd_boolean dspram;
5007   bfd_boolean conditional = (MINOR_OPCODE (template->meta_opcode) & 0x4);
5008 
5009   /* XFIXME: check the flags are valid with the instruction.  */
5010   if (is_quickrot64 && !(template->arg_type & DSP_ARGS_QR))
5011     {
5012       as_bad (_("QUICKRoT 64-bit extension not applicable to this instruction"));
5013       return NULL;
5014     }
5015 
5016   insn->bits = template->meta_opcode;
5017 
5018   memset (regs, 0, sizeof (regs));
5019   memset (&addr, 0, sizeof (addr));
5020 
5021   /* There are the following forms of DSP ALU instructions,
5022 
5023    * Group 1:
5024       19. D[T]  Op    De.r,Dx.r,De.r
5025       1.  D[T]  Op    De.r,Dx.r,De.r|ACe.r	[Accumulator in src 2]
5026       3.  D[T]  Op    De.r,Dx.r,De.r[,Ae.r]	[QUICKRoT]
5027       2.  D[T]  Op    ACe.e,ACx.r,ACo.e		[cross-unit accumulator op]
5028       5.  D[T]  Op    De.r|ACe.r,Dx.r,De.r
5029       20. D[T]  Op    De.r,Dx.r|ACx.r,De.r
5030       8.  D     Opcc  De.r,Dx.r,Rx.r
5031       6.  D     Op    De.r,Dx.r,Rx.r|RD
5032       17. D     Op    De.r|ACe.r,Dx.r,Rx.r|RD
5033       7.  D     Op    De.e,Dx.r,#I16
5034 
5035    * Group 2:
5036       4.  D[T]  Op    Dx.r,De.r
5037       10. D     Op    Dx.r,Rx.r|RD
5038       13. D     Op    Dx.r,Rx.r
5039       11. D     Op    Dx.r,#I16
5040       12. D[T]  Op    De.r,Dx.r
5041       14. D     Op    DSPe.r,Dx.r
5042       15. D     Op    DSPx.r,#I16
5043       16. D     Op    De.r,DSPx.r
5044       18. D     Op    De.r,Dx.r|ACx.r
5045 
5046    * Group 3:
5047       22. D     Op    De.r,Dx.r|ACx.r,De.r|#I5
5048       23. D     Op    Ux.r,Dx.r|ACx.r,De.r|#I5
5049       21. D     Op    De.r,Dx.r|ACx.r,#I5  */
5050 
5051   /* Group 1.  */
5052   if (template->arg_type & DSP_ARGS_1)
5053     {
5054       du_shift = 24;
5055 
5056       /* Could this be a cross-unit accumulator op,
5057 	 e.g. ACe.e,ACx.r,ACo.e */
5058       if (template->arg_type & DSP_ARGS_XACC)
5059 	{
5060 	  ll = parse_dsp_regs_list (l, regs, 3, &regs_read, FALSE, FALSE,
5061 				    FALSE, FALSE);
5062 	  if (ll != NULL && regs_read == 3
5063 	      && is_accumulator_reg (regs[0]))
5064 	    {
5065 	      if (regs[0]->unit != regs[1]->unit ||
5066 		  regs[2]->unit == regs[1]->unit)
5067 		{
5068 		  as_bad (_("invalid operands for cross-unit op"));
5069 		  return NULL;
5070 		}
5071 
5072 	      du = (regs[1]->unit == UNIT_ACC_D1);
5073 	      regs_shift[1] = 19;
5074 	      l = ll;
5075 
5076 	      /* All cross-unit accumulator ops have bits 8 and 6 set.  */
5077 	      insn->bits |= (5 << 6);
5078 
5079 	      goto check_for_template;
5080 	    }
5081 
5082 	  /* If we reach here, this instruction is not a
5083 	     cross-unit accumulator op.  */
5084 	}
5085 
5086       if (template->arg_type & DSP_ARGS_SPLIT8)
5087 	om_shift = 7;
5088 
5089       sc_shift = 5;
5090       l1_shift = 4;
5091       o2r_shift = 0;
5092 
5093       /* De.r|ACe.r,Dx.r,De.r */
5094       if (template->arg_type & DSP_ARGS_DACC)
5095 	{
5096 	  /* XFIXME: these need moving?  */
5097 	  a2_shift = 7;
5098 	  su_shift = 6;
5099 	  a1_shift = 2;
5100 	  om_shift = 3;
5101 
5102 	  ll = parse_dsp_reg (l, &regs[0], FALSE, FALSE);
5103 	  if (ll != NULL)
5104 	    {
5105 	      /* Using ACe.r as the dst requires one of the P,N or Z
5106 		 flags to be used.  */
5107 	      if (!(insn->dsp_action_flags &
5108 		    (DSP_ACTION_ACC_SUB|DSP_ACTION_ACC_ADD|DSP_ACTION_ACC_ZERO)))
5109 		{
5110 		  as_bad (_("missing flags: one of 'P', 'N' or 'Z' required"));
5111 		  return NULL;
5112 		}
5113 
5114 	      l = ll;
5115 	      l = skip_comma (l);
5116 	      l = parse_dsp_regs_list (l, &regs[1], 2, &regs_read,
5117 				       TRUE, FALSE, FALSE, FALSE);
5118 	      if (l == NULL || regs_read != 2)
5119 		{
5120 		  as_bad (_("invalid register"));
5121 		  return NULL;
5122 		}
5123 
5124 	      if (regs[1]->unit == UNIT_D1 || regs[1]->unit == UNIT_RAM_D1)
5125 		du = 1;
5126 
5127 	      regs_shift[0] = 19;
5128 	      regs_shift[1] = 14;
5129 	      regs_shift[2] = 9;
5130 	      goto check_for_template;
5131 	    }
5132 
5133 	  /* If we reach here, this instruction does not use the
5134 	     accumulator as the destination register.  */
5135 	  if ((insn->dsp_action_flags &
5136 	       (DSP_ACTION_ACC_SUB|DSP_ACTION_ACC_ADD|DSP_ACTION_ACC_ZERO)))
5137 	    {
5138 	      as_bad (_("'P', 'N' or 'Z' flags may only be specified when accumulating"));
5139 	      return NULL;
5140 	    }
5141 	}
5142 
5143       regs_shift[0] = 19;
5144 
5145 
5146       l = parse_dsp_regs_list (l, regs, 2, &regs_read, TRUE, FALSE, FALSE, TRUE);
5147       if (l == NULL || regs_read != 2)
5148 	return NULL;
5149 
5150       l = skip_comma (l);
5151       if (l == NULL)
5152 	return NULL;
5153 
5154       if (regs[1]->unit == UNIT_D1 || regs[1]->unit == UNIT_RAM_D1)
5155 	du = 1;
5156 
5157       if (is_accumulator_reg(regs[0]) && !(template->arg_type & DSP_ARGS_DACC))
5158        {
5159 	 as_bad (_("accumulator not a valid destination"));
5160 	 return NULL;
5161        }
5162 
5163       /* Check for immediate, e.g. De.r,Dx.r,#I16 */
5164       if (*l == IMM_CHAR)
5165 	{
5166 	  l = parse_imm16 (l, insn, &imm);
5167 	  if (l == NULL)
5168 	    {
5169 	      as_bad (_("invalid immediate value"));
5170 	      return NULL;
5171 	    }
5172 
5173 	  if (!within_signed_range (imm, IMM16_BITS))
5174 	    {
5175 	      as_bad (_("immediate value out of range"));
5176 	      return NULL;
5177 	    }
5178 
5179 	  if (regs[0]->unit != regs[1]->unit || regs[0]->no != regs[1]->no)
5180 	    {
5181 	      as_bad (_("immediate value not allowed when source & dest differ"));
5182 	      return NULL;
5183 	    }
5184 
5185 	  imm_mask = 0xffff;
5186 	  imm_shift = 3;
5187 
5188 	  /* Set the I-bit */
5189 	  insn->bits |= (1 << 25);
5190 
5191 	  insn->bits |= (0x3 << 0);
5192 
5193 	  l1_shift = 2;
5194 
5195 	  /* Remove any bits that have been set in the immediate
5196 	     field.  */
5197 	  insn->bits &= ~(imm_mask << imm_shift);
5198 	}
5199       else
5200 	{
5201 
5202 	  regs_shift[1] = 14;
5203 	  regs_shift[2] = 9;
5204 
5205 	  /* Is Rs2 an accumulator reg, e.g. De.r,Dx.r,De.r|ACe.r */
5206 	  ll = parse_dsp_reg (l, &regs[2], FALSE, FALSE);
5207 	  if (ll != NULL)
5208 	    {
5209 	      l = ll;
5210 
5211 	      if (!(template->arg_type & DSP_ARGS_ACC2))
5212 		{
5213 		  as_bad (_("invalid register operand: %s"), regs[2]->name);
5214 		  return NULL;
5215 		}
5216 
5217 	      om_shift = 3;
5218 	      ar_shift = 7;
5219 	      ar = TRUE;
5220 	    }
5221 	  else
5222 	    {
5223 	      /* De.r,Dx.r,De.r */
5224 	      l = __parse_gp_reg (l, &regs[2], TRUE);
5225 	      if (l == NULL)
5226 		return NULL;
5227 	    }
5228 
5229 	  if (template->arg_type & DSP_ARGS_ACC2)
5230 	    om_shift = 3;
5231 
5232 	  /* Is this a QUICKRoT instruction? De.r,Dx.r,De.r[,Ae.r] */
5233 	  if (template->arg_type & DSP_ARGS_QR)
5234 	    {
5235 	      if (conditional)
5236 		qn_shift = 5;
5237 	      else
5238 		{
5239 		  qn_shift = 7;
5240 		  qr_shift = 6;
5241 		  qd_shift = 5;
5242 		}
5243 
5244 	      l = skip_comma (l);
5245 	      if (l == NULL)
5246 		{
5247 		  as_bad (_("QUICKRoT extension requires 4 registers"));
5248 		  return NULL;
5249 		}
5250 
5251 	      l = __parse_gp_reg (l, &regs[3], TRUE);
5252 	      if (l == NULL)
5253 		{
5254 		  as_bad (_("invalid fourth register"));
5255 		  return NULL;
5256 		}
5257 
5258 	      if (!is_addr_unit (regs[3]->unit) ||
5259 		  !is_quickrot_reg (regs[3]->no))
5260 		{
5261 		  as_bad (_("A0.2,A0.3,A1.2,A1.3 required for QUICKRoT register"));
5262 		  return NULL;
5263 		}
5264 
5265 	      qn = (regs[3]->no == 3);
5266 	    }
5267 	}
5268 
5269     check_for_template:
5270       /* This is the common exit path. Check for o2r.  */
5271       if (regs[2] != NULL)
5272 	{
5273 	  o2r = units_need_o2r (regs[1]->unit, regs[2]->unit);
5274 	  if (o2r)
5275 	    {
5276 	      o2r_reg.no = lookup_o2r (0, du, regs[2]);
5277 	      o2r_reg.unit = regs[2]->unit;
5278 	      regs[2] = &o2r_reg;
5279 	    }
5280 	}
5281 
5282       /* Check any DSP RAM pointers are valid for this unit.  */
5283       if ((du && (regs[0]->unit == UNIT_RAM_D0)) ||
5284 	  (!du && (regs[0]->unit == UNIT_RAM_D1)) ||
5285 	  (du && (regs[1]->unit == UNIT_RAM_D0)) ||
5286 	  (!du && (regs[1]->unit == UNIT_RAM_D1)) ||
5287 	  (du && regs[2] && (regs[2]->unit == UNIT_RAM_D0)) ||
5288 	  (!du && regs[2] && (regs[2]->unit == UNIT_RAM_D1))) {
5289 	as_bad (_("DSP RAM pointer in incorrect unit"));
5290 	return NULL;
5291       }
5292 
5293       /* Is this a template definition?  */
5294       if (IS_TEMPLATE_DEF (insn))
5295 	{
5296 	  l = interpret_template_regs(l, insn, regs, regs_shift, &load,
5297 				      &dspram, size, &ls_shift, &au_shift,
5298 				      &au, &imm, &imm_shift, &imm_mask);
5299 
5300 	  if (l == NULL)
5301 	    return NULL;
5302 
5303 	  if (!dspram)
5304 	    mx_shift = 0;
5305 	}
5306 
5307       goto matched;
5308     }
5309 
5310   /* Group 2.  */
5311   if (template->arg_type & DSP_ARGS_2)
5312     {
5313       bfd_boolean is_xsd = ((MAJOR_OPCODE (template->meta_opcode) == OPC_MISC) &&
5314 			    (MINOR_OPCODE (template->meta_opcode) == 0xa));
5315       bfd_boolean is_fpu_mov = template->insn_type == INSN_DSP_FPU;
5316       bfd_boolean to_fpu = (template->meta_opcode >> 7) & 0x1;
5317 
5318       if (is_xsd)
5319 	du_shift = 0;
5320       else
5321 	du_shift = 24;
5322 
5323       l1_shift = 4;
5324 
5325       /* CMPs and TSTs don't store to their destination operand.  */
5326       ll = __parse_gp_reg (l, regs, is_cmp);
5327       if (ll == NULL)
5328 	{
5329 	  /* DSPe.r,Dx.r or DSPx.r,#I16 */
5330 	  if (template->arg_type & DSP_ARGS_DSP_SRC1)
5331 	    {
5332 	      l = parse_dsp_reg (l, regs, FALSE, FALSE);
5333 	      if (l == NULL)
5334 		{
5335 		  as_bad (_("invalid register operand #1"));
5336 		  return NULL;
5337 		}
5338 
5339 	      /* Only MOV instructions have a DSP register as a
5340 		 destination. Set the MOV DSPe.r opcode. The simple
5341 		 OR'ing is OK because the usual MOV opcode is 0x00.  */
5342 	      insn->bits = (0x91 << 24);
5343 	      du_shift = 0;
5344 	      l1_shift = 2;
5345 	      regs_shift[0] = 19;
5346 	    }
5347 	  else
5348 	    {
5349 	      as_bad (_("invalid register operand #2"));
5350 	      return NULL;
5351 	    }
5352 	}
5353       else
5354 	{
5355 	  l = ll;
5356 
5357 	  /* Everything but CMP and TST.  */
5358 	  if (MAJOR_OPCODE (template->meta_opcode) == OPC_ADD ||
5359 	      MAJOR_OPCODE (template->meta_opcode) == OPC_SUB ||
5360 	      MAJOR_OPCODE (insn->bits) == OPC_9 ||
5361 	      MAJOR_OPCODE (template->meta_opcode) == OPC_MISC ||
5362 	      ((template->meta_opcode & 0x0000002c) != 0))
5363 	    regs_shift[0] = 19;
5364 	  else
5365 	    regs_shift[0] = 14;
5366 	}
5367 
5368       if (!is_dsp_data_unit (regs[0]) && !(regs[0]->unit == UNIT_FX &&
5369 					   is_fpu_mov && to_fpu))
5370 	return NULL;
5371 
5372       du = (regs[0]->unit == UNIT_D1 || regs[0]->unit == UNIT_RAM_D1 ||
5373 	    regs[0]->unit == UNIT_ACC_D1);
5374 
5375       l = skip_comma (l);
5376 
5377       if (*l == IMM_CHAR)
5378 	{
5379 	  if (template->arg_type & DSP_ARGS_IMM &&
5380 	      !(is_mov && (MAJOR_OPCODE (insn->bits) != OPC_9)))
5381 	    {
5382 	      l = parse_imm16 (l, insn, &imm);
5383 	      if (l == NULL)
5384 		{
5385 		  as_bad (_("invalid immediate value"));
5386 		  return NULL;
5387 		}
5388 
5389 	      if (!within_signed_range (imm, IMM16_BITS))
5390 		return NULL;
5391 
5392 	      l1_shift = 2;
5393 	      regs_shift[0] = 19;
5394 
5395 	      imm_mask = 0xffff;
5396 	      imm_shift = 3;
5397 
5398 	      /* Set the I-bit unless it's a MOV because they're
5399 		 different.  */
5400 	      if (!(is_mov && MAJOR_OPCODE (insn->bits) == OPC_9))
5401 		insn->bits |= (1 << 25);
5402 
5403 	      /* All instructions that takes immediates also have bit 1 set.  */
5404 	      insn->bits |= (1 << 1);
5405 
5406 	      if (MAJOR_OPCODE (insn->bits) != OPC_9)
5407 		insn->bits |= (1 << 0);
5408 
5409 	      insn->bits &= ~(1 << 8);
5410 	    }
5411 	  else
5412 	    {
5413 	      as_bad (_("this instruction does not accept an immediate"));
5414 	      return NULL;
5415 	    }
5416 	}
5417       else
5418 	{
5419 	  if (MAJOR_OPCODE (insn->bits) != OPC_9)
5420 	    {
5421 	      insn->bits |= (1 << 8);
5422 	      l1_shift = 4;
5423 	    }
5424 
5425 	  ll = __parse_gp_reg (l, &regs[1], TRUE);
5426 	  if (ll == NULL)
5427 	    {
5428 	      if (template->arg_type & DSP_ARGS_DSP_SRC2)
5429 		{
5430 		  l = parse_dsp_reg (l, &regs[1], FALSE, FALSE);
5431 		  if (l == NULL)
5432 		    {
5433 		      as_bad (_("invalid register operand #3"));
5434 		      return NULL;
5435 		    }
5436 
5437 		  /* MOV and NEG.  */
5438 		  if ((is_mov && (MAJOR_OPCODE (insn->bits) != OPC_9)) ||
5439 		      MAJOR_OPCODE (template->meta_opcode) == OPC_SUB)
5440 		    {
5441 		      if (is_accumulator_reg (regs[1]))
5442 			{
5443 			  if (is_fpu_mov)
5444 			    {
5445 			      as_bad (_("this instruction does not accept an accumulator"));
5446 			      return NULL;
5447 			    }
5448 			  ar_shift = 7;
5449 			  ar = 1;
5450 			  regs_shift[1] = 9;
5451 			}
5452 		      else
5453 			{
5454 			  du_shift = 0;
5455 			  l1_shift = 2;
5456 			  regs_shift[1] = 14;
5457 			  insn->bits = (0x92 << 24); /* Set opcode.  */
5458 			}
5459 		    }
5460 		}
5461 	      else
5462 		{
5463 		  as_bad (_("invalid register operand #4"));
5464 		  return NULL;
5465 		}
5466 	    }
5467 	  else
5468 	    {
5469 	      /* Set the o2r bit if required.  */
5470 	      if (!is_fpu_mov && units_need_o2r (regs[0]->unit, regs[1]->unit))
5471 		{
5472 		  o2r_reg = *regs[1];
5473 		  o2r_reg.no = lookup_o2r (0, du, regs[1]);
5474 		  regs[1] = &o2r_reg;
5475 		  o2r_shift = 0;
5476 		  o2r = 1;
5477 		}
5478 	      else if (!is_dsp_data_unit (regs[1]) &&
5479 		       !(is_fpu_mov && !to_fpu && regs[1]->unit == UNIT_FX))
5480 		return NULL;
5481 
5482 	      if (is_fpu_mov && to_fpu)
5483 		du = (regs[1]->unit == UNIT_D1 ||
5484 		      regs[1]->unit == UNIT_RAM_D1 ||
5485 		      regs[1]->unit == UNIT_ACC_D1);
5486 
5487 	      l = ll;
5488 
5489 	      if (MAJOR_OPCODE (insn->bits) == OPC_ADD ||
5490 		  MAJOR_OPCODE (template->meta_opcode) == OPC_SUB ||
5491 		  (((template->meta_opcode & 0x0000002c) == 0) &&
5492 		   MAJOR_OPCODE (template->meta_opcode) != OPC_MISC))
5493 		regs_shift[1] = 9;
5494 	      else
5495 		regs_shift[1] = 14;
5496 	    }
5497 	}
5498 
5499       /* If it's an 0x0 MOV or NEG set some lower bits.  */
5500       if ((MAJOR_OPCODE (insn->bits) == OPC_ADD ||
5501 	   MAJOR_OPCODE (template->meta_opcode) == OPC_SUB) && !is_fpu_mov)
5502 	{
5503 	  om_shift = 3;
5504 	  sc_shift = 5;
5505 	  insn->bits |= (1 << 2);
5506 	}
5507 
5508       /* Check for template definitons.  */
5509       if (IS_TEMPLATE_DEF (insn))
5510 	{
5511 	  l = interpret_template_regs(l, insn, regs, regs_shift, &load,
5512 				      &dspram, size, &ls_shift, &au_shift,
5513 				      &au, &imm, &imm_shift, &imm_mask);
5514 	  mx_shift = 0;
5515 
5516 	  if (l == NULL)
5517 	    return NULL;
5518 	}
5519       goto matched;
5520     }
5521 
5522   /* Group 3.  */
5523   du_shift = 24;
5524   l1_shift = 4;
5525 
5526   l = __parse_gp_reg (l, regs, FALSE);
5527   if (l == NULL)
5528     {
5529       as_bad (_("invalid register operand"));
5530       return NULL;
5531     }
5532 
5533   l = skip_comma (l);
5534 
5535   if (*l == 'A')
5536     {
5537       l = parse_dsp_reg (l, &regs[1], FALSE, FALSE);
5538       if (l == NULL)
5539 	{
5540 	  as_bad (_("invalid accumulator register"));
5541 	  return NULL;
5542 	}
5543       ac = 1;
5544       ac_shift = 0;
5545     }
5546   else
5547     {
5548       l = __parse_gp_reg (l, &regs[1], TRUE);
5549       if (l == NULL)
5550 	{
5551 	  as_bad (_("invalid register operand"));
5552 	  return NULL;
5553 	}
5554     }
5555 
5556   regs_shift[0] = 19;
5557   regs_shift[1] = 14;
5558 
5559   du = (regs[1]->unit == UNIT_D1 || regs[1]->unit == UNIT_ACC_D1
5560 	|| regs[1]->unit == UNIT_RAM_D1);
5561 
5562   l = skip_comma (l);
5563 
5564   if (*l == IMM_CHAR)
5565     {
5566       l = parse_imm_constant (l, insn, &imm);
5567       if (l == NULL)
5568 	{
5569 	  as_bad (_("invalid immediate value"));
5570 	  return NULL;
5571 	}
5572 
5573       if (!within_unsigned_range (imm, IMM5_BITS))
5574 	return NULL;
5575 
5576       imm_mask = 0x1f;
5577       imm_shift = 9;
5578 
5579       /* Set the I-bit */
5580       insn->bits |= (1 << 25);
5581     }
5582   else
5583     {
5584       regs_shift[2] = 9;
5585       l = __parse_gp_reg (l, &regs[2], TRUE);
5586       if (l == NULL)
5587 	return NULL;
5588     }
5589 
5590   /* Check for post-processing R,G,B flags. Conditional instructions
5591      do not have these bits.  */
5592   if (insn->dsp_action_flags & DSP_ACTION_CLAMP9)
5593     {
5594       if ((template->meta_opcode >> 26) & 0x1)
5595 	{
5596 	  as_bad (_("conditional instruction cannot use G flag"));
5597 	  return NULL;
5598 	}
5599 
5600       insn->bits |= (1 << 3);
5601     }
5602 
5603   if (insn->dsp_action_flags & DSP_ACTION_CLAMP8)
5604     {
5605       if ((template->meta_opcode >> 26) & 0x1)
5606 	{
5607 	  as_bad (_("conditional instruction cannot use B flag"));
5608 	  return NULL;
5609 	}
5610 
5611       insn->bits |= (0x3 << 2);
5612     }
5613 
5614   if (insn->dsp_action_flags & DSP_ACTION_ROUND)
5615     {
5616       if ((template->meta_opcode >> 26) & 0x1)
5617 	{
5618 	  as_bad (_("conditional instruction cannot use R flag"));
5619 	  return NULL;
5620 	}
5621       insn->bits |= (1 << 2);
5622     }
5623 
5624   /* Conditional Data Unit Shift instructions cannot be dual unit.  */
5625   if ((template->meta_opcode >> 26) & 0x1)
5626     ls_shift = INVALID_SHIFT;
5627 
5628   /* The Condition Is Always (CA) bit must be set if we're targetting a
5629      Ux.r register as the destination. This means that we can't have
5630      any other condition bits set.  */
5631   if (!is_same_data_unit (regs[1]->unit, regs[0]->unit))
5632     {
5633       /* Set both the Conditional bit and the Condition is Always bit.  */
5634       insn->bits |= (1 << 26);
5635       insn->bits |= (1 << 5);
5636 
5637       /* Fill out the Ud field.  */
5638       insn->bits |= (regs[0]->unit << 1);
5639     }
5640 
5641   if (IS_TEMPLATE_DEF (insn))
5642     {
5643       l = interpret_template_regs(l, insn, regs, regs_shift, &load,
5644 				  &dspram, size, &ls_shift, &au_shift,
5645 				  &au, &imm, &imm_shift, &imm_mask);
5646 
5647       if (l == NULL)
5648 	return NULL;
5649 
5650       if (!dspram)
5651 	mx_shift = 5;
5652     }
5653 
5654   /* Fall through.  */
5655  matched:
5656 
5657   /* Set the registers and immediate values.  */
5658   if (regs_shift[0] != INVALID_SHIFT)
5659     insn->bits |= (regs[0]->no << regs_shift[0]);
5660 
5661   if (regs_shift[1] != INVALID_SHIFT)
5662     insn->bits |= (regs[1]->no << regs_shift[1]);
5663 
5664   if (regs_shift[2] != INVALID_SHIFT)
5665     insn->bits |= (regs[2]->no << regs_shift[2]);
5666 
5667   /* Does this insn have an 'IMM' bit? The immediate value should
5668      already have been masked.  */
5669   if (imm_shift != INVALID_SHIFT)
5670     insn->bits |= ((imm & imm_mask) << imm_shift);
5671 
5672   /* Does this insn have an 'AU' bit? */
5673   if (au_shift != INVALID_SHIFT)
5674     insn->bits |= (au << au_shift);
5675 
5676   /* Does this instruction have an 'LS' bit?  */
5677   if (ls_shift != INVALID_SHIFT)
5678     insn->bits |= (load << ls_shift);
5679 
5680   /* Does this instruction have an 'AR' bit?  */
5681   if (ar)
5682       insn->bits |= (1 << ar_shift);
5683 
5684   if (du_shift != INVALID_SHIFT)
5685     insn->bits |= (du << du_shift);
5686 
5687   if (sc_shift != INVALID_SHIFT)
5688     insn->bits |= (sc << sc_shift);
5689 
5690   if (om_shift != INVALID_SHIFT)
5691     insn->bits |= (om << om_shift);
5692 
5693   if (o2r_shift != INVALID_SHIFT)
5694     insn->bits |= (o2r << o2r_shift);
5695 
5696   if (qn_shift != INVALID_SHIFT)
5697     insn->bits |= (qn << qn_shift);
5698 
5699   if (qr_shift != INVALID_SHIFT)
5700     insn->bits |= (qr << qr_shift);
5701 
5702   if (qd_shift != INVALID_SHIFT)
5703     insn->bits |= (is_quickrot64 << qd_shift);
5704 
5705   if (a1_shift != INVALID_SHIFT)
5706     insn->bits |= (a1 << a1_shift);
5707 
5708   if (a2_shift != INVALID_SHIFT)
5709     insn->bits |= (a2 << a2_shift);
5710 
5711   if (su_shift != INVALID_SHIFT)
5712     insn->bits |= (su << su_shift);
5713 
5714   if (imm_shift != INVALID_SHIFT)
5715     insn->bits |= ((imm & imm_mask) << imm_shift);
5716 
5717   if (ac_shift != INVALID_SHIFT)
5718     insn->bits |= (ac << ac_shift);
5719 
5720   if (mx_shift != INVALID_SHIFT)
5721     insn->bits |= (mx << mx_shift);
5722 
5723   if (is_dual)
5724     {
5725       if (l1_shift == INVALID_SHIFT)
5726 	{
5727 	  as_bad (_("'L' modifier not valid for this instruction"));
5728 	  return NULL;
5729 	}
5730 
5731       insn->bits |= (1 << l1_shift);
5732     }
5733 
5734   insn->len = 4;
5735 
5736   return l;
5737 }
5738 
5739 typedef const char *(*insn_parser)(const char *, metag_insn *,
5740 				   const insn_template *);
5741 
5742 /* Parser table.  */
5743 static const insn_parser insn_parsers[ENC_MAX] =
5744   {
5745     [ENC_NONE] = parse_none,
5746     [ENC_MOV_U2U] = parse_mov_u2u,
5747     [ENC_MOV_PORT] = parse_mov_port,
5748     [ENC_MMOV] = parse_mmov,
5749     [ENC_MDRD] = parse_mdrd,
5750     [ENC_MOVL_TTREC] = parse_movl_ttrec,
5751     [ENC_GET_SET] = parse_get_set,
5752     [ENC_GET_SET_EXT] = parse_get_set_ext,
5753     [ENC_MGET_MSET] = parse_mget_mset,
5754     [ENC_COND_SET] = parse_cond_set,
5755     [ENC_XFR] = parse_xfr,
5756     [ENC_MOV_CT] = parse_mov_ct,
5757     [ENC_SWAP] = parse_swap,
5758     [ENC_JUMP] = parse_jump,
5759     [ENC_CALLR] = parse_callr,
5760     [ENC_ALU] = parse_alu,
5761     [ENC_SHIFT] = parse_shift,
5762     [ENC_MIN_MAX] = parse_min_max,
5763     [ENC_BITOP] = parse_bitop,
5764     [ENC_CMP] = parse_cmp,
5765     [ENC_BRANCH] = parse_branch,
5766     [ENC_KICK] = parse_kick,
5767     [ENC_SWITCH] = parse_switch,
5768     [ENC_CACHER] = parse_cacher,
5769     [ENC_CACHEW] = parse_cachew,
5770     [ENC_ICACHE] = parse_icache,
5771     [ENC_LNKGET] = parse_lnkget,
5772     [ENC_FMOV] = parse_fmov,
5773     [ENC_FMMOV] = parse_fmmov,
5774     [ENC_FMOV_DATA] = parse_fmov_data,
5775     [ENC_FMOV_I] = parse_fmov_i,
5776     [ENC_FPACK] = parse_fpack,
5777     [ENC_FSWAP] = parse_fswap,
5778     [ENC_FCMP] = parse_fcmp,
5779     [ENC_FMINMAX] = parse_fminmax,
5780     [ENC_FCONV] = parse_fconv,
5781     [ENC_FCONVX] = parse_fconvx,
5782     [ENC_FBARITH] = parse_fbarith,
5783     [ENC_FEARITH] = parse_fearith,
5784     [ENC_FREC] = parse_frec,
5785     [ENC_FSIMD] = parse_fsimd,
5786     [ENC_FGET_SET_ACF] = parse_fget_set_acf,
5787     [ENC_DGET_SET] = parse_dget_set,
5788     [ENC_DTEMPLATE] = parse_dtemplate,
5789     [ENC_DALU] = parse_dalu,
5790   };
5791 
5792 struct metag_core_option
5793 {
5794   char *name;
5795   unsigned int value;
5796 };
5797 
5798 /* CPU type options.  */
5799 static const struct metag_core_option metag_cpus[] =
5800   {
5801     {"all",               CoreMeta11|CoreMeta12|CoreMeta21},
5802     {"metac11",           CoreMeta11},
5803     {"metac12",           CoreMeta12},
5804     {"metac21",           CoreMeta21},
5805     {NULL,                0},
5806   };
5807 
5808 /* FPU type options.  */
5809 static const struct metag_core_option metag_fpus[] =
5810   {
5811     {"metac21",           FpuMeta21},
5812     {NULL,                0},
5813   };
5814 
5815 /* DSP type options.  */
5816 static const struct metag_core_option metag_dsps[] =
5817   {
5818     {"metac21",           DspMeta21},
5819     {NULL,                0},
5820   };
5821 
5822 /* Parse a CPU command line option.  */
5823 static int
metag_parse_cpu(char * str)5824 metag_parse_cpu (char * str)
5825 {
5826   const struct metag_core_option * opt;
5827   int optlen;
5828 
5829   optlen = strlen (str);
5830 
5831   if (optlen == 0)
5832     {
5833       as_bad (_("missing cpu name `%s'"), str);
5834       return 0;
5835     }
5836 
5837   for (opt = metag_cpus; opt->name != NULL; opt++)
5838     if (strncmp (opt->name, str, optlen) == 0)
5839       {
5840 	mcpu_opt = opt->value;
5841 	return 1;
5842       }
5843 
5844   as_bad (_("unknown cpu `%s'"), str);
5845   return 0;
5846 }
5847 
5848 /* Parse an FPU command line option.  */
5849 static int
metag_parse_fpu(char * str)5850 metag_parse_fpu (char * str)
5851 {
5852   const struct metag_core_option * opt;
5853   int optlen;
5854 
5855   optlen = strlen (str);
5856 
5857   if (optlen == 0)
5858     {
5859       as_bad (_("missing fpu name `%s'"), str);
5860       return 0;
5861     }
5862 
5863   for (opt = metag_fpus; opt->name != NULL; opt++)
5864     if (strncmp (opt->name, str, optlen) == 0)
5865       {
5866 	mfpu_opt = opt->value;
5867 	return 1;
5868       }
5869 
5870   as_bad (_("unknown fpu `%s'"), str);
5871   return 0;
5872 }
5873 
5874 /* Parse a DSP command line option.  */
5875 static int
metag_parse_dsp(char * str)5876 metag_parse_dsp (char * str)
5877 {
5878   const struct metag_core_option * opt;
5879   int optlen;
5880 
5881   optlen = strlen (str);
5882 
5883   if (optlen == 0)
5884     {
5885       as_bad (_("missing DSP name `%s'"), str);
5886       return 0;
5887     }
5888 
5889   for (opt = metag_dsps; opt->name != NULL; opt++)
5890     if (strncmp (opt->name, str, optlen) == 0)
5891       {
5892 	mdsp_opt = opt->value;
5893 	return 1;
5894       }
5895 
5896   as_bad (_("unknown DSP `%s'"), str);
5897   return 0;
5898 }
5899 
5900 struct metag_long_option
5901 {
5902   char * option;                /* Substring to match.  */
5903   char * help;                  /* Help information.  */
5904   int (* func) (char * subopt); /* Function to decode sub-option.  */
5905   char * deprecated;            /* If non-null, print this message.  */
5906 };
5907 
5908 struct metag_long_option metag_long_opts[] =
5909   {
5910     {"mcpu=", N_("<cpu name>\t  assemble for CPU <cpu name>"),
5911      metag_parse_cpu, NULL},
5912     {"mfpu=", N_("<fpu name>\t  assemble for FPU architecture <fpu name>"),
5913      metag_parse_fpu, NULL},
5914     {"mdsp=", N_("<dsp name>\t  assemble for DSP architecture <dsp name>"),
5915      metag_parse_dsp, NULL},
5916     {NULL, NULL, 0, NULL}
5917   };
5918 
5919 int
md_parse_option(int c,char * arg)5920 md_parse_option (int c, char * arg)
5921 {
5922   struct metag_long_option *lopt;
5923 
5924   for (lopt = metag_long_opts; lopt->option != NULL; lopt++)
5925     {
5926       /* These options are expected to have an argument.  */
5927       if (c == lopt->option[0]
5928 	  && arg != NULL
5929 	  && strncmp (arg, lopt->option + 1,
5930 		      strlen (lopt->option + 1)) == 0)
5931 	{
5932 #if WARN_DEPRECATED
5933 	      /* If the option is deprecated, tell the user.  */
5934 	      if (lopt->deprecated != NULL)
5935 		as_tsktsk (_("option `-%c%s' is deprecated: %s"), c, arg,
5936 			   _(lopt->deprecated));
5937 #endif
5938 
5939 	      /* Call the sup-option parser.  */
5940 	      return lopt->func (arg + strlen (lopt->option) - 1);
5941 	}
5942     }
5943 
5944   return 0;
5945 }
5946 
5947 void
md_show_usage(FILE * stream)5948 md_show_usage (FILE * stream)
5949 {
5950   struct metag_long_option *lopt;
5951 
5952   fprintf (stream, _(" Meta specific command line options:\n"));
5953 
5954   for (lopt = metag_long_opts; lopt->option != NULL; lopt++)
5955     if (lopt->help != NULL)
5956       fprintf (stream, "  -%s%s\n", lopt->option, _(lopt->help));
5957 }
5958 
5959 /* The target specific pseudo-ops which we support.  */
5960 const pseudo_typeS md_pseudo_table[] =
5961 {
5962   { "word",	cons,		2 },
5963   { NULL, 	NULL, 		0 }
5964 };
5965 
5966 void
md_begin(void)5967 md_begin (void)
5968 {
5969   int c;
5970 
5971   for (c = 0; c < 256; c++)
5972     {
5973       if (ISDIGIT (c))
5974 	{
5975 	  register_chars[c] = c;
5976 	  /* LOCK0, LOCK1, LOCK2.  */
5977 	  mnemonic_chars[c] = c;
5978 	}
5979       else if (ISLOWER (c))
5980 	{
5981 	  register_chars[c] = c;
5982 	  mnemonic_chars[c] = c;
5983 	}
5984       else if (ISUPPER (c))
5985 	{
5986 	  register_chars[c] = c;
5987 	  mnemonic_chars[c] = c;
5988 	}
5989       else if (c == '.')
5990 	{
5991 	  register_chars[c] = c;
5992 	}
5993     }
5994 }
5995 
5996 /* Parse a split condition code prefix.  */
5997 static const char *
parse_split_condition(const char * line,metag_insn * insn)5998 parse_split_condition (const char *line, metag_insn *insn)
5999 {
6000   const char *l = line;
6001   const split_condition *scond;
6002   split_condition entry;
6003   char buf[4];
6004 
6005   memcpy (buf, l, 4);
6006   buf[3] = '\0';
6007 
6008   entry.name = buf;
6009 
6010   scond = (const split_condition *) htab_find (scond_htab, &entry);
6011 
6012   if (!scond)
6013     return NULL;
6014 
6015   insn->scond = scond->code;
6016 
6017   return l + strlen (scond->name);
6018 }
6019 
6020 /* Parse an instruction prefix - F for float, D for DSP - and associated
6021    flags and condition codes.  */
6022 static const char *
parse_prefix(const char * line,metag_insn * insn)6023 parse_prefix (const char *line, metag_insn *insn)
6024 {
6025   const char *l = line;
6026 
6027   l = skip_whitespace (l);
6028 
6029   insn->type = INSN_GP;
6030 
6031   if (TOLOWER (*l) == FPU_PREFIX_CHAR)
6032     {
6033       if (strncasecmp (l, FFB_INSN, strlen(FFB_INSN)))
6034 	{
6035 	  insn->type = INSN_FPU;
6036 
6037 	  l++;
6038 
6039 	  if (*l == END_OF_INSN)
6040 	    {
6041 	      as_bad (_("premature end of floating point prefix"));
6042 	      return NULL;
6043 	    }
6044 
6045 	  if (TOLOWER (*l) == FPU_DOUBLE_CHAR)
6046 	    {
6047 	      insn->fpu_width = FPU_WIDTH_DOUBLE;
6048 	      l++;
6049 	    }
6050 	  else if (TOLOWER (*l) == FPU_PAIR_CHAR)
6051 	    {
6052 	      const char *l2 = l;
6053 
6054 	      /* Check this isn't a split condition beginning with L.  */
6055 	      l2 = parse_split_condition (l2, insn);
6056 
6057 	      if (l2 && is_whitespace_char (*l2))
6058 		{
6059 		  l = l2;
6060 		}
6061 	      else
6062 		{
6063 		  insn->fpu_width = FPU_WIDTH_PAIR;
6064 		  l++;
6065 		}
6066 	    }
6067 	  else
6068 	    {
6069 	      insn->fpu_width = FPU_WIDTH_SINGLE;
6070 	    }
6071 
6072 	  if (TOLOWER (*l) == FPU_ACTION_ABS_CHAR)
6073 	    {
6074 	      insn->fpu_action_flags |= FPU_ACTION_ABS;
6075 	      l++;
6076 	    }
6077 	  else if (TOLOWER (*l) == FPU_ACTION_INV_CHAR)
6078 	    {
6079 	      insn->fpu_action_flags |= FPU_ACTION_INV;
6080 	      l++;
6081 	    }
6082 
6083 	  if (TOLOWER (*l) == FPU_ACTION_QUIET_CHAR)
6084 	    {
6085 	      insn->fpu_action_flags |= FPU_ACTION_QUIET;
6086 	      l++;
6087 	    }
6088 
6089 	  if (TOLOWER (*l) == FPU_ACTION_ZERO_CHAR)
6090 	    {
6091 	      insn->fpu_action_flags |= FPU_ACTION_ZERO;
6092 	      l++;
6093 	    }
6094 
6095 	  if (! is_whitespace_char (*l))
6096 	    {
6097 	      l = parse_split_condition (l, insn);
6098 
6099 	      if (!l)
6100 		{
6101 		  as_bad (_("unknown floating point prefix character"));
6102 		  return NULL;
6103 		}
6104 	    }
6105 
6106 	  l = skip_space (l);
6107 	}
6108     }
6109   else if (TOLOWER (*l) == DSP_PREFIX_CHAR)
6110     {
6111       if (strncasecmp (l, DCACHE_INSN, strlen (DCACHE_INSN)) &&
6112 	  strncasecmp (l, DEFR_INSN, strlen (DEFR_INSN)))
6113 	{
6114 	  const char *ll = l;
6115 	  insn->type = INSN_DSP;
6116 
6117 	  l++;
6118 
6119 	  insn->dsp_width = DSP_WIDTH_SINGLE;
6120 
6121 	  while (!is_whitespace_char (*l))
6122 	    {
6123 	      /* We have to check for split condition codes first
6124 		 because they are the longest strings to match,
6125 		 e.g. if the string contains "LLS" we want it to match
6126 		 the split condition code "LLS", not the dual unit
6127 		 character "L".  */
6128 	      ll = l;
6129 	      l = parse_split_condition (l, insn);
6130 
6131 	      if (l == NULL)
6132 		l = ll;
6133 	      else
6134 		continue;
6135 
6136 	      /* Accept an FPU prefix char which may be used when doing
6137 		 template MOV with FPU registers. */
6138 	      if (TOLOWER(*l) == FPU_PREFIX_CHAR)
6139 		{
6140 		  insn->type = INSN_DSP_FPU;
6141 		  l++;
6142 		  continue;
6143 		}
6144 
6145 	      if (TOLOWER(*l) == DSP_DUAL_CHAR)
6146 		{
6147 		  insn->dsp_width = DSP_WIDTH_DUAL;
6148 		  l++;
6149 		  continue;
6150 		}
6151 
6152 	      if (TOLOWER(*l) == DSP_ACTION_QR64_CHAR)
6153 		{
6154 		  insn->dsp_action_flags |= DSP_ACTION_QR64;
6155 		  l++;
6156 		  continue;
6157 		}
6158 
6159 	      if (TOLOWER(*l) == DSP_ACTION_UMUL_CHAR)
6160 		{
6161 		  insn->dsp_action_flags |= DSP_ACTION_UMUL;
6162 		  l++;
6163 		  continue;
6164 		}
6165 
6166 	      if (TOLOWER(*l) == DSP_ACTION_ROUND_CHAR)
6167 		{
6168 		  insn->dsp_action_flags |= DSP_ACTION_ROUND;
6169 		  l++;
6170 		  continue;
6171 		}
6172 
6173 	      if (TOLOWER(*l) == DSP_ACTION_CLAMP9_CHAR)
6174 		{
6175 		  insn->dsp_action_flags |= DSP_ACTION_CLAMP9;
6176 		  l++;
6177 		  continue;
6178 		}
6179 
6180 	      if (TOLOWER(*l) == DSP_ACTION_CLAMP8_CHAR)
6181 		{
6182 		  insn->dsp_action_flags |= DSP_ACTION_CLAMP8;
6183 		  l++;
6184 		  continue;
6185 		}
6186 
6187 	      if (TOLOWER(*l) == DSP_ACTION_MOD_CHAR)
6188 		{
6189 		  insn->dsp_action_flags |= DSP_ACTION_MOD;
6190 		  l++;
6191 		  continue;
6192 		}
6193 
6194 	      if (TOLOWER(*l) == DSP_ACTION_ACC_ZERO_CHAR)
6195 		{
6196 		  insn->dsp_action_flags |= DSP_ACTION_ACC_ZERO;
6197 		  l++;
6198 		  continue;
6199 		}
6200 
6201 	      if (TOLOWER(*l) == DSP_ACTION_ACC_ADD_CHAR)
6202 		{
6203 		  insn->dsp_action_flags |= DSP_ACTION_ACC_ADD;
6204 		  l++;
6205 		  continue;
6206 		}
6207 
6208 	      if (TOLOWER(*l) == DSP_ACTION_ACC_SUB_CHAR)
6209 		{
6210 		  insn->dsp_action_flags |= DSP_ACTION_ACC_SUB;
6211 		  l++;
6212 		  continue;
6213 		}
6214 
6215 	      if (TOLOWER(*l) == DSP_ACTION_OV_CHAR)
6216 		{
6217 		  insn->dsp_action_flags |= DSP_ACTION_OV;
6218 		  l++;
6219 		  continue;
6220 		}
6221 
6222 	      if (TOLOWER(*l) == DSP_DAOPPAME_8_CHAR)
6223 		{
6224 		  insn->dsp_daoppame_flags |= DSP_DAOPPAME_8;
6225 		  l++;
6226 		  continue;
6227 		}
6228 
6229 	      if (TOLOWER(*l) == DSP_DAOPPAME_16_CHAR)
6230 		{
6231 		  insn->dsp_daoppame_flags |= DSP_DAOPPAME_16;
6232 		  l++;
6233 		  continue;
6234 		}
6235 
6236 	      if (TOLOWER(*l) == DSP_DAOPPAME_TEMP_CHAR)
6237 		{
6238 		  insn->dsp_daoppame_flags |= DSP_DAOPPAME_TEMP;
6239 		  l++;
6240 		  continue;
6241 		}
6242 
6243 	      if (TOLOWER(*l) == DSP_DAOPPAME_HIGH_CHAR)
6244 		{
6245 		  insn->dsp_daoppame_flags |= DSP_DAOPPAME_HIGH;
6246 		  l++;
6247 		  continue;
6248 		}
6249 
6250 	      as_bad (_("unknown DSP prefix character %c %s"), *l, l);
6251 	      return NULL;
6252 	    }
6253 
6254 	  l = skip_space (l);
6255 	}
6256     }
6257 
6258   return l;
6259 }
6260 
6261 /* Return a list of appropriate instruction parsers for MNEMONIC.  */
6262 static insn_templates *
find_insn_templates(const char * mnemonic)6263 find_insn_templates (const char *mnemonic)
6264 {
6265   insn_template template;
6266   insn_templates entry;
6267   insn_templates *slot;
6268 
6269   entry.template = &template;
6270 
6271   memcpy ((void *)&entry.template->name, &mnemonic, sizeof (char *));
6272 
6273   slot = (insn_templates *) htab_find (mnemonic_htab, &entry);
6274 
6275   if (slot)
6276     return slot;
6277 
6278   return NULL;
6279 }
6280 
6281 /* Make an uppercase copy of SRC into DST and return DST.  */
6282 static char *
strupper(char * dst,const char * src)6283 strupper (char * dst, const char *src)
6284 {
6285   size_t i = 0;
6286 
6287   while (src[i])
6288     {
6289       dst[i] = TOUPPER (src[i]);
6290       i++;
6291     }
6292 
6293   dst[i] = 0;
6294 
6295   return dst;
6296 }
6297 
6298 /* Calculate a hash value for a template. */
6299 static hashval_t
hash_templates(const void * p)6300 hash_templates (const void *p)
6301 {
6302   insn_templates *tp = (insn_templates *)p;
6303   char buf[MAX_MNEMONIC_LEN];
6304 
6305   strupper (buf, tp->template->name);
6306 
6307   return htab_hash_string (buf);
6308 }
6309 
6310 /* Check if two templates are equal.  */
6311 static int
eq_templates(const void * a,const void * b)6312 eq_templates (const void *a, const void *b)
6313 {
6314   insn_templates *ta = (insn_templates *)a;
6315   insn_templates *tb = (insn_templates *)b;
6316   return strcasecmp (ta->template->name, tb->template->name) == 0;
6317 }
6318 
6319 /* Create the hash table required for parsing instructions.  */
6320 static void
create_mnemonic_htab(void)6321 create_mnemonic_htab (void)
6322 {
6323   size_t i, num_templates = sizeof(metag_optab)/sizeof(metag_optab[0]);
6324 
6325   mnemonic_htab = htab_create_alloc (num_templates, hash_templates,
6326 				     eq_templates, NULL, xcalloc, free);
6327 
6328   for (i = 0; i < num_templates; i++)
6329     {
6330       const insn_template *template = &metag_optab[i];
6331       insn_templates **slot = NULL;
6332       insn_templates *new_entry;
6333 
6334       new_entry = xmalloc (sizeof (insn_templates));
6335 
6336       new_entry->template = template;
6337       new_entry->next = NULL;
6338 
6339       slot = (insn_templates **) htab_find_slot (mnemonic_htab, new_entry,
6340 						 INSERT);
6341 
6342       if (*slot)
6343 	{
6344 	  insn_templates *last_entry = *slot;
6345 
6346 	  while (last_entry->next)
6347 	    last_entry = last_entry->next;
6348 
6349 	  last_entry->next = new_entry;
6350 	}
6351       else
6352 	{
6353 	  *slot = new_entry;
6354 	}
6355     }
6356 }
6357 
6358 /* Calculate a hash value for a register. */
6359 static hashval_t
hash_regs(const void * p)6360 hash_regs (const void *p)
6361 {
6362   metag_reg *rp = (metag_reg *)p;
6363   char buf[MAX_REG_LEN];
6364 
6365   strupper (buf, rp->name);
6366 
6367   return htab_hash_string (buf);
6368 }
6369 
6370 /* Check if two registers are equal.  */
6371 static int
eq_regs(const void * a,const void * b)6372 eq_regs (const void *a, const void *b)
6373 {
6374   metag_reg *ra = (metag_reg *)a;
6375   metag_reg *rb = (metag_reg *)b;
6376   return strcasecmp (ra->name, rb->name) == 0;
6377 }
6378 
6379 /* Create the hash table required for parsing registers.  */
6380 static void
create_reg_htab(void)6381 create_reg_htab (void)
6382 {
6383   size_t i, num_regs = sizeof(metag_regtab)/sizeof(metag_regtab[0]);
6384 
6385   reg_htab = htab_create_alloc (num_regs, hash_regs,
6386 				eq_regs, NULL, xcalloc, free);
6387 
6388   for (i = 0; i < num_regs; i++)
6389     {
6390       const metag_reg *reg = &metag_regtab[i];
6391       const metag_reg **slot;
6392 
6393       slot = (const metag_reg **) htab_find_slot (reg_htab, reg, INSERT);
6394 
6395       if (!*slot)
6396 	*slot = reg;
6397     }
6398 }
6399 
6400 /* Create the hash table required for parsing DSP registers.  */
6401 static void
create_dspreg_htabs(void)6402 create_dspreg_htabs (void)
6403 {
6404   size_t i, num_regs = sizeof(metag_dsp_regtab)/sizeof(metag_dsp_regtab[0]);
6405   size_t h;
6406 
6407   dsp_reg_htab = htab_create_alloc (num_regs, hash_regs,
6408 				    eq_regs, NULL, xcalloc, free);
6409 
6410   for (i = 0; i < num_regs; i++)
6411     {
6412       const metag_reg *reg = &metag_dsp_regtab[i];
6413       const metag_reg **slot;
6414 
6415       slot = (const metag_reg **) htab_find_slot (dsp_reg_htab, reg, INSERT);
6416 
6417       /* Make sure there are no hash table collisions, which would
6418 	 require chaining entries.  */
6419       BFD_ASSERT (*slot == NULL);
6420       *slot = reg;
6421     }
6422 
6423   num_regs = sizeof(metag_dsp_tmpl_regtab[0])/sizeof(metag_dsp_tmpl_regtab[0][0]);
6424 
6425   for (h = 0; h < 2; h++)
6426     {
6427       dsp_tmpl_reg_htab[h] = htab_create_alloc (num_regs, hash_regs,
6428 						eq_regs, NULL, xcalloc, free);
6429     }
6430 
6431   for (h = 0; h < 2; h++)
6432     {
6433       for (i = 0; i < num_regs; i++)
6434 	{
6435 	  const metag_reg *reg = &metag_dsp_tmpl_regtab[h][i];
6436 	  const metag_reg **slot;
6437 	  slot = (const metag_reg **) htab_find_slot (dsp_tmpl_reg_htab[h],
6438 						      reg, INSERT);
6439 
6440 	  /* Make sure there are no hash table collisions, which would
6441 	     require chaining entries.  */
6442 	  BFD_ASSERT (*slot == NULL);
6443 	  *slot = reg;
6444 	}
6445     }
6446 }
6447 
6448 /* Calculate a hash value for a split condition code. */
6449 static hashval_t
hash_scond(const void * p)6450 hash_scond (const void *p)
6451 {
6452   split_condition *cp = (split_condition *)p;
6453   char buf[4];
6454 
6455   strupper (buf, cp->name);
6456 
6457   return htab_hash_string (buf);
6458 }
6459 
6460 /* Check if two split condition codes are equal.  */
6461 static int
eq_scond(const void * a,const void * b)6462 eq_scond (const void *a, const void *b)
6463 {
6464   split_condition *ra = (split_condition *)a;
6465   split_condition *rb = (split_condition *)b;
6466 
6467   return strcasecmp (ra->name, rb->name) == 0;
6468 }
6469 
6470 /* Create the hash table required for parsing split condition codes.  */
6471 static void
create_scond_htab(void)6472 create_scond_htab (void)
6473 {
6474   size_t i, nentries;
6475 
6476   nentries = sizeof (metag_scondtab) / sizeof (metag_scondtab[0]);
6477 
6478   scond_htab = htab_create_alloc (nentries, hash_scond, eq_scond,
6479 				  NULL, xcalloc, free);
6480   for (i = 0; i < nentries; i++)
6481     {
6482       const split_condition *scond = &metag_scondtab[i];
6483       const split_condition **slot;
6484 
6485       slot = (const split_condition **) htab_find_slot (scond_htab,
6486 							scond, INSERT);
6487       /* Make sure there are no hash table collisions, which would
6488 	 require chaining entries.  */
6489       BFD_ASSERT (*slot == NULL);
6490       *slot = scond;
6491     }
6492 }
6493 
6494 /* Entry point for instruction parsing.  */
6495 static bfd_boolean
parse_insn(const char * line,metag_insn * insn)6496 parse_insn (const char *line, metag_insn *insn)
6497 {
6498   char mnemonic[MAX_MNEMONIC_LEN];
6499   const char *l = line;
6500   size_t mnemonic_len = 0;
6501   insn_templates *templates;
6502 
6503   l = skip_space (l);
6504 
6505   while (is_mnemonic_char(*l))
6506     {
6507       l++;
6508       mnemonic_len++;
6509     }
6510 
6511   if (mnemonic_len >= MAX_MNEMONIC_LEN)
6512     {
6513       as_bad (_("instruction mnemonic too long: %s"), line);
6514       return FALSE;
6515     }
6516 
6517   strncpy(mnemonic, line, mnemonic_len);
6518 
6519   mnemonic[mnemonic_len] = '\0';
6520 
6521   templates = find_insn_templates (mnemonic);
6522 
6523   if (templates)
6524     {
6525       insn_templates *current_template = templates;
6526 
6527       l = skip_space (l);
6528 
6529       while (current_template)
6530 	{
6531 	  const insn_template *template = current_template->template;
6532 	  enum insn_encoding encoding = template->encoding;
6533 	  insn_parser parser = insn_parsers[encoding];
6534 
6535 	  current_template = current_template->next;
6536 
6537 	  if (template->insn_type == INSN_GP &&
6538 	      !(template->core_flags & mcpu_opt))
6539 	    continue;
6540 
6541 	  if (template->insn_type == INSN_FPU &&
6542 	      !(template->core_flags & mfpu_opt))
6543 	    continue;
6544 
6545 	  if (template->insn_type == INSN_DSP &&
6546 	      !(template->core_flags & mdsp_opt))
6547 	    continue;
6548 
6549 	  if (template->insn_type == INSN_DSP_FPU &&
6550 	      !((template->core_flags & mdsp_opt) &&
6551 		(template->core_flags & mfpu_opt)))
6552 	    continue;
6553 
6554 	  /* DSP instructions always require special decoding */
6555 	  if ((insn->type == INSN_DSP && (template->insn_type != INSN_DSP)) ||
6556 	      ((template->insn_type == INSN_DSP) && insn->type != INSN_DSP) ||
6557 	      (insn->type == INSN_DSP_FPU && (template->insn_type != INSN_DSP_FPU)) ||
6558 	      ((template->insn_type == INSN_DSP_FPU) && insn->type != INSN_DSP_FPU))
6559 	    continue;
6560 
6561 	  if (parser)
6562 	    {
6563 	      const char *end = parser(l, insn, template);
6564 
6565 	      if (end != NULL)
6566 		{
6567 		  if (*end != END_OF_INSN)
6568 		    as_bad (_("junk at end of line: \"%s\""), line);
6569 		  else
6570 		    return TRUE;
6571 		}
6572 	    }
6573 	}
6574 
6575       as_bad (_("failed to assemble instruction: \"%s\""), line);
6576     }
6577   else
6578     {
6579       if (insn->type == INSN_FPU)
6580 	as_bad (_("unknown floating point mnemonic: \"%s\""), mnemonic);
6581       else
6582 	as_bad (_("unknown mnemonic: \"%s\""), mnemonic);
6583     }
6584   return FALSE;
6585 }
6586 
6587 static void
output_insn(metag_insn * insn)6588 output_insn (metag_insn *insn)
6589 {
6590   char *output;
6591 
6592   output = frag_more (insn->len);
6593   dwarf2_emit_insn (insn->len);
6594 
6595   if (insn->reloc_type != BFD_RELOC_UNUSED)
6596     {
6597       fix_new_exp (frag_now, output - frag_now->fr_literal,
6598 		   insn->reloc_size, &insn->reloc_exp,
6599 		   insn->reloc_pcrel, insn->reloc_type);
6600     }
6601 
6602   md_number_to_chars (output, insn->bits, insn->len);
6603 }
6604 
6605 void
md_assemble(char * line)6606 md_assemble (char *line)
6607 {
6608   const char *l = line;
6609   metag_insn insn;
6610 
6611   memset (&insn, 0, sizeof(insn));
6612 
6613   insn.reloc_type = BFD_RELOC_UNUSED;
6614   insn.reloc_pcrel = 0;
6615   insn.reloc_size = 4;
6616 
6617   if (!mnemonic_htab)
6618     {
6619       create_mnemonic_htab ();
6620       create_reg_htab ();
6621       create_dspreg_htabs ();
6622       create_scond_htab ();
6623     }
6624 
6625   l = parse_prefix (l, &insn);
6626 
6627   if (l == NULL)
6628     return;
6629 
6630   if (insn.type == INSN_DSP &&
6631       !mdsp_opt)
6632     {
6633       as_bad (_("cannot assemble DSP instruction, DSP option not set: %s"),
6634 	      line);
6635       return;
6636     }
6637   else if (insn.type == INSN_FPU &&
6638 	   !mfpu_opt)
6639     {
6640       as_bad (_("cannot assemble FPU instruction, FPU option not set: %s"),
6641 	      line);
6642       return;
6643     }
6644 
6645   if (!parse_insn (l, &insn))
6646     return;
6647 
6648   output_insn (&insn);
6649 }
6650 
6651 void
md_operand(expressionS * expressionP)6652 md_operand (expressionS * expressionP)
6653 {
6654   if (* input_line_pointer == IMM_CHAR)
6655     {
6656       input_line_pointer ++;
6657       expression (expressionP);
6658     }
6659 }
6660 
6661 valueT
md_section_align(segT segment ATTRIBUTE_UNUSED,valueT size)6662 md_section_align (segT segment ATTRIBUTE_UNUSED, valueT size)
6663 {
6664   return size;
6665 }
6666 
6667 symbolS *
md_undefined_symbol(char * name ATTRIBUTE_UNUSED)6668 md_undefined_symbol (char *name ATTRIBUTE_UNUSED)
6669 {
6670   return NULL;
6671 }
6672 
6673 /* Functions concerning relocs.  */
6674 
6675 /* The location from which a PC relative jump should be calculated,
6676    given a PC relative reloc.  */
6677 
6678 long
md_pcrel_from_section(fixS * fixP,segT sec)6679 md_pcrel_from_section (fixS * fixP, segT sec)
6680 {
6681   if ((fixP->fx_addsy != (symbolS *) NULL
6682        && (! S_IS_DEFINED (fixP->fx_addsy)
6683 	   || S_GET_SEGMENT (fixP->fx_addsy) != sec))
6684       || metag_force_relocation (fixP))
6685     {
6686       /* The symbol is undefined (or is defined but not in this section).
6687 	 Let the linker figure it out.  */
6688       return 0;
6689     }
6690 
6691   return fixP->fx_frag->fr_address + fixP->fx_where;
6692 }
6693 
6694 /* Write a value out to the object file, using the appropriate endianness.  */
6695 
6696 void
md_number_to_chars(char * buf,valueT val,int n)6697 md_number_to_chars (char * buf, valueT val, int n)
6698 {
6699   number_to_chars_littleendian (buf, val, n);
6700 }
6701 
6702 /* Turn a string in input_line_pointer into a floating point constant of type
6703    type, and store the appropriate bytes in *litP.  The number of LITTLENUMS
6704    emitted is stored in *sizeP .  An error message is returned, or NULL on OK.
6705 */
6706 
6707 /* Equal to MAX_PRECISION in atof-ieee.c */
6708 #define MAX_LITTLENUMS 6
6709 
6710 char *
md_atof(int type,char * litP,int * sizeP)6711 md_atof (int type, char * litP, int * sizeP)
6712 {
6713   int              i;
6714   int              prec;
6715   LITTLENUM_TYPE   words [MAX_LITTLENUMS];
6716   char *           t;
6717 
6718   switch (type)
6719     {
6720     case 'f':
6721     case 'F':
6722     case 's':
6723     case 'S':
6724       prec = 2;
6725       break;
6726 
6727     case 'd':
6728     case 'D':
6729     case 'r':
6730     case 'R':
6731       prec = 4;
6732       break;
6733 
6734    /* FIXME: Some targets allow other format chars for bigger sizes here.  */
6735 
6736     default:
6737       * sizeP = 0;
6738       return _("Bad call to md_atof()");
6739     }
6740 
6741   t = atof_ieee (input_line_pointer, type, words);
6742   if (t)
6743     input_line_pointer = t;
6744   * sizeP = prec * sizeof (LITTLENUM_TYPE);
6745 
6746   for (i = 0; i < prec; i++)
6747     {
6748       md_number_to_chars (litP, (valueT) words[i],
6749 			  sizeof (LITTLENUM_TYPE));
6750       litP += sizeof (LITTLENUM_TYPE);
6751     }
6752 
6753   return 0;
6754 }
6755 
6756 /* If this function returns non-zero, it prevents the relocation
6757    against symbol(s) in the FIXP from being replaced with relocations
6758    against section symbols, and guarantees that a relocation will be
6759    emitted even when the value can be resolved locally.  */
6760 
6761 int
metag_force_relocation(fixS * fix)6762 metag_force_relocation (fixS * fix)
6763 {
6764   switch (fix->fx_r_type)
6765     {
6766     case BFD_RELOC_METAG_RELBRANCH_PLT:
6767     case BFD_RELOC_METAG_TLS_LE:
6768     case BFD_RELOC_METAG_TLS_IE:
6769     case BFD_RELOC_METAG_TLS_LDO:
6770     case BFD_RELOC_METAG_TLS_LDM:
6771     case BFD_RELOC_METAG_TLS_GD:
6772       return 1;
6773     default:
6774       ;
6775     }
6776 
6777   return generic_force_reloc (fix);
6778 }
6779 
6780 bfd_boolean
metag_fix_adjustable(fixS * fixP)6781 metag_fix_adjustable (fixS * fixP)
6782 {
6783   if (fixP->fx_addsy == NULL)
6784     return 1;
6785 
6786   /* Prevent all adjustments to global symbols.  */
6787   if (S_IS_EXTERNAL (fixP->fx_addsy))
6788     return 0;
6789   if (S_IS_WEAK (fixP->fx_addsy))
6790     return 0;
6791 
6792   if (fixP->fx_r_type == BFD_RELOC_METAG_HI16_GOTOFF ||
6793       fixP->fx_r_type == BFD_RELOC_METAG_LO16_GOTOFF ||
6794       fixP->fx_r_type == BFD_RELOC_METAG_GETSET_GOTOFF ||
6795       fixP->fx_r_type == BFD_RELOC_METAG_GETSET_GOT ||
6796       fixP->fx_r_type == BFD_RELOC_METAG_HI16_GOTPC ||
6797       fixP->fx_r_type == BFD_RELOC_METAG_LO16_GOTPC ||
6798       fixP->fx_r_type == BFD_RELOC_METAG_HI16_PLT ||
6799       fixP->fx_r_type == BFD_RELOC_METAG_LO16_PLT ||
6800       fixP->fx_r_type == BFD_RELOC_METAG_RELBRANCH_PLT)
6801     return 0;
6802 
6803   /* We need the symbol name for the VTABLE entries.  */
6804   if (fixP->fx_r_type == BFD_RELOC_VTABLE_INHERIT
6805       || fixP->fx_r_type == BFD_RELOC_VTABLE_ENTRY)
6806     return 0;
6807 
6808   return 1;
6809 }
6810 
6811 /* Return an initial guess of the length by which a fragment must grow to
6812    hold a branch to reach its destination.
6813    Also updates fr_type/fr_subtype as necessary.
6814 
6815    Called just before doing relaxation.
6816    Any symbol that is now undefined will not become defined.
6817    The guess for fr_var is ACTUALLY the growth beyond fr_fix.
6818    Whatever we do to grow fr_fix or fr_var contributes to our returned value.
6819    Although it may not be explicit in the frag, pretend fr_var starts with a
6820    0 value.  */
6821 
6822 int
md_estimate_size_before_relax(fragS * fragP ATTRIBUTE_UNUSED,segT segment ATTRIBUTE_UNUSED)6823 md_estimate_size_before_relax (fragS * fragP ATTRIBUTE_UNUSED,
6824 			       segT    segment ATTRIBUTE_UNUSED)
6825 {
6826   /* No assembler relaxation is defined (or necessary) for this port.  */
6827   abort ();
6828 }
6829 
6830 /* *fragP has been relaxed to its final size, and now needs to have
6831    the bytes inside it modified to conform to the new size.
6832 
6833    Called after relaxation is finished.
6834    fragP->fr_type == rs_machine_dependent.
6835    fragP->fr_subtype is the subtype of what the address relaxed to.  */
6836 
6837 void
md_convert_frag(bfd * abfd ATTRIBUTE_UNUSED,segT sec ATTRIBUTE_UNUSED,fragS * fragP ATTRIBUTE_UNUSED)6838 md_convert_frag (bfd * abfd ATTRIBUTE_UNUSED, segT sec ATTRIBUTE_UNUSED,
6839 		 fragS * fragP ATTRIBUTE_UNUSED)
6840 {
6841   /* No assembler relaxation is defined (or necessary) for this port.  */
6842   abort ();
6843 }
6844 
6845 /* This is called from HANDLE_ALIGN in tc-metag.h.  */
6846 
6847 void
metag_handle_align(fragS * fragP)6848 metag_handle_align (fragS * fragP)
6849 {
6850   static char const noop[4] = { 0xfe, 0xff, 0xff, 0xa0 };
6851   int bytes, fix;
6852   char *p;
6853 
6854   if (fragP->fr_type != rs_align_code)
6855     return;
6856 
6857   bytes = fragP->fr_next->fr_address - fragP->fr_address - fragP->fr_fix;
6858   p = fragP->fr_literal + fragP->fr_fix;
6859   fix = 0;
6860 
6861   if (bytes & 3)
6862     {
6863       fix = bytes & 3;
6864       memset (p, 0, fix);
6865       p += fix;
6866       bytes -= fix;
6867     }
6868 
6869   while (bytes >= 4)
6870     {
6871       memcpy (p, noop, 4);
6872       p += 4;
6873       bytes -= 4;
6874       fix += 4;
6875     }
6876 
6877   fragP->fr_fix += fix;
6878   fragP->fr_var = 4;
6879 }
6880 
6881 static char *
metag_end_of_match(char * cont,char * what)6882 metag_end_of_match (char * cont, char * what)
6883 {
6884   int len = strlen (what);
6885 
6886   if (strncasecmp (cont, what, strlen (what)) == 0
6887       && ! is_part_of_name (cont[len]))
6888     return cont + len;
6889 
6890   return NULL;
6891 }
6892 
6893 int
metag_parse_name(char const * name,expressionS * exprP,enum expr_mode mode,char * nextcharP)6894 metag_parse_name (char const * name, expressionS * exprP, enum expr_mode mode,
6895 		  char * nextcharP)
6896 {
6897   char *next = input_line_pointer;
6898   char *next_end;
6899   int reloc_type;
6900   operatorT op_type;
6901   segT segment;
6902 
6903   exprP->X_op_symbol = NULL;
6904   exprP->X_md = BFD_RELOC_UNUSED;
6905 
6906   if (strcmp (name, GOT_NAME) == 0)
6907     {
6908       if (! GOT_symbol)
6909 	GOT_symbol = symbol_find_or_make (name);
6910 
6911       exprP->X_add_symbol = GOT_symbol;
6912     no_suffix:
6913       /* If we have an absolute symbol or a
6914 	 reg, then we know its value now.  */
6915       segment = S_GET_SEGMENT (exprP->X_add_symbol);
6916       if (mode != expr_defer && segment == absolute_section)
6917 	{
6918 	  exprP->X_op = O_constant;
6919 	  exprP->X_add_number = S_GET_VALUE (exprP->X_add_symbol);
6920 	  exprP->X_add_symbol = NULL;
6921 	}
6922       else if (mode != expr_defer && segment == reg_section)
6923 	{
6924 	  exprP->X_op = O_register;
6925 	  exprP->X_add_number = S_GET_VALUE (exprP->X_add_symbol);
6926 	  exprP->X_add_symbol = NULL;
6927 	}
6928       else
6929 	{
6930 	  exprP->X_op = O_symbol;
6931 	  exprP->X_add_number = 0;
6932 	}
6933 
6934       return 1;
6935     }
6936 
6937   exprP->X_add_symbol = symbol_find_or_make (name);
6938 
6939   if (*nextcharP != '@')
6940     goto no_suffix;
6941   else if ((next_end = metag_end_of_match (next + 1, "GOTOFF")))
6942     {
6943       reloc_type = BFD_RELOC_METAG_GOTOFF;
6944       op_type = O_PIC_reloc;
6945     }
6946   else if ((next_end = metag_end_of_match (next + 1, "GOT")))
6947     {
6948       reloc_type = BFD_RELOC_METAG_GETSET_GOT;
6949       op_type = O_PIC_reloc;
6950     }
6951   else if ((next_end = metag_end_of_match (next + 1, "PLT")))
6952     {
6953       reloc_type = BFD_RELOC_METAG_PLT;
6954       op_type = O_PIC_reloc;
6955     }
6956   else if ((next_end = metag_end_of_match (next + 1, "TLSGD")))
6957     {
6958       reloc_type = BFD_RELOC_METAG_TLS_GD;
6959       op_type = O_PIC_reloc;
6960     }
6961   else if ((next_end = metag_end_of_match (next + 1, "TLSLDM")))
6962     {
6963       reloc_type = BFD_RELOC_METAG_TLS_LDM;
6964       op_type = O_PIC_reloc;
6965     }
6966   else if ((next_end = metag_end_of_match (next + 1, "TLSLDO")))
6967     {
6968       reloc_type = BFD_RELOC_METAG_TLS_LDO;
6969       op_type = O_PIC_reloc;
6970     }
6971   else if ((next_end = metag_end_of_match (next + 1, "TLSIE")))
6972     {
6973       reloc_type = BFD_RELOC_METAG_TLS_IE;
6974       op_type = O_PIC_reloc;
6975     }
6976   else if ((next_end = metag_end_of_match (next + 1, "TLSIENONPIC")))
6977     {
6978       reloc_type = BFD_RELOC_METAG_TLS_IENONPIC;
6979       op_type = O_PIC_reloc;	/* FIXME: is this correct? */
6980     }
6981   else if ((next_end = metag_end_of_match (next + 1, "TLSLE")))
6982     {
6983       reloc_type = BFD_RELOC_METAG_TLS_LE;
6984       op_type = O_PIC_reloc;
6985     }
6986   else
6987     goto no_suffix;
6988 
6989   *input_line_pointer = *nextcharP;
6990   input_line_pointer = next_end;
6991   *nextcharP = *input_line_pointer;
6992   *input_line_pointer = '\0';
6993 
6994   exprP->X_op = op_type;
6995   exprP->X_add_number = 0;
6996   exprP->X_md = reloc_type;
6997 
6998   return 1;
6999 }
7000 
7001 /* If while processing a fixup, a reloc really needs to be created
7002    then it is done here.  */
7003 
7004 arelent *
tc_gen_reloc(seg,fixp)7005 tc_gen_reloc (seg, fixp)
7006      asection *seg ATTRIBUTE_UNUSED;
7007      fixS *fixp;
7008 {
7009   arelent *reloc;
7010 
7011   reloc		      = (arelent *) xmalloc (sizeof (arelent));
7012   reloc->sym_ptr_ptr  = (asymbol **) xmalloc (sizeof (asymbol *));
7013   *reloc->sym_ptr_ptr = symbol_get_bfdsym (fixp->fx_addsy);
7014   reloc->address      = fixp->fx_frag->fr_address + fixp->fx_where;
7015 
7016   reloc->addend = fixp->fx_offset;
7017   reloc->howto = bfd_reloc_type_lookup (stdoutput, fixp->fx_r_type);
7018 
7019   if (reloc->howto == (reloc_howto_type *) NULL)
7020     {
7021       as_bad_where (fixp->fx_file, fixp->fx_line,
7022 		    /* xgettext:c-format.  */
7023 		    _("reloc %d not supported by object file format"),
7024 		    (int) fixp->fx_r_type);
7025 
7026       xfree (reloc);
7027 
7028       return NULL;
7029     }
7030 
7031   return reloc;
7032 }
7033 
7034 static unsigned int
md_chars_to_number(char * val,int n)7035 md_chars_to_number (char *val, int n)
7036 {
7037   int retval;
7038   unsigned char * where = (unsigned char *) val;
7039 
7040   for (retval = 0; n--;)
7041     {
7042       retval <<= 8;
7043       retval |= where[n];
7044     }
7045   return retval;
7046 }
7047 
7048 void
md_apply_fix(fixS * fixP,valueT * valP,segT seg ATTRIBUTE_UNUSED)7049 md_apply_fix (fixS *fixP, valueT *valP, segT seg ATTRIBUTE_UNUSED)
7050 {
7051   char *buf = fixP->fx_where + fixP->fx_frag->fr_literal;
7052   int value = (int)*valP;
7053 
7054   switch (fixP->fx_r_type)
7055     {
7056     case BFD_RELOC_METAG_TLS_GD:
7057     case BFD_RELOC_METAG_TLS_LE_HI16:
7058     case BFD_RELOC_METAG_TLS_LE_LO16:
7059     case BFD_RELOC_METAG_TLS_IE:
7060     case BFD_RELOC_METAG_TLS_IENONPIC_HI16:
7061     case BFD_RELOC_METAG_TLS_IENONPIC_LO16:
7062     case BFD_RELOC_METAG_TLS_LDM:
7063     case BFD_RELOC_METAG_TLS_LDO_HI16:
7064     case BFD_RELOC_METAG_TLS_LDO_LO16:
7065       S_SET_THREAD_LOCAL (fixP->fx_addsy);
7066       /* Fall through */
7067 
7068     case BFD_RELOC_METAG_HIADDR16:
7069     case BFD_RELOC_METAG_LOADDR16:
7070     case BFD_RELOC_VTABLE_INHERIT:
7071     case BFD_RELOC_VTABLE_ENTRY:
7072       fixP->fx_done = FALSE;
7073       break;
7074 
7075     case BFD_RELOC_METAG_REL8:
7076       if (!within_unsigned_range (value, IMM8_BITS))
7077 	{
7078 	  as_bad_where (fixP->fx_file, fixP->fx_line,
7079 			"rel8 out of range %d", value);
7080 	}
7081       else
7082 	{
7083 	  unsigned int newval;
7084 	  newval = md_chars_to_number (buf, 4);
7085 	  newval = (newval & 0xffffc03f) | ((value & IMM8_MASK) << 6);
7086 	  md_number_to_chars (buf, newval, 4);
7087 	}
7088       break;
7089     case BFD_RELOC_METAG_REL16:
7090       if (!within_unsigned_range (value, IMM16_BITS))
7091 	{
7092 	  as_bad_where (fixP->fx_file, fixP->fx_line,
7093 			"rel16 out of range %d", value);
7094 	}
7095       else
7096 	{
7097 	  unsigned int newval;
7098 	  newval = md_chars_to_number (buf, 4);
7099 	  newval = (newval & 0xfff80007) | ((value & IMM16_MASK) << 3);
7100 	  md_number_to_chars (buf, newval, 4);
7101 	}
7102       break;
7103 
7104     case BFD_RELOC_8:
7105       md_number_to_chars (buf, value, 1);
7106       break;
7107     case BFD_RELOC_16:
7108       md_number_to_chars (buf, value, 2);
7109       break;
7110     case BFD_RELOC_32:
7111       md_number_to_chars (buf, value, 4);
7112       break;
7113     case BFD_RELOC_64:
7114       md_number_to_chars (buf, value, 8);
7115 
7116     case BFD_RELOC_METAG_RELBRANCH:
7117       if (!value)
7118 	break;
7119 
7120       value = value / 4;
7121 
7122       if (!within_signed_range (value, IMM19_BITS))
7123 	{
7124 	  as_bad_where (fixP->fx_file, fixP->fx_line,
7125 			"relbranch out of range %d", value);
7126 	}
7127       else
7128 	{
7129 	  unsigned int newval;
7130 	  newval = md_chars_to_number (buf, 4);
7131 	  newval = (newval & 0xff00001f) | ((value & IMM19_MASK) << 5);
7132 	  md_number_to_chars (buf, newval, 4);
7133 	}
7134 	break;
7135     default:
7136       break;
7137     }
7138 
7139   if (fixP->fx_addsy == NULL)
7140     fixP->fx_done = TRUE;
7141 }
7142