1 /* Disassembler interface for targets using CGEN. -*- C -*-
2    CGEN: Cpu tools GENerator
3 
4    THIS FILE IS MACHINE GENERATED WITH CGEN.
5    - the resultant file is machine generated, cgen-dis.in isn't
6 
7    Copyright (C) 1996-2014 Free Software Foundation, Inc.
8 
9    This file is part of libopcodes.
10 
11    This library is free software; you can redistribute it and/or modify
12    it under the terms of the GNU General Public License as published by
13    the Free Software Foundation; either version 3, or (at your option)
14    any later version.
15 
16    It is distributed in the hope that it will be useful, but WITHOUT
17    ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
18    or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
19    License for more details.
20 
21    You should have received a copy of the GNU General Public License
22    along with this program; if not, write to the Free Software Foundation, Inc.,
23    51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.  */
24 
25 /* ??? Eventually more and more of this stuff can go to cpu-independent files.
26    Keep that in mind.  */
27 
28 #include "sysdep.h"
29 #include <stdio.h>
30 #include "ansidecl.h"
31 #include "dis-asm.h"
32 #include "bfd.h"
33 #include "symcat.h"
34 #include "libiberty.h"
35 #include "epiphany-desc.h"
36 #include "epiphany-opc.h"
37 #include "opintl.h"
38 
39 /* Default text to print if an instruction isn't recognized.  */
40 #define UNKNOWN_INSN_MSG _("*unknown*")
41 
42 static void print_normal
43   (CGEN_CPU_DESC, void *, long, unsigned int, bfd_vma, int);
44 static void print_address
45   (CGEN_CPU_DESC, void *, bfd_vma, unsigned int, bfd_vma, int) ATTRIBUTE_UNUSED;
46 static void print_keyword
47   (CGEN_CPU_DESC, void *, CGEN_KEYWORD *, long, unsigned int) ATTRIBUTE_UNUSED;
48 static void print_insn_normal
49   (CGEN_CPU_DESC, void *, const CGEN_INSN *, CGEN_FIELDS *, bfd_vma, int);
50 static int print_insn
51   (CGEN_CPU_DESC, bfd_vma,  disassemble_info *, bfd_byte *, unsigned);
52 static int default_print_insn
53   (CGEN_CPU_DESC, bfd_vma, disassemble_info *) ATTRIBUTE_UNUSED;
54 static int read_insn
55   (CGEN_CPU_DESC, bfd_vma, disassemble_info *, bfd_byte *, int, CGEN_EXTRACT_INFO *,
56    unsigned long *);
57 
58 /* -- disassembler routines inserted here.  */
59 
60 /* -- dis.c */
61 
62 #define CGEN_PRINT_INSN epiphany_print_insn
63 
64 static int
epiphany_print_insn(CGEN_CPU_DESC cd,bfd_vma pc,disassemble_info * info)65 epiphany_print_insn (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
66 {
67   bfd_byte buf[CGEN_MAX_INSN_SIZE];
68   int buflen;
69   int status;
70 
71   info->bytes_per_chunk = 2;
72 
73   /* Attempt to read the base part of the insn.  */
74   info->bytes_per_line = buflen = cd->base_insn_bitsize / 8;
75   status = (*info->read_memory_func) (pc, buf, buflen, info);
76 
77   /* Try again with the minimum part, if min < base.  */
78   if (status != 0 && (cd->min_insn_bitsize < cd->base_insn_bitsize))
79     {
80       info->bytes_per_line = buflen = cd->min_insn_bitsize / 8;
81       status = (*info->read_memory_func) (pc, buf, buflen, info);
82     }
83 
84   if (status != 0)
85     {
86       (*info->memory_error_func) (status, pc, info);
87       return -1;
88     }
89 
90   return print_insn (cd, pc, info, buf, buflen);
91 }
92 
93 
94 static void
print_postindex(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)95 print_postindex (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
96 		 void * dis_info,
97 		 long value,
98 		 unsigned int attrs ATTRIBUTE_UNUSED,
99 		 bfd_vma pc ATTRIBUTE_UNUSED,
100 		 int length ATTRIBUTE_UNUSED)
101 {
102   disassemble_info *info = (disassemble_info *) dis_info;
103   (*info->fprintf_func) (info->stream, value ? "-" : "+");
104 }
105 
106 static void
print_simm_not_reg(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)107 print_simm_not_reg (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
108 		    void * dis_info,
109 		    long value,
110 		    unsigned int attrs ATTRIBUTE_UNUSED,
111 		    bfd_vma pc ATTRIBUTE_UNUSED,
112 		    int length ATTRIBUTE_UNUSED)
113 {
114   print_address (cd, dis_info, value, attrs, pc, length);
115 }
116 
117 static void
print_uimm_not_reg(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,unsigned long value,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)118 print_uimm_not_reg (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
119 		    void * dis_info,
120 		    unsigned long value,
121 		    unsigned int attrs ATTRIBUTE_UNUSED,
122 		    bfd_vma pc ATTRIBUTE_UNUSED,
123 		    int length ATTRIBUTE_UNUSED)
124 {
125   disassemble_info *info = (disassemble_info *)dis_info;
126 
127   if (value & 0x800)
128     (*info->fprintf_func) (info->stream, "-");
129 
130   value &= 0x7ff;
131   print_address (cd, dis_info, value, attrs, pc, length);
132 }
133 
134 
135 /* -- */
136 
137 void epiphany_cgen_print_operand
138   (CGEN_CPU_DESC, int, PTR, CGEN_FIELDS *, void const *, bfd_vma, int);
139 
140 /* Main entry point for printing operands.
141    XINFO is a `void *' and not a `disassemble_info *' to not put a requirement
142    of dis-asm.h on cgen.h.
143 
144    This function is basically just a big switch statement.  Earlier versions
145    used tables to look up the function to use, but
146    - if the table contains both assembler and disassembler functions then
147      the disassembler contains much of the assembler and vice-versa,
148    - there's a lot of inlining possibilities as things grow,
149    - using a switch statement avoids the function call overhead.
150 
151    This function could be moved into `print_insn_normal', but keeping it
152    separate makes clear the interface between `print_insn_normal' and each of
153    the handlers.  */
154 
155 void
epiphany_cgen_print_operand(CGEN_CPU_DESC cd,int opindex,void * xinfo,CGEN_FIELDS * fields,void const * attrs ATTRIBUTE_UNUSED,bfd_vma pc,int length)156 epiphany_cgen_print_operand (CGEN_CPU_DESC cd,
157 			   int opindex,
158 			   void * xinfo,
159 			   CGEN_FIELDS *fields,
160 			   void const *attrs ATTRIBUTE_UNUSED,
161 			   bfd_vma pc,
162 			   int length)
163 {
164   disassemble_info *info = (disassemble_info *) xinfo;
165 
166   switch (opindex)
167     {
168     case EPIPHANY_OPERAND_DIRECTION :
169       print_postindex (cd, info, fields->f_addsubx, 0, pc, length);
170       break;
171     case EPIPHANY_OPERAND_DISP11 :
172       print_uimm_not_reg (cd, info, fields->f_disp11, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
173       break;
174     case EPIPHANY_OPERAND_DISP3 :
175       print_normal (cd, info, fields->f_disp3, 0, pc, length);
176       break;
177     case EPIPHANY_OPERAND_DPMI :
178       print_postindex (cd, info, fields->f_subd, 0, pc, length);
179       break;
180     case EPIPHANY_OPERAND_FRD :
181       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rd, 0);
182       break;
183     case EPIPHANY_OPERAND_FRD6 :
184       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rd6, 0|(1<<CGEN_OPERAND_VIRTUAL));
185       break;
186     case EPIPHANY_OPERAND_FRM :
187       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rm, 0);
188       break;
189     case EPIPHANY_OPERAND_FRM6 :
190       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rm6, 0|(1<<CGEN_OPERAND_VIRTUAL));
191       break;
192     case EPIPHANY_OPERAND_FRN :
193       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rn, 0);
194       break;
195     case EPIPHANY_OPERAND_FRN6 :
196       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rn6, 0|(1<<CGEN_OPERAND_VIRTUAL));
197       break;
198     case EPIPHANY_OPERAND_IMM16 :
199       print_address (cd, info, fields->f_imm16, 0|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
200       break;
201     case EPIPHANY_OPERAND_IMM8 :
202       print_address (cd, info, fields->f_imm8, 0|(1<<CGEN_OPERAND_RELAX), pc, length);
203       break;
204     case EPIPHANY_OPERAND_RD :
205       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rd, 0);
206       break;
207     case EPIPHANY_OPERAND_RD6 :
208       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rd6, 0|(1<<CGEN_OPERAND_VIRTUAL));
209       break;
210     case EPIPHANY_OPERAND_RM :
211       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rm, 0);
212       break;
213     case EPIPHANY_OPERAND_RM6 :
214       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rm6, 0|(1<<CGEN_OPERAND_VIRTUAL));
215       break;
216     case EPIPHANY_OPERAND_RN :
217       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rn, 0);
218       break;
219     case EPIPHANY_OPERAND_RN6 :
220       print_keyword (cd, info, & epiphany_cgen_opval_gr_names, fields->f_rn6, 0|(1<<CGEN_OPERAND_VIRTUAL));
221       break;
222     case EPIPHANY_OPERAND_SD :
223       print_keyword (cd, info, & epiphany_cgen_opval_cr_names, fields->f_sd, 0);
224       break;
225     case EPIPHANY_OPERAND_SD6 :
226       print_keyword (cd, info, & epiphany_cgen_opval_cr_names, fields->f_sd6, 0|(1<<CGEN_OPERAND_VIRTUAL));
227       break;
228     case EPIPHANY_OPERAND_SDDMA :
229       print_keyword (cd, info, & epiphany_cgen_opval_crdma_names, fields->f_sd6, 0|(1<<CGEN_OPERAND_VIRTUAL));
230       break;
231     case EPIPHANY_OPERAND_SDMEM :
232       print_keyword (cd, info, & epiphany_cgen_opval_crmem_names, fields->f_sd6, 0|(1<<CGEN_OPERAND_VIRTUAL));
233       break;
234     case EPIPHANY_OPERAND_SDMESH :
235       print_keyword (cd, info, & epiphany_cgen_opval_crmesh_names, fields->f_sd6, 0|(1<<CGEN_OPERAND_VIRTUAL));
236       break;
237     case EPIPHANY_OPERAND_SHIFT :
238       print_normal (cd, info, fields->f_shift, 0, pc, length);
239       break;
240     case EPIPHANY_OPERAND_SIMM11 :
241       print_simm_not_reg (cd, info, fields->f_sdisp11, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
242       break;
243     case EPIPHANY_OPERAND_SIMM24 :
244       print_address (cd, info, fields->f_simm24, 0|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
245       break;
246     case EPIPHANY_OPERAND_SIMM3 :
247       print_simm_not_reg (cd, info, fields->f_sdisp3, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELAX), pc, length);
248       break;
249     case EPIPHANY_OPERAND_SIMM8 :
250       print_address (cd, info, fields->f_simm8, 0|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
251       break;
252     case EPIPHANY_OPERAND_SN :
253       print_keyword (cd, info, & epiphany_cgen_opval_cr_names, fields->f_sn, 0);
254       break;
255     case EPIPHANY_OPERAND_SN6 :
256       print_keyword (cd, info, & epiphany_cgen_opval_cr_names, fields->f_sn6, 0|(1<<CGEN_OPERAND_VIRTUAL));
257       break;
258     case EPIPHANY_OPERAND_SNDMA :
259       print_keyword (cd, info, & epiphany_cgen_opval_crdma_names, fields->f_sn6, 0|(1<<CGEN_OPERAND_VIRTUAL));
260       break;
261     case EPIPHANY_OPERAND_SNMEM :
262       print_keyword (cd, info, & epiphany_cgen_opval_crmem_names, fields->f_sn6, 0|(1<<CGEN_OPERAND_VIRTUAL));
263       break;
264     case EPIPHANY_OPERAND_SNMESH :
265       print_keyword (cd, info, & epiphany_cgen_opval_crmesh_names, fields->f_sn6, 0|(1<<CGEN_OPERAND_VIRTUAL));
266       break;
267     case EPIPHANY_OPERAND_SWI_NUM :
268       print_uimm_not_reg (cd, info, fields->f_trap_num, 0, pc, length);
269       break;
270     case EPIPHANY_OPERAND_TRAPNUM6 :
271       print_normal (cd, info, fields->f_trap_num, 0, pc, length);
272       break;
273 
274     default :
275       /* xgettext:c-format */
276       fprintf (stderr, _("Unrecognized field %d while printing insn.\n"),
277 	       opindex);
278     abort ();
279   }
280 }
281 
282 cgen_print_fn * const epiphany_cgen_print_handlers[] =
283 {
284   print_insn_normal,
285 };
286 
287 
288 void
epiphany_cgen_init_dis(CGEN_CPU_DESC cd)289 epiphany_cgen_init_dis (CGEN_CPU_DESC cd)
290 {
291   epiphany_cgen_init_opcode_table (cd);
292   epiphany_cgen_init_ibld_table (cd);
293   cd->print_handlers = & epiphany_cgen_print_handlers[0];
294   cd->print_operand = epiphany_cgen_print_operand;
295 }
296 
297 
298 /* Default print handler.  */
299 
300 static void
print_normal(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value,unsigned int attrs,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)301 print_normal (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
302 	      void *dis_info,
303 	      long value,
304 	      unsigned int attrs,
305 	      bfd_vma pc ATTRIBUTE_UNUSED,
306 	      int length ATTRIBUTE_UNUSED)
307 {
308   disassemble_info *info = (disassemble_info *) dis_info;
309 
310   /* Print the operand as directed by the attributes.  */
311   if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SEM_ONLY))
312     ; /* nothing to do */
313   else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SIGNED))
314     (*info->fprintf_func) (info->stream, "%ld", value);
315   else
316     (*info->fprintf_func) (info->stream, "0x%lx", value);
317 }
318 
319 /* Default address handler.  */
320 
321 static void
print_address(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,bfd_vma value,unsigned int attrs,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)322 print_address (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
323 	       void *dis_info,
324 	       bfd_vma value,
325 	       unsigned int attrs,
326 	       bfd_vma pc ATTRIBUTE_UNUSED,
327 	       int length ATTRIBUTE_UNUSED)
328 {
329   disassemble_info *info = (disassemble_info *) dis_info;
330 
331   /* Print the operand as directed by the attributes.  */
332   if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SEM_ONLY))
333     ; /* Nothing to do.  */
334   else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_PCREL_ADDR))
335     (*info->print_address_func) (value, info);
336   else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_ABS_ADDR))
337     (*info->print_address_func) (value, info);
338   else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SIGNED))
339     (*info->fprintf_func) (info->stream, "%ld", (long) value);
340   else
341     (*info->fprintf_func) (info->stream, "0x%lx", (long) value);
342 }
343 
344 /* Keyword print handler.  */
345 
346 static void
print_keyword(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,CGEN_KEYWORD * keyword_table,long value,unsigned int attrs ATTRIBUTE_UNUSED)347 print_keyword (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
348 	       void *dis_info,
349 	       CGEN_KEYWORD *keyword_table,
350 	       long value,
351 	       unsigned int attrs ATTRIBUTE_UNUSED)
352 {
353   disassemble_info *info = (disassemble_info *) dis_info;
354   const CGEN_KEYWORD_ENTRY *ke;
355 
356   ke = cgen_keyword_lookup_value (keyword_table, value);
357   if (ke != NULL)
358     (*info->fprintf_func) (info->stream, "%s", ke->name);
359   else
360     (*info->fprintf_func) (info->stream, "???");
361 }
362 
363 /* Default insn printer.
364 
365    DIS_INFO is defined as `void *' so the disassembler needn't know anything
366    about disassemble_info.  */
367 
368 static void
print_insn_normal(CGEN_CPU_DESC cd,void * dis_info,const CGEN_INSN * insn,CGEN_FIELDS * fields,bfd_vma pc,int length)369 print_insn_normal (CGEN_CPU_DESC cd,
370 		   void *dis_info,
371 		   const CGEN_INSN *insn,
372 		   CGEN_FIELDS *fields,
373 		   bfd_vma pc,
374 		   int length)
375 {
376   const CGEN_SYNTAX *syntax = CGEN_INSN_SYNTAX (insn);
377   disassemble_info *info = (disassemble_info *) dis_info;
378   const CGEN_SYNTAX_CHAR_TYPE *syn;
379 
380   CGEN_INIT_PRINT (cd);
381 
382   for (syn = CGEN_SYNTAX_STRING (syntax); *syn; ++syn)
383     {
384       if (CGEN_SYNTAX_MNEMONIC_P (*syn))
385 	{
386 	  (*info->fprintf_func) (info->stream, "%s", CGEN_INSN_MNEMONIC (insn));
387 	  continue;
388 	}
389       if (CGEN_SYNTAX_CHAR_P (*syn))
390 	{
391 	  (*info->fprintf_func) (info->stream, "%c", CGEN_SYNTAX_CHAR (*syn));
392 	  continue;
393 	}
394 
395       /* We have an operand.  */
396       epiphany_cgen_print_operand (cd, CGEN_SYNTAX_FIELD (*syn), info,
397 				 fields, CGEN_INSN_ATTRS (insn), pc, length);
398     }
399 }
400 
401 /* Subroutine of print_insn. Reads an insn into the given buffers and updates
402    the extract info.
403    Returns 0 if all is well, non-zero otherwise.  */
404 
405 static int
read_insn(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,bfd_vma pc,disassemble_info * info,bfd_byte * buf,int buflen,CGEN_EXTRACT_INFO * ex_info,unsigned long * insn_value)406 read_insn (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
407 	   bfd_vma pc,
408 	   disassemble_info *info,
409 	   bfd_byte *buf,
410 	   int buflen,
411 	   CGEN_EXTRACT_INFO *ex_info,
412 	   unsigned long *insn_value)
413 {
414   int status = (*info->read_memory_func) (pc, buf, buflen, info);
415 
416   if (status != 0)
417     {
418       (*info->memory_error_func) (status, pc, info);
419       return -1;
420     }
421 
422   ex_info->dis_info = info;
423   ex_info->valid = (1 << buflen) - 1;
424   ex_info->insn_bytes = buf;
425 
426   *insn_value = bfd_get_bits (buf, buflen * 8, info->endian == BFD_ENDIAN_BIG);
427   return 0;
428 }
429 
430 /* Utility to print an insn.
431    BUF is the base part of the insn, target byte order, BUFLEN bytes long.
432    The result is the size of the insn in bytes or zero for an unknown insn
433    or -1 if an error occurs fetching data (memory_error_func will have
434    been called).  */
435 
436 static int
print_insn(CGEN_CPU_DESC cd,bfd_vma pc,disassemble_info * info,bfd_byte * buf,unsigned int buflen)437 print_insn (CGEN_CPU_DESC cd,
438 	    bfd_vma pc,
439 	    disassemble_info *info,
440 	    bfd_byte *buf,
441 	    unsigned int buflen)
442 {
443   CGEN_INSN_INT insn_value;
444   const CGEN_INSN_LIST *insn_list;
445   CGEN_EXTRACT_INFO ex_info;
446   int basesize;
447 
448   /* Extract base part of instruction, just in case CGEN_DIS_* uses it. */
449   basesize = cd->base_insn_bitsize < buflen * 8 ?
450                                      cd->base_insn_bitsize : buflen * 8;
451   insn_value = cgen_get_insn_value (cd, buf, basesize);
452 
453 
454   /* Fill in ex_info fields like read_insn would.  Don't actually call
455      read_insn, since the incoming buffer is already read (and possibly
456      modified a la m32r).  */
457   ex_info.valid = (1 << buflen) - 1;
458   ex_info.dis_info = info;
459   ex_info.insn_bytes = buf;
460 
461   /* The instructions are stored in hash lists.
462      Pick the first one and keep trying until we find the right one.  */
463 
464   insn_list = CGEN_DIS_LOOKUP_INSN (cd, (char *) buf, insn_value);
465   while (insn_list != NULL)
466     {
467       const CGEN_INSN *insn = insn_list->insn;
468       CGEN_FIELDS fields;
469       int length;
470       unsigned long insn_value_cropped;
471 
472 #ifdef CGEN_VALIDATE_INSN_SUPPORTED
473       /* Not needed as insn shouldn't be in hash lists if not supported.  */
474       /* Supported by this cpu?  */
475       if (! epiphany_cgen_insn_supported (cd, insn))
476         {
477           insn_list = CGEN_DIS_NEXT_INSN (insn_list);
478 	  continue;
479         }
480 #endif
481 
482       /* Basic bit mask must be correct.  */
483       /* ??? May wish to allow target to defer this check until the extract
484 	 handler.  */
485 
486       /* Base size may exceed this instruction's size.  Extract the
487          relevant part from the buffer. */
488       if ((unsigned) (CGEN_INSN_BITSIZE (insn) / 8) < buflen &&
489 	  (unsigned) (CGEN_INSN_BITSIZE (insn) / 8) <= sizeof (unsigned long))
490 	insn_value_cropped = bfd_get_bits (buf, CGEN_INSN_BITSIZE (insn),
491 					   info->endian == BFD_ENDIAN_BIG);
492       else
493 	insn_value_cropped = insn_value;
494 
495       if ((insn_value_cropped & CGEN_INSN_BASE_MASK (insn))
496 	  == CGEN_INSN_BASE_VALUE (insn))
497 	{
498 	  /* Printing is handled in two passes.  The first pass parses the
499 	     machine insn and extracts the fields.  The second pass prints
500 	     them.  */
501 
502 	  /* Make sure the entire insn is loaded into insn_value, if it
503 	     can fit.  */
504 	  if (((unsigned) CGEN_INSN_BITSIZE (insn) > cd->base_insn_bitsize) &&
505 	      (unsigned) (CGEN_INSN_BITSIZE (insn) / 8) <= sizeof (unsigned long))
506 	    {
507 	      unsigned long full_insn_value;
508 	      int rc = read_insn (cd, pc, info, buf,
509 				  CGEN_INSN_BITSIZE (insn) / 8,
510 				  & ex_info, & full_insn_value);
511 	      if (rc != 0)
512 		return rc;
513 	      length = CGEN_EXTRACT_FN (cd, insn)
514 		(cd, insn, &ex_info, full_insn_value, &fields, pc);
515 	    }
516 	  else
517 	    length = CGEN_EXTRACT_FN (cd, insn)
518 	      (cd, insn, &ex_info, insn_value_cropped, &fields, pc);
519 
520 	  /* Length < 0 -> error.  */
521 	  if (length < 0)
522 	    return length;
523 	  if (length > 0)
524 	    {
525 	      CGEN_PRINT_FN (cd, insn) (cd, info, insn, &fields, pc, length);
526 	      /* Length is in bits, result is in bytes.  */
527 	      return length / 8;
528 	    }
529 	}
530 
531       insn_list = CGEN_DIS_NEXT_INSN (insn_list);
532     }
533 
534   return 0;
535 }
536 
537 /* Default value for CGEN_PRINT_INSN.
538    The result is the size of the insn in bytes or zero for an unknown insn
539    or -1 if an error occured fetching bytes.  */
540 
541 #ifndef CGEN_PRINT_INSN
542 #define CGEN_PRINT_INSN default_print_insn
543 #endif
544 
545 static int
default_print_insn(CGEN_CPU_DESC cd,bfd_vma pc,disassemble_info * info)546 default_print_insn (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
547 {
548   bfd_byte buf[CGEN_MAX_INSN_SIZE];
549   int buflen;
550   int status;
551 
552   /* Attempt to read the base part of the insn.  */
553   buflen = cd->base_insn_bitsize / 8;
554   status = (*info->read_memory_func) (pc, buf, buflen, info);
555 
556   /* Try again with the minimum part, if min < base.  */
557   if (status != 0 && (cd->min_insn_bitsize < cd->base_insn_bitsize))
558     {
559       buflen = cd->min_insn_bitsize / 8;
560       status = (*info->read_memory_func) (pc, buf, buflen, info);
561     }
562 
563   if (status != 0)
564     {
565       (*info->memory_error_func) (status, pc, info);
566       return -1;
567     }
568 
569   return print_insn (cd, pc, info, buf, buflen);
570 }
571 
572 /* Main entry point.
573    Print one instruction from PC on INFO->STREAM.
574    Return the size of the instruction (in bytes).  */
575 
576 typedef struct cpu_desc_list
577 {
578   struct cpu_desc_list *next;
579   CGEN_BITSET *isa;
580   int mach;
581   int endian;
582   CGEN_CPU_DESC cd;
583 } cpu_desc_list;
584 
585 int
print_insn_epiphany(bfd_vma pc,disassemble_info * info)586 print_insn_epiphany (bfd_vma pc, disassemble_info *info)
587 {
588   static cpu_desc_list *cd_list = 0;
589   cpu_desc_list *cl = 0;
590   static CGEN_CPU_DESC cd = 0;
591   static CGEN_BITSET *prev_isa;
592   static int prev_mach;
593   static int prev_endian;
594   int length;
595   CGEN_BITSET *isa;
596   int mach;
597   int endian = (info->endian == BFD_ENDIAN_BIG
598 		? CGEN_ENDIAN_BIG
599 		: CGEN_ENDIAN_LITTLE);
600   enum bfd_architecture arch;
601 
602   /* ??? gdb will set mach but leave the architecture as "unknown" */
603 #ifndef CGEN_BFD_ARCH
604 #define CGEN_BFD_ARCH bfd_arch_epiphany
605 #endif
606   arch = info->arch;
607   if (arch == bfd_arch_unknown)
608     arch = CGEN_BFD_ARCH;
609 
610   /* There's no standard way to compute the machine or isa number
611      so we leave it to the target.  */
612 #ifdef CGEN_COMPUTE_MACH
613   mach = CGEN_COMPUTE_MACH (info);
614 #else
615   mach = info->mach;
616 #endif
617 
618 #ifdef CGEN_COMPUTE_ISA
619   {
620     static CGEN_BITSET *permanent_isa;
621 
622     if (!permanent_isa)
623       permanent_isa = cgen_bitset_create (MAX_ISAS);
624     isa = permanent_isa;
625     cgen_bitset_clear (isa);
626     cgen_bitset_add (isa, CGEN_COMPUTE_ISA (info));
627   }
628 #else
629   isa = info->insn_sets;
630 #endif
631 
632   /* If we've switched cpu's, try to find a handle we've used before */
633   if (cd
634       && (cgen_bitset_compare (isa, prev_isa) != 0
635 	  || mach != prev_mach
636 	  || endian != prev_endian))
637     {
638       cd = 0;
639       for (cl = cd_list; cl; cl = cl->next)
640 	{
641 	  if (cgen_bitset_compare (cl->isa, isa) == 0 &&
642 	      cl->mach == mach &&
643 	      cl->endian == endian)
644 	    {
645 	      cd = cl->cd;
646  	      prev_isa = cd->isas;
647 	      break;
648 	    }
649 	}
650     }
651 
652   /* If we haven't initialized yet, initialize the opcode table.  */
653   if (! cd)
654     {
655       const bfd_arch_info_type *arch_type = bfd_lookup_arch (arch, mach);
656       const char *mach_name;
657 
658       if (!arch_type)
659 	abort ();
660       mach_name = arch_type->printable_name;
661 
662       prev_isa = cgen_bitset_copy (isa);
663       prev_mach = mach;
664       prev_endian = endian;
665       cd = epiphany_cgen_cpu_open (CGEN_CPU_OPEN_ISAS, prev_isa,
666 				 CGEN_CPU_OPEN_BFDMACH, mach_name,
667 				 CGEN_CPU_OPEN_ENDIAN, prev_endian,
668 				 CGEN_CPU_OPEN_END);
669       if (!cd)
670 	abort ();
671 
672       /* Save this away for future reference.  */
673       cl = xmalloc (sizeof (struct cpu_desc_list));
674       cl->cd = cd;
675       cl->isa = prev_isa;
676       cl->mach = mach;
677       cl->endian = endian;
678       cl->next = cd_list;
679       cd_list = cl;
680 
681       epiphany_cgen_init_dis (cd);
682     }
683 
684   /* We try to have as much common code as possible.
685      But at this point some targets need to take over.  */
686   /* ??? Some targets may need a hook elsewhere.  Try to avoid this,
687      but if not possible try to move this hook elsewhere rather than
688      have two hooks.  */
689   length = CGEN_PRINT_INSN (cd, pc, info);
690   if (length > 0)
691     return length;
692   if (length < 0)
693     return -1;
694 
695   (*info->fprintf_func) (info->stream, UNKNOWN_INSN_MSG);
696   return cd->default_insn_bitsize / 8;
697 }
698