Index: opcodes
[deliverable/binutils-gdb.git] / opcodes / mep-dis.c
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 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2005, 2007,
8 2008 Free Software Foundation, Inc.
9
10 This file is part of libopcodes.
11
12 This library is free software; you can redistribute it and/or modify
13 it under the terms of the GNU General Public License as published by
14 the Free Software Foundation; either version 3, or (at your option)
15 any later version.
16
17 It is distributed in the hope that it will be useful, but WITHOUT
18 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
19 or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
20 License for more details.
21
22 You should have received a copy of the GNU General Public License
23 along with this program; if not, write to the Free Software Foundation, Inc.,
24 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */
25
26 /* ??? Eventually more and more of this stuff can go to cpu-independent files.
27 Keep that in mind. */
28
29 #include "sysdep.h"
30 #include <stdio.h>
31 #include "ansidecl.h"
32 #include "dis-asm.h"
33 #include "bfd.h"
34 #include "symcat.h"
35 #include "libiberty.h"
36 #include "mep-desc.h"
37 #include "mep-opc.h"
38 #include "opintl.h"
39
40 /* Default text to print if an instruction isn't recognized. */
41 #define UNKNOWN_INSN_MSG _("*unknown*")
42
43 static void print_normal
44 (CGEN_CPU_DESC, void *, long, unsigned int, bfd_vma, int);
45 static void print_address
46 (CGEN_CPU_DESC, void *, bfd_vma, unsigned int, bfd_vma, int) ATTRIBUTE_UNUSED;
47 static void print_keyword
48 (CGEN_CPU_DESC, void *, CGEN_KEYWORD *, long, unsigned int) ATTRIBUTE_UNUSED;
49 static void print_insn_normal
50 (CGEN_CPU_DESC, void *, const CGEN_INSN *, CGEN_FIELDS *, bfd_vma, int);
51 static int print_insn
52 (CGEN_CPU_DESC, bfd_vma, disassemble_info *, bfd_byte *, unsigned);
53 static int default_print_insn
54 (CGEN_CPU_DESC, bfd_vma, disassemble_info *) ATTRIBUTE_UNUSED;
55 static int read_insn
56 (CGEN_CPU_DESC, bfd_vma, disassemble_info *, bfd_byte *, int, CGEN_EXTRACT_INFO *,
57 unsigned long *);
58 \f
59 /* -- disassembler routines inserted here. */
60
61 /* -- dis.c */
62
63 #include "elf/mep.h"
64 #include "elf-bfd.h"
65
66 #define CGEN_VALIDATE_INSN_SUPPORTED
67
68 static void print_tpreg (CGEN_CPU_DESC, PTR, CGEN_KEYWORD *, long, unsigned int);
69 static void print_spreg (CGEN_CPU_DESC, PTR, CGEN_KEYWORD *, long, unsigned int);
70
71 static void
72 print_tpreg (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED, PTR dis_info,
73 CGEN_KEYWORD *table ATTRIBUTE_UNUSED, long val ATTRIBUTE_UNUSED,
74 unsigned int flags ATTRIBUTE_UNUSED)
75 {
76 disassemble_info *info = (disassemble_info *) dis_info;
77
78 (*info->fprintf_func) (info->stream, "$tp");
79 }
80
81 static void
82 print_spreg (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED, PTR dis_info,
83 CGEN_KEYWORD *table ATTRIBUTE_UNUSED, long val ATTRIBUTE_UNUSED,
84 unsigned int flags ATTRIBUTE_UNUSED)
85 {
86 disassemble_info *info = (disassemble_info *) dis_info;
87
88 (*info->fprintf_func) (info->stream, "$sp");
89 }
90
91 /* begin-cop-ip-print-handlers */
92 static void
93 print_ivc2_cr (CGEN_CPU_DESC,
94 void *,
95 CGEN_KEYWORD *,
96 long,
97 unsigned int) ATTRIBUTE_UNUSED;
98 static void
99 print_ivc2_cr (CGEN_CPU_DESC cd,
100 void *dis_info,
101 CGEN_KEYWORD *keyword_table ATTRIBUTE_UNUSED,
102 long value,
103 unsigned int attrs)
104 {
105 print_keyword (cd, dis_info, & mep_cgen_opval_h_cr_ivc2, value, attrs);
106 }
107 static void
108 print_ivc2_ccr (CGEN_CPU_DESC,
109 void *,
110 CGEN_KEYWORD *,
111 long,
112 unsigned int) ATTRIBUTE_UNUSED;
113 static void
114 print_ivc2_ccr (CGEN_CPU_DESC cd,
115 void *dis_info,
116 CGEN_KEYWORD *keyword_table ATTRIBUTE_UNUSED,
117 long value,
118 unsigned int attrs)
119 {
120 print_keyword (cd, dis_info, & mep_cgen_opval_h_ccr_ivc2, value, attrs);
121 }
122 /* end-cop-ip-print-handlers */
123
124 /************************************************************\
125 *********************** Experimental *************************
126 \************************************************************/
127
128 #undef CGEN_PRINT_INSN
129 #define CGEN_PRINT_INSN mep_print_insn
130
131 static int
132 mep_print_vliw_insns (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info,
133 bfd_byte *buf, int corelength, int copro1length,
134 int copro2length ATTRIBUTE_UNUSED)
135 {
136 int i;
137 int status = 0;
138 /* char insnbuf[CGEN_MAX_INSN_SIZE]; */
139 bfd_byte insnbuf[64];
140
141 /* If corelength > 0 then there is a core insn present. It
142 will be at the beginning of the buffer. After printing
143 the core insn, we need to print the + on the next line. */
144 if (corelength > 0)
145 {
146 int my_status = 0;
147
148 for (i = 0; i < corelength; i++ )
149 insnbuf[i] = buf[i];
150 cd->isas = & MEP_CORE_ISA;
151
152 my_status = print_insn (cd, pc, info, insnbuf, corelength);
153 if (my_status != corelength)
154 {
155 (*info->fprintf_func) (info->stream, UNKNOWN_INSN_MSG);
156 my_status = corelength;
157 }
158 status += my_status;
159
160 /* Print the + to indicate that the following copro insn is */
161 /* part of a vliw group. */
162 if (copro1length > 0)
163 (*info->fprintf_func) (info->stream, " + ");
164 }
165
166 /* Now all that is left to be processed is the coprocessor insns
167 In vliw mode, there will always be one. Its positioning will
168 be from byte corelength to byte corelength+copro1length -1.
169 No need to check for existence. Also, the first vliw insn,
170 will, as spec'd, always be at least as long as the core insn
171 so we don't need to flush the buffer. */
172 if (copro1length > 0)
173 {
174 int my_status = 0;
175
176 for (i = corelength; i < corelength + copro1length; i++ )
177 insnbuf[i - corelength] = buf[i];
178
179 switch (copro1length)
180 {
181 case 0:
182 break;
183 case 2:
184 cd->isas = & MEP_COP16_ISA;
185 break;
186 case 4:
187 cd->isas = & MEP_COP32_ISA;
188 break;
189 case 6:
190 cd->isas = & MEP_COP48_ISA;
191 break;
192 case 8:
193 cd->isas = & MEP_COP64_ISA;
194 break;
195 default:
196 /* Shouldn't be anything but 16,32,48,64. */
197 break;
198 }
199
200 my_status = print_insn (cd, pc, info, insnbuf, copro1length);
201
202 if (my_status != copro1length)
203 {
204 (*info->fprintf_func) (info->stream, UNKNOWN_INSN_MSG);
205 my_status = copro1length;
206 }
207 status += my_status;
208 }
209
210 #if 0
211 /* Now we need to process the second copro insn if it exists. We
212 have no guarantee that the second copro insn will be longer
213 than the first, so we have to flush the buffer if we are have
214 a second copro insn to process. If present, this insn will
215 be in the position from byte corelength+copro1length to byte
216 corelength+copro1length+copro2length-1 (which better equal 8
217 or else we're in big trouble. */
218 if (copro2length > 0)
219 {
220 int my_status = 0;
221
222 for (i = 0; i < 64 ; i++)
223 insnbuf[i] = 0;
224
225 for (i = corelength + copro1length; i < 64; i++)
226 insnbuf[i - (corelength + copro1length)] = buf[i];
227
228 switch (copro2length)
229 {
230 case 2:
231 cd->isas = 1 << ISA_EXT_COP1_16;
232 break;
233 case 4:
234 cd->isas = 1 << ISA_EXT_COP1_32;
235 break;
236 case 6:
237 cd->isas = 1 << ISA_EXT_COP1_48;
238 break;
239 case 8:
240 cd->isas = 1 << ISA_EXT_COP1_64;
241 break;
242 default:
243 /* Shouldn't be anything but 16,32,48,64. */
244 break;
245 }
246
247 my_status = print_insn (cd, pc, info, insnbuf, copro2length);
248
249 if (my_status != copro2length)
250 {
251 (*info->fprintf_func) (info->stream, UNKNOWN_INSN_MSG);
252 my_status = copro2length;
253 }
254
255 status += my_status;
256 }
257 #endif
258
259 /* Status should now be the number of bytes that were printed
260 which should be 4 for VLIW32 mode and 64 for VLIW64 mode. */
261
262 if ((!MEP_VLIW64 && (status != 4)) || (MEP_VLIW64 && (status != 8)))
263 return -1;
264 else
265 return status;
266 }
267
268 /* The two functions mep_examine_vliw[32,64]_insns are used find out
269 which vliw combinaion (16 bit core with 48 bit copro, 32 bit core
270 with 32 bit copro, etc.) is present. Later on, when internally
271 parallel coprocessors are handled, only these functions should
272 need to be changed.
273
274 At this time only the following combinations are supported:
275
276 VLIW32 Mode:
277 16 bit core insn (core) and 16 bit coprocessor insn (cop1)
278 32 bit core insn (core)
279 32 bit coprocessor insn (cop1)
280 Note: As of this time, I do not believe we have enough information
281 to distinguish a 32 bit core insn from a 32 bit cop insn. Also,
282 no 16 bit coprocessor insns have been specified.
283
284 VLIW64 Mode:
285 16 bit core insn (core) and 48 bit coprocessor insn (cop1)
286 32 bit core insn (core) and 32 bit coprocessor insn (cop1)
287 64 bit coprocessor insn (cop1)
288
289 The framework for an internally parallel coprocessor is also
290 present (2nd coprocessor insn is cop2), but at this time it
291 is not used. This only appears to be valid in VLIW64 mode. */
292
293 static int
294 mep_examine_vliw32_insns (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
295 {
296 int status;
297 int buflength;
298 int corebuflength;
299 int cop1buflength;
300 int cop2buflength;
301 bfd_byte buf[CGEN_MAX_INSN_SIZE];
302 char indicator16[1];
303 char indicatorcop32[2];
304
305 /* At this time we're not supporting internally parallel coprocessors,
306 so cop2buflength will always be 0. */
307 cop2buflength = 0;
308
309 /* Read in 32 bits. */
310 buflength = 4; /* VLIW insn spans 4 bytes. */
311 status = (*info->read_memory_func) (pc, buf, buflength, info);
312
313 if (status != 0)
314 {
315 (*info->memory_error_func) (status, pc, info);
316 return -1;
317 }
318
319 /* Put the big endian representation of the bytes to be examined
320 in the temporary buffers for examination. */
321
322 if (info->endian == BFD_ENDIAN_BIG)
323 {
324 indicator16[0] = buf[0];
325 indicatorcop32[0] = buf[0];
326 indicatorcop32[1] = buf[1];
327 }
328 else
329 {
330 indicator16[0] = buf[1];
331 indicatorcop32[0] = buf[1];
332 indicatorcop32[1] = buf[0];
333 }
334
335 /* If the two high order bits are 00, 01 or 10, we have a 16 bit
336 core insn and a 48 bit copro insn. */
337
338 if ((indicator16[0] & 0x80) && (indicator16[0] & 0x40))
339 {
340 if ((indicatorcop32[0] & 0xf0) == 0xf0 && (indicatorcop32[1] & 0x07) == 0x07)
341 {
342 /* We have a 32 bit copro insn. */
343 corebuflength = 0;
344 /* All 4 4ytes are one copro insn. */
345 cop1buflength = 4;
346 }
347 else
348 {
349 /* We have a 32 bit core. */
350 corebuflength = 4;
351 cop1buflength = 0;
352 }
353 }
354 else
355 {
356 /* We have a 16 bit core insn and a 16 bit copro insn. */
357 corebuflength = 2;
358 cop1buflength = 2;
359 }
360
361 /* Now we have the distrubution set. Print them out. */
362 status = mep_print_vliw_insns (cd, pc, info, buf, corebuflength,
363 cop1buflength, cop2buflength);
364
365 return status;
366 }
367
368 static int
369 mep_examine_vliw64_insns (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
370 {
371 int status;
372 int buflength;
373 int corebuflength;
374 int cop1buflength;
375 int cop2buflength;
376 bfd_byte buf[CGEN_MAX_INSN_SIZE];
377 char indicator16[1];
378 char indicator64[4];
379
380 /* At this time we're not supporting internally parallel
381 coprocessors, so cop2buflength will always be 0. */
382 cop2buflength = 0;
383
384 /* Read in 64 bits. */
385 buflength = 8; /* VLIW insn spans 8 bytes. */
386 status = (*info->read_memory_func) (pc, buf, buflength, info);
387
388 if (status != 0)
389 {
390 (*info->memory_error_func) (status, pc, info);
391 return -1;
392 }
393
394 /* We have all 64 bits in the buffer now. We have to figure out
395 what combination of instruction sizes are present. The two
396 high order bits will indicate whether or not we have a 16 bit
397 core insn or not. If not, then we have to look at the 7,8th
398 bytes to tell whether we have 64 bit copro insn or a 32 bit
399 core insn with a 32 bit copro insn. Endianness will make a
400 difference here. */
401
402 /* Put the big endian representation of the bytes to be examined
403 in the temporary buffers for examination. */
404
405 /* indicator16[0] = buf[0]; */
406 if (info->endian == BFD_ENDIAN_BIG)
407 {
408 indicator16[0] = buf[0];
409 indicator64[0] = buf[0];
410 indicator64[1] = buf[1];
411 indicator64[2] = buf[2];
412 indicator64[3] = buf[3];
413 }
414 else
415 {
416 indicator16[0] = buf[1];
417 indicator64[0] = buf[1];
418 indicator64[1] = buf[0];
419 indicator64[2] = buf[3];
420 indicator64[3] = buf[2];
421 }
422
423 /* If the two high order bits are 00, 01 or 10, we have a 16 bit
424 core insn and a 48 bit copro insn. */
425
426 if ((indicator16[0] & 0x80) && (indicator16[0] & 0x40))
427 {
428 if ((indicator64[0] & 0xf0) == 0xf0 && (indicator64[1] & 0x07) == 0x07
429 && ((indicator64[2] & 0xfe) != 0xf0 || (indicator64[3] & 0xf4) != 0))
430 {
431 /* We have a 64 bit copro insn. */
432 corebuflength = 0;
433 /* All 8 bytes are one copro insn. */
434 cop1buflength = 8;
435 }
436 else
437 {
438 /* We have a 32 bit core insn and a 32 bit copro insn. */
439 corebuflength = 4;
440 cop1buflength = 4;
441 }
442 }
443 else
444 {
445 /* We have a 16 bit core insn and a 48 bit copro insn. */
446 corebuflength = 2;
447 cop1buflength = 6;
448 }
449
450 /* Now we have the distrubution set. Print them out. */
451 status = mep_print_vliw_insns (cd, pc, info, buf, corebuflength,
452 cop1buflength, cop2buflength);
453
454 return status;
455 }
456
457 #ifdef MEP_IVC2_SUPPORTED
458
459 static int
460 print_slot_insn (CGEN_CPU_DESC cd,
461 bfd_vma pc,
462 disassemble_info *info,
463 SLOTS_ATTR slot,
464 bfd_byte *buf)
465 {
466 const CGEN_INSN_LIST *insn_list;
467 CGEN_INSN_INT insn_value;
468 CGEN_EXTRACT_INFO ex_info;
469
470 insn_value = cgen_get_insn_value (cd, buf, 32);
471
472 /* Fill in ex_info fields like read_insn would. Don't actually call
473 read_insn, since the incoming buffer is already read (and possibly
474 modified a la m32r). */
475 ex_info.valid = (1 << 8) - 1;
476 ex_info.dis_info = info;
477 ex_info.insn_bytes = buf;
478
479 /* The instructions are stored in hash lists.
480 Pick the first one and keep trying until we find the right one. */
481
482 insn_list = CGEN_DIS_LOOKUP_INSN (cd, (char *) buf, insn_value);
483 while (insn_list != NULL)
484 {
485 const CGEN_INSN *insn = insn_list->insn;
486 CGEN_FIELDS fields;
487 int length;
488
489 if ((CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_CONFIG)
490 && CGEN_INSN_ATTR_VALUE (insn, CGEN_INSN_CONFIG) != MEP_CONFIG)
491 || ! (CGEN_ATTR_CGEN_INSN_SLOTS_VALUE (CGEN_INSN_ATTRS (insn)) & (1 << slot)))
492 {
493 insn_list = CGEN_DIS_NEXT_INSN (insn_list);
494 continue;
495 }
496
497 if ((insn_value & CGEN_INSN_BASE_MASK (insn))
498 == CGEN_INSN_BASE_VALUE (insn))
499 {
500 /* Printing is handled in two passes. The first pass parses the
501 machine insn and extracts the fields. The second pass prints
502 them. */
503
504 length = CGEN_EXTRACT_FN (cd, insn)
505 (cd, insn, &ex_info, insn_value, &fields, pc);
506
507 /* Length < 0 -> error. */
508 if (length < 0)
509 return length;
510 if (length > 0)
511 {
512 CGEN_PRINT_FN (cd, insn) (cd, info, insn, &fields, pc, length);
513 /* Length is in bits, result is in bytes. */
514 return length / 8;
515 }
516 }
517
518 insn_list = CGEN_DIS_NEXT_INSN (insn_list);
519 }
520
521 if (slot == SLOTS_P0S)
522 (*info->fprintf_func) (info->stream, "*unknown-p0s*");
523 else if (slot == SLOTS_P0)
524 (*info->fprintf_func) (info->stream, "*unknown-p0*");
525 else if (slot == SLOTS_P1)
526 (*info->fprintf_func) (info->stream, "*unknown-p1*");
527 else if (slot == SLOTS_C3)
528 (*info->fprintf_func) (info->stream, "*unknown-c3*");
529 return 0;
530 }
531
532 static int
533 mep_examine_ivc2_insns (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED, bfd_vma pc ATTRIBUTE_UNUSED, disassemble_info *info ATTRIBUTE_UNUSED)
534 {
535 int status;
536 int buflength;
537 int cop2buflength;
538 bfd_byte buf[8];
539 bfd_byte insn[8];
540 int e;
541
542 /* At this time we're not supporting internally parallel
543 coprocessors, so cop2buflength will always be 0. */
544 cop2buflength = 0;
545
546 /* Read in 64 bits. */
547 buflength = 8; /* VLIW insn spans 8 bytes. */
548 status = (*info->read_memory_func) (pc, buf, buflength, info);
549
550 if (status != 0)
551 {
552 (*info->memory_error_func) (status, pc, info);
553 return -1;
554 }
555
556 if (info->endian == BFD_ENDIAN_LITTLE)
557 e = 1;
558 else
559 e = 0;
560
561 if ((buf[0^e] & 0xf0) != 0xf0)
562 {
563 /* <--00--><--11--><--22--><--33--><--44--><--55--><--66--><--77--> */
564 /* V1 [-----core-----][--------p0s-------][------------p1------------] */
565
566 print_insn (cd, pc, info, buf, 2);
567
568 insn[0^e] = 0;
569 insn[1^e] = buf[2^e];
570 insn[2^e] = buf[3^e];
571 insn[3^e] = buf[4^e] & 0xf0;
572 (*info->fprintf_func) (info->stream, " + ");
573 print_slot_insn (cd, pc, info, SLOTS_P0S, insn);
574
575 insn[0^e] = buf[4^e] << 4 | buf[5^e] >> 4;
576 insn[1^e] = buf[5^e] << 4 | buf[6^e] >> 4;
577 insn[2^e] = buf[6^e] << 4 | buf[7^e] >> 4;
578 insn[3^e] = buf[7^e] << 4;
579 (*info->fprintf_func) (info->stream, " + ");
580 print_slot_insn (cd, pc, info, SLOTS_P1, insn);
581 }
582 else if ((buf[0^e] & 0xf0) == 0xf0 && (buf[1^e] & 0x0f) == 0x07)
583 {
584 /* <--00--><--11--><--22--><--33--><--44--><--55--><--66--><--77--> */
585 /* V3 1111[--p0--]0111[--------p0--------][------------p1------------] */
586 /* 00000000111111112222222233333333 */
587
588 insn[0^e] = buf[0^e] << 4 | buf[1^e] >> 4;
589 insn[1^e] = buf[2^e];
590 insn[2^e] = buf[3^e];
591 insn[3^e] = buf[4^e] & 0xf0;
592 print_slot_insn (cd, pc, info, SLOTS_P0, insn);
593
594 insn[0^e] = buf[4^e] << 4 | buf[5^e] >> 4;
595 insn[1^e] = buf[5^e] << 4 | buf[6^e] >> 4;
596 insn[2^e] = buf[6^e] << 4 | buf[7^e] >> 4;
597 insn[3^e] = buf[7^e] << 4;
598 (*info->fprintf_func) (info->stream, " + ");
599 print_slot_insn (cd, pc, info, SLOTS_P1, insn);
600 }
601 else
602 {
603 /* <--00--><--11--><--22--><--33--><--44--><--55--><--66--><--77--> */
604 /* V2 [-------------core-------------]xxxx[------------p1------------] */
605 print_insn (cd, pc, info, buf, 4);
606
607 insn[0^e] = buf[4^e] << 4 | buf[5^e] >> 4;
608 insn[1^e] = buf[5^e] << 4 | buf[6^e] >> 4;
609 insn[2^e] = buf[6^e] << 4 | buf[7^e] >> 4;
610 insn[3^e] = buf[7^e] << 4;
611 (*info->fprintf_func) (info->stream, " + ");
612 print_slot_insn (cd, pc, info, SLOTS_P1, insn);
613 }
614
615 return 8;
616 }
617
618 #endif /* MEP_IVC2_SUPPORTED */
619
620 /* This is a hack. SID calls this to update the disassembler as the
621 CPU changes modes. */
622 static int mep_ivc2_disassemble_p = 0;
623 static int mep_ivc2_vliw_disassemble_p = 0;
624
625 void
626 mep_print_insn_set_ivc2_mode (int ivc2_p, int vliw_p, int cfg_idx);
627 void
628 mep_print_insn_set_ivc2_mode (int ivc2_p, int vliw_p, int cfg_idx)
629 {
630 mep_ivc2_disassemble_p = ivc2_p;
631 mep_ivc2_vliw_disassemble_p = vliw_p;
632 mep_config_index = cfg_idx;
633 }
634
635 static int
636 mep_print_insn (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
637 {
638 int status;
639 int cop_type;
640 int ivc2 = 0;
641 static CGEN_ATTR_VALUE_BITSET_TYPE *ivc2_core_isa = NULL;
642
643 if (ivc2_core_isa == NULL)
644 {
645 /* IVC2 has some core-only coprocessor instructions. We
646 use COP32 to flag those, and COP64 for the VLIW ones,
647 since they have the same names. */
648 ivc2_core_isa = cgen_bitset_create (MAX_ISAS);
649 }
650
651 /* Extract and adapt to configuration number, if available. */
652 if (info->section && info->section->owner)
653 {
654 bfd *abfd = info->section->owner;
655 mep_config_index = abfd->tdata.elf_obj_data->elf_header->e_flags & EF_MEP_INDEX_MASK;
656 /* This instantly redefines MEP_CONFIG, MEP_OMASK, .... MEP_VLIW64 */
657
658 cop_type = abfd->tdata.elf_obj_data->elf_header->e_flags & EF_MEP_COP_MASK;
659 if (cop_type == EF_MEP_COP_IVC2)
660 ivc2 = 1;
661 }
662
663 /* Picking the right ISA bitmask for the current context is tricky. */
664 if (info->section)
665 {
666 if (info->section->flags & SEC_MEP_VLIW)
667 {
668 #ifdef MEP_IVC2_SUPPORTED
669 if (ivc2)
670 {
671 /* ivc2 has its own way of selecting its functions. */
672 cd->isas = & MEP_CORE_ISA;
673 status = mep_examine_ivc2_insns (cd, pc, info);
674 }
675 else
676 #endif
677 /* Are we in 32 or 64 bit vliw mode? */
678 if (MEP_VLIW64)
679 status = mep_examine_vliw64_insns (cd, pc, info);
680 else
681 status = mep_examine_vliw32_insns (cd, pc, info);
682 /* Both the above branches set their own isa bitmasks. */
683 }
684 else
685 {
686 if (ivc2)
687 {
688 cgen_bitset_clear (ivc2_core_isa);
689 cgen_bitset_union (ivc2_core_isa, &MEP_CORE_ISA, ivc2_core_isa);
690 cgen_bitset_union (ivc2_core_isa, &MEP_COP32_ISA, ivc2_core_isa);
691 cd->isas = ivc2_core_isa;
692 }
693 else
694 cd->isas = & MEP_CORE_ISA;
695 status = default_print_insn (cd, pc, info);
696 }
697 }
698 else /* sid or gdb */
699 {
700 #ifdef MEP_IVC2_SUPPORTED
701 if (mep_ivc2_disassemble_p)
702 {
703 if (mep_ivc2_vliw_disassemble_p)
704 {
705 cd->isas = & MEP_CORE_ISA;
706 status = mep_examine_ivc2_insns (cd, pc, info);
707 return status;
708 }
709 else
710 {
711 if (ivc2)
712 cd->isas = ivc2_core_isa;
713 }
714 }
715 #endif
716
717 status = default_print_insn (cd, pc, info);
718 }
719
720 return status;
721 }
722
723
724 /* -- opc.c */
725
726 void mep_cgen_print_operand
727 (CGEN_CPU_DESC, int, PTR, CGEN_FIELDS *, void const *, bfd_vma, int);
728
729 /* Main entry point for printing operands.
730 XINFO is a `void *' and not a `disassemble_info *' to not put a requirement
731 of dis-asm.h on cgen.h.
732
733 This function is basically just a big switch statement. Earlier versions
734 used tables to look up the function to use, but
735 - if the table contains both assembler and disassembler functions then
736 the disassembler contains much of the assembler and vice-versa,
737 - there's a lot of inlining possibilities as things grow,
738 - using a switch statement avoids the function call overhead.
739
740 This function could be moved into `print_insn_normal', but keeping it
741 separate makes clear the interface between `print_insn_normal' and each of
742 the handlers. */
743
744 void
745 mep_cgen_print_operand (CGEN_CPU_DESC cd,
746 int opindex,
747 void * xinfo,
748 CGEN_FIELDS *fields,
749 void const *attrs ATTRIBUTE_UNUSED,
750 bfd_vma pc,
751 int length)
752 {
753 disassemble_info *info = (disassemble_info *) xinfo;
754
755 switch (opindex)
756 {
757 case MEP_OPERAND_ADDR24A4 :
758 print_normal (cd, info, fields->f_24u8a4n, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
759 break;
760 case MEP_OPERAND_C5RMUIMM20 :
761 print_normal (cd, info, fields->f_c5_rmuimm20, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
762 break;
763 case MEP_OPERAND_C5RNMUIMM24 :
764 print_normal (cd, info, fields->f_c5_rnmuimm24, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
765 break;
766 case MEP_OPERAND_CALLNUM :
767 print_normal (cd, info, fields->f_callnum, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
768 break;
769 case MEP_OPERAND_CCCC :
770 print_normal (cd, info, fields->f_rm, 0, pc, length);
771 break;
772 case MEP_OPERAND_CCRN :
773 print_keyword (cd, info, & mep_cgen_opval_h_ccr, fields->f_ccrn, 0|(1<<CGEN_OPERAND_VIRTUAL));
774 break;
775 case MEP_OPERAND_CDISP10 :
776 print_normal (cd, info, fields->f_cdisp10, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
777 break;
778 case MEP_OPERAND_CDISP10A2 :
779 print_normal (cd, info, fields->f_cdisp10, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
780 break;
781 case MEP_OPERAND_CDISP10A4 :
782 print_normal (cd, info, fields->f_cdisp10, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
783 break;
784 case MEP_OPERAND_CDISP10A8 :
785 print_normal (cd, info, fields->f_cdisp10, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
786 break;
787 case MEP_OPERAND_CDISP12 :
788 print_normal (cd, info, fields->f_12s20, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
789 break;
790 case MEP_OPERAND_CIMM4 :
791 print_normal (cd, info, fields->f_rn, 0, pc, length);
792 break;
793 case MEP_OPERAND_CIMM5 :
794 print_normal (cd, info, fields->f_5u24, 0, pc, length);
795 break;
796 case MEP_OPERAND_CODE16 :
797 print_normal (cd, info, fields->f_16u16, 0, pc, length);
798 break;
799 case MEP_OPERAND_CODE24 :
800 print_normal (cd, info, fields->f_24u4n, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
801 break;
802 case MEP_OPERAND_CP_FLAG :
803 print_keyword (cd, info, & mep_cgen_opval_h_ccr, 0, 0);
804 break;
805 case MEP_OPERAND_CRN :
806 print_keyword (cd, info, & mep_cgen_opval_h_cr, fields->f_crn, 0);
807 break;
808 case MEP_OPERAND_CRN64 :
809 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_crn, 0);
810 break;
811 case MEP_OPERAND_CRNX :
812 print_keyword (cd, info, & mep_cgen_opval_h_cr, fields->f_crnx, 0|(1<<CGEN_OPERAND_VIRTUAL));
813 break;
814 case MEP_OPERAND_CRNX64 :
815 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_crnx, 0|(1<<CGEN_OPERAND_VIRTUAL));
816 break;
817 case MEP_OPERAND_CROC :
818 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_5u7, 0);
819 break;
820 case MEP_OPERAND_CROP :
821 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_5u23, 0);
822 break;
823 case MEP_OPERAND_CRPC :
824 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_5u26, 0);
825 break;
826 case MEP_OPERAND_CRPP :
827 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_5u18, 0);
828 break;
829 case MEP_OPERAND_CRQC :
830 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_5u21, 0);
831 break;
832 case MEP_OPERAND_CRQP :
833 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_5u13, 0);
834 break;
835 case MEP_OPERAND_CSRN :
836 print_keyword (cd, info, & mep_cgen_opval_h_csr, fields->f_csrn, 0|(1<<CGEN_OPERAND_VIRTUAL));
837 break;
838 case MEP_OPERAND_CSRN_IDX :
839 print_normal (cd, info, fields->f_csrn, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
840 break;
841 case MEP_OPERAND_DBG :
842 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
843 break;
844 case MEP_OPERAND_DEPC :
845 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
846 break;
847 case MEP_OPERAND_EPC :
848 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
849 break;
850 case MEP_OPERAND_EXC :
851 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
852 break;
853 case MEP_OPERAND_HI :
854 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
855 break;
856 case MEP_OPERAND_IMM16P0 :
857 print_normal (cd, info, fields->f_ivc2_imm16p0, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
858 break;
859 case MEP_OPERAND_IMM3P12 :
860 print_normal (cd, info, fields->f_ivc2_3u12, 0, pc, length);
861 break;
862 case MEP_OPERAND_IMM3P25 :
863 print_normal (cd, info, fields->f_ivc2_3u25, 0, pc, length);
864 break;
865 case MEP_OPERAND_IMM3P4 :
866 print_normal (cd, info, fields->f_ivc2_3u4, 0, pc, length);
867 break;
868 case MEP_OPERAND_IMM3P5 :
869 print_normal (cd, info, fields->f_ivc2_3u5, 0, pc, length);
870 break;
871 case MEP_OPERAND_IMM3P9 :
872 print_normal (cd, info, fields->f_ivc2_3u9, 0, pc, length);
873 break;
874 case MEP_OPERAND_IMM4P10 :
875 print_normal (cd, info, fields->f_ivc2_4u10, 0, pc, length);
876 break;
877 case MEP_OPERAND_IMM4P4 :
878 print_normal (cd, info, fields->f_ivc2_4u4, 0, pc, length);
879 break;
880 case MEP_OPERAND_IMM4P8 :
881 print_normal (cd, info, fields->f_ivc2_4u8, 0, pc, length);
882 break;
883 case MEP_OPERAND_IMM5P23 :
884 print_normal (cd, info, fields->f_ivc2_5u23, 0, pc, length);
885 break;
886 case MEP_OPERAND_IMM5P3 :
887 print_normal (cd, info, fields->f_ivc2_5u3, 0, pc, length);
888 break;
889 case MEP_OPERAND_IMM5P7 :
890 print_normal (cd, info, fields->f_ivc2_5u7, 0, pc, length);
891 break;
892 case MEP_OPERAND_IMM5P8 :
893 print_normal (cd, info, fields->f_ivc2_5u8, 0, pc, length);
894 break;
895 case MEP_OPERAND_IMM6P2 :
896 print_normal (cd, info, fields->f_ivc2_6u2, 0, pc, length);
897 break;
898 case MEP_OPERAND_IMM6P6 :
899 print_normal (cd, info, fields->f_ivc2_6u6, 0, pc, length);
900 break;
901 case MEP_OPERAND_IMM8P0 :
902 print_normal (cd, info, fields->f_ivc2_8u0, 0, pc, length);
903 break;
904 case MEP_OPERAND_IMM8P20 :
905 print_normal (cd, info, fields->f_ivc2_8u20, 0, pc, length);
906 break;
907 case MEP_OPERAND_IMM8P4 :
908 print_normal (cd, info, fields->f_ivc2_8u4, 0, pc, length);
909 break;
910 case MEP_OPERAND_IVC_X_0_2 :
911 print_normal (cd, info, fields->f_ivc2_2u0, 0, pc, length);
912 break;
913 case MEP_OPERAND_IVC_X_0_3 :
914 print_normal (cd, info, fields->f_ivc2_3u0, 0, pc, length);
915 break;
916 case MEP_OPERAND_IVC_X_0_4 :
917 print_normal (cd, info, fields->f_ivc2_4u0, 0, pc, length);
918 break;
919 case MEP_OPERAND_IVC_X_0_5 :
920 print_normal (cd, info, fields->f_ivc2_5u0, 0, pc, length);
921 break;
922 case MEP_OPERAND_IVC_X_6_1 :
923 print_normal (cd, info, fields->f_ivc2_1u6, 0, pc, length);
924 break;
925 case MEP_OPERAND_IVC_X_6_2 :
926 print_normal (cd, info, fields->f_ivc2_2u6, 0, pc, length);
927 break;
928 case MEP_OPERAND_IVC_X_6_3 :
929 print_normal (cd, info, fields->f_ivc2_3u6, 0, pc, length);
930 break;
931 case MEP_OPERAND_IVC2CCRN :
932 print_keyword (cd, info, & mep_cgen_opval_h_ccr, fields->f_ivc2_ccrn, 0|(1<<CGEN_OPERAND_VIRTUAL));
933 break;
934 case MEP_OPERAND_IVC2CRN :
935 print_keyword (cd, info, & mep_cgen_opval_h_cr64, fields->f_ivc2_crnx, 0|(1<<CGEN_OPERAND_VIRTUAL));
936 break;
937 case MEP_OPERAND_IVC2RM :
938 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_ivc2_crm, 0);
939 break;
940 case MEP_OPERAND_LO :
941 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
942 break;
943 case MEP_OPERAND_LP :
944 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
945 break;
946 case MEP_OPERAND_MB0 :
947 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
948 break;
949 case MEP_OPERAND_MB1 :
950 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
951 break;
952 case MEP_OPERAND_ME0 :
953 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
954 break;
955 case MEP_OPERAND_ME1 :
956 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
957 break;
958 case MEP_OPERAND_NPC :
959 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
960 break;
961 case MEP_OPERAND_OPT :
962 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
963 break;
964 case MEP_OPERAND_PCABS24A2 :
965 print_address (cd, info, fields->f_24u5a2n, 0|(1<<CGEN_OPERAND_ABS_ADDR)|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
966 break;
967 case MEP_OPERAND_PCREL12A2 :
968 print_address (cd, info, fields->f_12s4a2, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
969 break;
970 case MEP_OPERAND_PCREL17A2 :
971 print_address (cd, info, fields->f_17s16a2, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
972 break;
973 case MEP_OPERAND_PCREL24A2 :
974 print_address (cd, info, fields->f_24s5a2n, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_PCREL_ADDR)|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
975 break;
976 case MEP_OPERAND_PCREL8A2 :
977 print_address (cd, info, fields->f_8s8a2, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELAX)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
978 break;
979 case MEP_OPERAND_PSW :
980 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
981 break;
982 case MEP_OPERAND_R0 :
983 print_keyword (cd, info, & mep_cgen_opval_h_gpr, 0, 0);
984 break;
985 case MEP_OPERAND_R1 :
986 print_keyword (cd, info, & mep_cgen_opval_h_gpr, 0, 0);
987 break;
988 case MEP_OPERAND_RL :
989 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rl, 0);
990 break;
991 case MEP_OPERAND_RL5 :
992 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rl5, 0);
993 break;
994 case MEP_OPERAND_RM :
995 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rm, 0);
996 break;
997 case MEP_OPERAND_RMA :
998 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rm, 0);
999 break;
1000 case MEP_OPERAND_RN :
1001 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1002 break;
1003 case MEP_OPERAND_RN3 :
1004 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1005 break;
1006 case MEP_OPERAND_RN3C :
1007 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1008 break;
1009 case MEP_OPERAND_RN3L :
1010 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1011 break;
1012 case MEP_OPERAND_RN3S :
1013 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1014 break;
1015 case MEP_OPERAND_RN3UC :
1016 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1017 break;
1018 case MEP_OPERAND_RN3UL :
1019 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1020 break;
1021 case MEP_OPERAND_RN3US :
1022 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn3, 0);
1023 break;
1024 case MEP_OPERAND_RNC :
1025 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1026 break;
1027 case MEP_OPERAND_RNL :
1028 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1029 break;
1030 case MEP_OPERAND_RNS :
1031 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1032 break;
1033 case MEP_OPERAND_RNUC :
1034 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1035 break;
1036 case MEP_OPERAND_RNUL :
1037 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1038 break;
1039 case MEP_OPERAND_RNUS :
1040 print_keyword (cd, info, & mep_cgen_opval_h_gpr, fields->f_rn, 0);
1041 break;
1042 case MEP_OPERAND_SAR :
1043 print_keyword (cd, info, & mep_cgen_opval_h_csr, 0, 0);
1044 break;
1045 case MEP_OPERAND_SDISP16 :
1046 print_normal (cd, info, fields->f_16s16, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1047 break;
1048 case MEP_OPERAND_SIMM16 :
1049 print_normal (cd, info, fields->f_16s16, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1050 break;
1051 case MEP_OPERAND_SIMM16P0 :
1052 print_normal (cd, info, fields->f_ivc2_simm16p0, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
1053 break;
1054 case MEP_OPERAND_SIMM6 :
1055 print_normal (cd, info, fields->f_6s8, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1056 break;
1057 case MEP_OPERAND_SIMM8 :
1058 print_normal (cd, info, fields->f_8s8, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELOC_IMPLIES_OVERFLOW), pc, length);
1059 break;
1060 case MEP_OPERAND_SIMM8P0 :
1061 print_normal (cd, info, fields->f_ivc2_8s0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1062 break;
1063 case MEP_OPERAND_SIMM8P4 :
1064 print_normal (cd, info, fields->f_ivc2_8s4, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1065 break;
1066 case MEP_OPERAND_SP :
1067 print_keyword (cd, info, & mep_cgen_opval_h_gpr, 0, 0);
1068 break;
1069 case MEP_OPERAND_SPR :
1070 print_spreg (cd, info, & mep_cgen_opval_h_gpr, 0, 0);
1071 break;
1072 case MEP_OPERAND_TP :
1073 print_keyword (cd, info, & mep_cgen_opval_h_gpr, 0, 0);
1074 break;
1075 case MEP_OPERAND_TPR :
1076 print_tpreg (cd, info, & mep_cgen_opval_h_gpr, 0, 0);
1077 break;
1078 case MEP_OPERAND_UDISP2 :
1079 print_normal (cd, info, fields->f_2u6, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1080 break;
1081 case MEP_OPERAND_UDISP7 :
1082 print_normal (cd, info, fields->f_7u9, 0, pc, length);
1083 break;
1084 case MEP_OPERAND_UDISP7A2 :
1085 print_normal (cd, info, fields->f_7u9a2, 0, pc, length);
1086 break;
1087 case MEP_OPERAND_UDISP7A4 :
1088 print_normal (cd, info, fields->f_7u9a4, 0, pc, length);
1089 break;
1090 case MEP_OPERAND_UIMM16 :
1091 print_normal (cd, info, fields->f_16u16, 0, pc, length);
1092 break;
1093 case MEP_OPERAND_UIMM2 :
1094 print_normal (cd, info, fields->f_2u10, 0, pc, length);
1095 break;
1096 case MEP_OPERAND_UIMM24 :
1097 print_normal (cd, info, fields->f_24u8n, 0|(1<<CGEN_OPERAND_VIRTUAL), pc, length);
1098 break;
1099 case MEP_OPERAND_UIMM3 :
1100 print_normal (cd, info, fields->f_3u5, 0, pc, length);
1101 break;
1102 case MEP_OPERAND_UIMM4 :
1103 print_normal (cd, info, fields->f_4u8, 0, pc, length);
1104 break;
1105 case MEP_OPERAND_UIMM5 :
1106 print_normal (cd, info, fields->f_5u8, 0, pc, length);
1107 break;
1108 case MEP_OPERAND_UIMM7A4 :
1109 print_normal (cd, info, fields->f_7u9a4, 0, pc, length);
1110 break;
1111 case MEP_OPERAND_ZERO :
1112 print_normal (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
1113 break;
1114
1115 default :
1116 /* xgettext:c-format */
1117 fprintf (stderr, _("Unrecognized field %d while printing insn.\n"),
1118 opindex);
1119 abort ();
1120 }
1121 }
1122
1123 cgen_print_fn * const mep_cgen_print_handlers[] =
1124 {
1125 print_insn_normal,
1126 };
1127
1128
1129 void
1130 mep_cgen_init_dis (CGEN_CPU_DESC cd)
1131 {
1132 mep_cgen_init_opcode_table (cd);
1133 mep_cgen_init_ibld_table (cd);
1134 cd->print_handlers = & mep_cgen_print_handlers[0];
1135 cd->print_operand = mep_cgen_print_operand;
1136 }
1137
1138 \f
1139 /* Default print handler. */
1140
1141 static void
1142 print_normal (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
1143 void *dis_info,
1144 long value,
1145 unsigned int attrs,
1146 bfd_vma pc ATTRIBUTE_UNUSED,
1147 int length ATTRIBUTE_UNUSED)
1148 {
1149 disassemble_info *info = (disassemble_info *) dis_info;
1150
1151 #ifdef CGEN_PRINT_NORMAL
1152 CGEN_PRINT_NORMAL (cd, info, value, attrs, pc, length);
1153 #endif
1154
1155 /* Print the operand as directed by the attributes. */
1156 if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SEM_ONLY))
1157 ; /* nothing to do */
1158 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SIGNED))
1159 (*info->fprintf_func) (info->stream, "%ld", value);
1160 else
1161 (*info->fprintf_func) (info->stream, "0x%lx", value);
1162 }
1163
1164 /* Default address handler. */
1165
1166 static void
1167 print_address (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
1168 void *dis_info,
1169 bfd_vma value,
1170 unsigned int attrs,
1171 bfd_vma pc ATTRIBUTE_UNUSED,
1172 int length ATTRIBUTE_UNUSED)
1173 {
1174 disassemble_info *info = (disassemble_info *) dis_info;
1175
1176 #ifdef CGEN_PRINT_ADDRESS
1177 CGEN_PRINT_ADDRESS (cd, info, value, attrs, pc, length);
1178 #endif
1179
1180 /* Print the operand as directed by the attributes. */
1181 if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SEM_ONLY))
1182 ; /* Nothing to do. */
1183 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_PCREL_ADDR))
1184 (*info->print_address_func) (value, info);
1185 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_ABS_ADDR))
1186 (*info->print_address_func) (value, info);
1187 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SIGNED))
1188 (*info->fprintf_func) (info->stream, "%ld", (long) value);
1189 else
1190 (*info->fprintf_func) (info->stream, "0x%lx", (long) value);
1191 }
1192
1193 /* Keyword print handler. */
1194
1195 static void
1196 print_keyword (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
1197 void *dis_info,
1198 CGEN_KEYWORD *keyword_table,
1199 long value,
1200 unsigned int attrs ATTRIBUTE_UNUSED)
1201 {
1202 disassemble_info *info = (disassemble_info *) dis_info;
1203 const CGEN_KEYWORD_ENTRY *ke;
1204
1205 ke = cgen_keyword_lookup_value (keyword_table, value);
1206 if (ke != NULL)
1207 (*info->fprintf_func) (info->stream, "%s", ke->name);
1208 else
1209 (*info->fprintf_func) (info->stream, "???");
1210 }
1211 \f
1212 /* Default insn printer.
1213
1214 DIS_INFO is defined as `void *' so the disassembler needn't know anything
1215 about disassemble_info. */
1216
1217 static void
1218 print_insn_normal (CGEN_CPU_DESC cd,
1219 void *dis_info,
1220 const CGEN_INSN *insn,
1221 CGEN_FIELDS *fields,
1222 bfd_vma pc,
1223 int length)
1224 {
1225 const CGEN_SYNTAX *syntax = CGEN_INSN_SYNTAX (insn);
1226 disassemble_info *info = (disassemble_info *) dis_info;
1227 const CGEN_SYNTAX_CHAR_TYPE *syn;
1228
1229 CGEN_INIT_PRINT (cd);
1230
1231 for (syn = CGEN_SYNTAX_STRING (syntax); *syn; ++syn)
1232 {
1233 if (CGEN_SYNTAX_MNEMONIC_P (*syn))
1234 {
1235 (*info->fprintf_func) (info->stream, "%s", CGEN_INSN_MNEMONIC (insn));
1236 continue;
1237 }
1238 if (CGEN_SYNTAX_CHAR_P (*syn))
1239 {
1240 (*info->fprintf_func) (info->stream, "%c", CGEN_SYNTAX_CHAR (*syn));
1241 continue;
1242 }
1243
1244 /* We have an operand. */
1245 mep_cgen_print_operand (cd, CGEN_SYNTAX_FIELD (*syn), info,
1246 fields, CGEN_INSN_ATTRS (insn), pc, length);
1247 }
1248 }
1249 \f
1250 /* Subroutine of print_insn. Reads an insn into the given buffers and updates
1251 the extract info.
1252 Returns 0 if all is well, non-zero otherwise. */
1253
1254 static int
1255 read_insn (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
1256 bfd_vma pc,
1257 disassemble_info *info,
1258 bfd_byte *buf,
1259 int buflen,
1260 CGEN_EXTRACT_INFO *ex_info,
1261 unsigned long *insn_value)
1262 {
1263 int status = (*info->read_memory_func) (pc, buf, buflen, info);
1264
1265 if (status != 0)
1266 {
1267 (*info->memory_error_func) (status, pc, info);
1268 return -1;
1269 }
1270
1271 ex_info->dis_info = info;
1272 ex_info->valid = (1 << buflen) - 1;
1273 ex_info->insn_bytes = buf;
1274
1275 *insn_value = bfd_get_bits (buf, buflen * 8, info->endian == BFD_ENDIAN_BIG);
1276 return 0;
1277 }
1278
1279 /* Utility to print an insn.
1280 BUF is the base part of the insn, target byte order, BUFLEN bytes long.
1281 The result is the size of the insn in bytes or zero for an unknown insn
1282 or -1 if an error occurs fetching data (memory_error_func will have
1283 been called). */
1284
1285 static int
1286 print_insn (CGEN_CPU_DESC cd,
1287 bfd_vma pc,
1288 disassemble_info *info,
1289 bfd_byte *buf,
1290 unsigned int buflen)
1291 {
1292 CGEN_INSN_INT insn_value;
1293 const CGEN_INSN_LIST *insn_list;
1294 CGEN_EXTRACT_INFO ex_info;
1295 int basesize;
1296
1297 /* Extract base part of instruction, just in case CGEN_DIS_* uses it. */
1298 basesize = cd->base_insn_bitsize < buflen * 8 ?
1299 cd->base_insn_bitsize : buflen * 8;
1300 insn_value = cgen_get_insn_value (cd, buf, basesize);
1301
1302
1303 /* Fill in ex_info fields like read_insn would. Don't actually call
1304 read_insn, since the incoming buffer is already read (and possibly
1305 modified a la m32r). */
1306 ex_info.valid = (1 << buflen) - 1;
1307 ex_info.dis_info = info;
1308 ex_info.insn_bytes = buf;
1309
1310 /* The instructions are stored in hash lists.
1311 Pick the first one and keep trying until we find the right one. */
1312
1313 insn_list = CGEN_DIS_LOOKUP_INSN (cd, (char *) buf, insn_value);
1314 while (insn_list != NULL)
1315 {
1316 const CGEN_INSN *insn = insn_list->insn;
1317 CGEN_FIELDS fields;
1318 int length;
1319 unsigned long insn_value_cropped;
1320
1321 #ifdef CGEN_VALIDATE_INSN_SUPPORTED
1322 /* Not needed as insn shouldn't be in hash lists if not supported. */
1323 /* Supported by this cpu? */
1324 if (! mep_cgen_insn_supported (cd, insn))
1325 {
1326 insn_list = CGEN_DIS_NEXT_INSN (insn_list);
1327 continue;
1328 }
1329 #endif
1330
1331 /* Basic bit mask must be correct. */
1332 /* ??? May wish to allow target to defer this check until the extract
1333 handler. */
1334
1335 /* Base size may exceed this instruction's size. Extract the
1336 relevant part from the buffer. */
1337 if ((unsigned) (CGEN_INSN_BITSIZE (insn) / 8) < buflen &&
1338 (unsigned) (CGEN_INSN_BITSIZE (insn) / 8) <= sizeof (unsigned long))
1339 insn_value_cropped = bfd_get_bits (buf, CGEN_INSN_BITSIZE (insn),
1340 info->endian == BFD_ENDIAN_BIG);
1341 else
1342 insn_value_cropped = insn_value;
1343
1344 if ((insn_value_cropped & CGEN_INSN_BASE_MASK (insn))
1345 == CGEN_INSN_BASE_VALUE (insn))
1346 {
1347 /* Printing is handled in two passes. The first pass parses the
1348 machine insn and extracts the fields. The second pass prints
1349 them. */
1350
1351 /* Make sure the entire insn is loaded into insn_value, if it
1352 can fit. */
1353 if (((unsigned) CGEN_INSN_BITSIZE (insn) > cd->base_insn_bitsize) &&
1354 (unsigned) (CGEN_INSN_BITSIZE (insn) / 8) <= sizeof (unsigned long))
1355 {
1356 unsigned long full_insn_value;
1357 int rc = read_insn (cd, pc, info, buf,
1358 CGEN_INSN_BITSIZE (insn) / 8,
1359 & ex_info, & full_insn_value);
1360 if (rc != 0)
1361 return rc;
1362 length = CGEN_EXTRACT_FN (cd, insn)
1363 (cd, insn, &ex_info, full_insn_value, &fields, pc);
1364 }
1365 else
1366 length = CGEN_EXTRACT_FN (cd, insn)
1367 (cd, insn, &ex_info, insn_value_cropped, &fields, pc);
1368
1369 /* Length < 0 -> error. */
1370 if (length < 0)
1371 return length;
1372 if (length > 0)
1373 {
1374 CGEN_PRINT_FN (cd, insn) (cd, info, insn, &fields, pc, length);
1375 /* Length is in bits, result is in bytes. */
1376 return length / 8;
1377 }
1378 }
1379
1380 insn_list = CGEN_DIS_NEXT_INSN (insn_list);
1381 }
1382
1383 return 0;
1384 }
1385
1386 /* Default value for CGEN_PRINT_INSN.
1387 The result is the size of the insn in bytes or zero for an unknown insn
1388 or -1 if an error occured fetching bytes. */
1389
1390 #ifndef CGEN_PRINT_INSN
1391 #define CGEN_PRINT_INSN default_print_insn
1392 #endif
1393
1394 static int
1395 default_print_insn (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
1396 {
1397 bfd_byte buf[CGEN_MAX_INSN_SIZE];
1398 int buflen;
1399 int status;
1400
1401 /* Attempt to read the base part of the insn. */
1402 buflen = cd->base_insn_bitsize / 8;
1403 status = (*info->read_memory_func) (pc, buf, buflen, info);
1404
1405 /* Try again with the minimum part, if min < base. */
1406 if (status != 0 && (cd->min_insn_bitsize < cd->base_insn_bitsize))
1407 {
1408 buflen = cd->min_insn_bitsize / 8;
1409 status = (*info->read_memory_func) (pc, buf, buflen, info);
1410 }
1411
1412 if (status != 0)
1413 {
1414 (*info->memory_error_func) (status, pc, info);
1415 return -1;
1416 }
1417
1418 return print_insn (cd, pc, info, buf, buflen);
1419 }
1420
1421 /* Main entry point.
1422 Print one instruction from PC on INFO->STREAM.
1423 Return the size of the instruction (in bytes). */
1424
1425 typedef struct cpu_desc_list
1426 {
1427 struct cpu_desc_list *next;
1428 CGEN_BITSET *isa;
1429 int mach;
1430 int endian;
1431 CGEN_CPU_DESC cd;
1432 } cpu_desc_list;
1433
1434 int
1435 print_insn_mep (bfd_vma pc, disassemble_info *info)
1436 {
1437 static cpu_desc_list *cd_list = 0;
1438 cpu_desc_list *cl = 0;
1439 static CGEN_CPU_DESC cd = 0;
1440 static CGEN_BITSET *prev_isa;
1441 static int prev_mach;
1442 static int prev_endian;
1443 int length;
1444 CGEN_BITSET *isa;
1445 int mach;
1446 int endian = (info->endian == BFD_ENDIAN_BIG
1447 ? CGEN_ENDIAN_BIG
1448 : CGEN_ENDIAN_LITTLE);
1449 enum bfd_architecture arch;
1450
1451 /* ??? gdb will set mach but leave the architecture as "unknown" */
1452 #ifndef CGEN_BFD_ARCH
1453 #define CGEN_BFD_ARCH bfd_arch_mep
1454 #endif
1455 arch = info->arch;
1456 if (arch == bfd_arch_unknown)
1457 arch = CGEN_BFD_ARCH;
1458
1459 /* There's no standard way to compute the machine or isa number
1460 so we leave it to the target. */
1461 #ifdef CGEN_COMPUTE_MACH
1462 mach = CGEN_COMPUTE_MACH (info);
1463 #else
1464 mach = info->mach;
1465 #endif
1466
1467 #ifdef CGEN_COMPUTE_ISA
1468 {
1469 static CGEN_BITSET *permanent_isa;
1470
1471 if (!permanent_isa)
1472 permanent_isa = cgen_bitset_create (MAX_ISAS);
1473 isa = permanent_isa;
1474 cgen_bitset_clear (isa);
1475 cgen_bitset_add (isa, CGEN_COMPUTE_ISA (info));
1476 }
1477 #else
1478 isa = info->insn_sets;
1479 #endif
1480
1481 /* If we've switched cpu's, try to find a handle we've used before */
1482 if (cd
1483 && (cgen_bitset_compare (isa, prev_isa) != 0
1484 || mach != prev_mach
1485 || endian != prev_endian))
1486 {
1487 cd = 0;
1488 for (cl = cd_list; cl; cl = cl->next)
1489 {
1490 if (cgen_bitset_compare (cl->isa, isa) == 0 &&
1491 cl->mach == mach &&
1492 cl->endian == endian)
1493 {
1494 cd = cl->cd;
1495 prev_isa = cd->isas;
1496 break;
1497 }
1498 }
1499 }
1500
1501 /* If we haven't initialized yet, initialize the opcode table. */
1502 if (! cd)
1503 {
1504 const bfd_arch_info_type *arch_type = bfd_lookup_arch (arch, mach);
1505 const char *mach_name;
1506
1507 if (!arch_type)
1508 abort ();
1509 mach_name = arch_type->printable_name;
1510
1511 prev_isa = cgen_bitset_copy (isa);
1512 prev_mach = mach;
1513 prev_endian = endian;
1514 cd = mep_cgen_cpu_open (CGEN_CPU_OPEN_ISAS, prev_isa,
1515 CGEN_CPU_OPEN_BFDMACH, mach_name,
1516 CGEN_CPU_OPEN_ENDIAN, prev_endian,
1517 CGEN_CPU_OPEN_END);
1518 if (!cd)
1519 abort ();
1520
1521 /* Save this away for future reference. */
1522 cl = xmalloc (sizeof (struct cpu_desc_list));
1523 cl->cd = cd;
1524 cl->isa = prev_isa;
1525 cl->mach = mach;
1526 cl->endian = endian;
1527 cl->next = cd_list;
1528 cd_list = cl;
1529
1530 mep_cgen_init_dis (cd);
1531 }
1532
1533 /* We try to have as much common code as possible.
1534 But at this point some targets need to take over. */
1535 /* ??? Some targets may need a hook elsewhere. Try to avoid this,
1536 but if not possible try to move this hook elsewhere rather than
1537 have two hooks. */
1538 length = CGEN_PRINT_INSN (cd, pc, info);
1539 if (length > 0)
1540 return length;
1541 if (length < 0)
1542 return -1;
1543
1544 (*info->fprintf_func) (info->stream, UNKNOWN_INSN_MSG);
1545 return cd->default_insn_bitsize / 8;
1546 }
This page took 0.08492 seconds and 5 git commands to generate.