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 "xc16x-desc.h"
36 #include "xc16x-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 /* Print an operand with a "." prefix.
63 NOTE: This prints the operand in hex.
64 ??? This exists to maintain disassembler compatibility with previous
65 versions. Ideally we'd print the "." in print_dot. */
66
67 static void
print_with_dot_prefix(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value,unsigned attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)68 print_with_dot_prefix (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
69 void * dis_info,
70 long value,
71 unsigned attrs ATTRIBUTE_UNUSED,
72 bfd_vma pc ATTRIBUTE_UNUSED,
73 int length ATTRIBUTE_UNUSED)
74 {
75 disassemble_info *info = (disassemble_info *) dis_info;
76
77 info->fprintf_func (info->stream, ".");
78 info->fprintf_func (info->stream, "0x%lx", value);
79 }
80
81 /* Print an operand with a "#pof:" prefix.
82 NOTE: This prints the operand as an address.
83 ??? This exists to maintain disassembler compatibility with previous
84 versions. Ideally we'd print "#pof:" in print_pof. */
85
86 static void
print_with_pof_prefix(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,bfd_vma value,unsigned attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)87 print_with_pof_prefix (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
88 void * dis_info,
89 bfd_vma value,
90 unsigned attrs ATTRIBUTE_UNUSED,
91 bfd_vma pc ATTRIBUTE_UNUSED,
92 int length ATTRIBUTE_UNUSED)
93 {
94 disassemble_info *info = (disassemble_info *) dis_info;
95
96 info->fprintf_func (info->stream, "#pof:");
97 info->fprintf_func (info->stream, "0x%lx", (long) value);
98 }
99
100 /* Print an operand with a "#pag:" prefix.
101 NOTE: This prints the operand in hex.
102 ??? This exists to maintain disassembler compatibility with previous
103 versions. Ideally we'd print "#pag:" in print_pag. */
104
105 static void
print_with_pag_prefix(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value,unsigned attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)106 print_with_pag_prefix (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
107 void * dis_info,
108 long value,
109 unsigned attrs ATTRIBUTE_UNUSED,
110 bfd_vma pc ATTRIBUTE_UNUSED,
111 int length ATTRIBUTE_UNUSED)
112 {
113 disassemble_info *info = (disassemble_info *) dis_info;
114
115 info->fprintf_func (info->stream, "#pag:");
116 info->fprintf_func (info->stream, "0x%lx", value);
117 }
118
119 /* Print a 'pof:' prefix to an operand. */
120
121 static void
print_pof(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info ATTRIBUTE_UNUSED,long value ATTRIBUTE_UNUSED,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)122 print_pof (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
123 void * dis_info ATTRIBUTE_UNUSED,
124 long value ATTRIBUTE_UNUSED,
125 unsigned int attrs ATTRIBUTE_UNUSED,
126 bfd_vma pc ATTRIBUTE_UNUSED,
127 int length ATTRIBUTE_UNUSED)
128 {
129 }
130
131 /* Print a 'pag:' prefix to an operand. */
132
133 static void
print_pag(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info ATTRIBUTE_UNUSED,long value ATTRIBUTE_UNUSED,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)134 print_pag (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
135 void * dis_info ATTRIBUTE_UNUSED,
136 long value ATTRIBUTE_UNUSED,
137 unsigned int attrs ATTRIBUTE_UNUSED,
138 bfd_vma pc ATTRIBUTE_UNUSED,
139 int length ATTRIBUTE_UNUSED)
140 {
141 }
142
143 /* Print a 'sof:' prefix to an operand. */
144
145 static void
print_sof(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value ATTRIBUTE_UNUSED,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)146 print_sof (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
147 void * dis_info,
148 long value ATTRIBUTE_UNUSED,
149 unsigned int attrs ATTRIBUTE_UNUSED,
150 bfd_vma pc ATTRIBUTE_UNUSED,
151 int length ATTRIBUTE_UNUSED)
152 {
153 disassemble_info *info = (disassemble_info *) dis_info;
154
155 info->fprintf_func (info->stream, "sof:");
156 }
157
158 /* Print a 'seg:' prefix to an operand. */
159
160 static void
print_seg(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value ATTRIBUTE_UNUSED,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)161 print_seg (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
162 void * dis_info,
163 long value ATTRIBUTE_UNUSED,
164 unsigned int attrs ATTRIBUTE_UNUSED,
165 bfd_vma pc ATTRIBUTE_UNUSED,
166 int length ATTRIBUTE_UNUSED)
167 {
168 disassemble_info *info = (disassemble_info *) dis_info;
169
170 info->fprintf_func (info->stream, "seg:");
171 }
172
173 /* Print a '#' prefix to an operand. */
174
175 static void
print_hash(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,long value ATTRIBUTE_UNUSED,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)176 print_hash (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
177 void * dis_info,
178 long value ATTRIBUTE_UNUSED,
179 unsigned int attrs ATTRIBUTE_UNUSED,
180 bfd_vma pc ATTRIBUTE_UNUSED,
181 int length ATTRIBUTE_UNUSED)
182 {
183 disassemble_info *info = (disassemble_info *) dis_info;
184
185 info->fprintf_func (info->stream, "#");
186 }
187
188 /* Print a '.' prefix to an operand. */
189
190 static void
print_dot(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info ATTRIBUTE_UNUSED,long value ATTRIBUTE_UNUSED,unsigned int attrs ATTRIBUTE_UNUSED,bfd_vma pc ATTRIBUTE_UNUSED,int length ATTRIBUTE_UNUSED)191 print_dot (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
192 void * dis_info ATTRIBUTE_UNUSED,
193 long value ATTRIBUTE_UNUSED,
194 unsigned int attrs ATTRIBUTE_UNUSED,
195 bfd_vma pc ATTRIBUTE_UNUSED,
196 int length ATTRIBUTE_UNUSED)
197 {
198 }
199
200 /* -- */
201
202 void xc16x_cgen_print_operand
203 (CGEN_CPU_DESC, int, PTR, CGEN_FIELDS *, void const *, bfd_vma, int);
204
205 /* Main entry point for printing operands.
206 XINFO is a `void *' and not a `disassemble_info *' to not put a requirement
207 of dis-asm.h on cgen.h.
208
209 This function is basically just a big switch statement. Earlier versions
210 used tables to look up the function to use, but
211 - if the table contains both assembler and disassembler functions then
212 the disassembler contains much of the assembler and vice-versa,
213 - there's a lot of inlining possibilities as things grow,
214 - using a switch statement avoids the function call overhead.
215
216 This function could be moved into `print_insn_normal', but keeping it
217 separate makes clear the interface between `print_insn_normal' and each of
218 the handlers. */
219
220 void
xc16x_cgen_print_operand(CGEN_CPU_DESC cd,int opindex,void * xinfo,CGEN_FIELDS * fields,void const * attrs ATTRIBUTE_UNUSED,bfd_vma pc,int length)221 xc16x_cgen_print_operand (CGEN_CPU_DESC cd,
222 int opindex,
223 void * xinfo,
224 CGEN_FIELDS *fields,
225 void const *attrs ATTRIBUTE_UNUSED,
226 bfd_vma pc,
227 int length)
228 {
229 disassemble_info *info = (disassemble_info *) xinfo;
230
231 switch (opindex)
232 {
233 case XC16X_OPERAND_REGNAM :
234 print_keyword (cd, info, & xc16x_cgen_opval_psw_names, fields->f_reg8, 0);
235 break;
236 case XC16X_OPERAND_BIT01 :
237 print_normal (cd, info, fields->f_op_1bit, 0, pc, length);
238 break;
239 case XC16X_OPERAND_BIT1 :
240 print_normal (cd, info, fields->f_op_bit1, 0, pc, length);
241 break;
242 case XC16X_OPERAND_BIT2 :
243 print_normal (cd, info, fields->f_op_bit2, 0, pc, length);
244 break;
245 case XC16X_OPERAND_BIT4 :
246 print_normal (cd, info, fields->f_op_bit4, 0, pc, length);
247 break;
248 case XC16X_OPERAND_BIT8 :
249 print_normal (cd, info, fields->f_op_bit8, 0, pc, length);
250 break;
251 case XC16X_OPERAND_BITONE :
252 print_normal (cd, info, fields->f_op_onebit, 0, pc, length);
253 break;
254 case XC16X_OPERAND_CADDR :
255 print_address (cd, info, fields->f_offset16, 0|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_ABS_ADDR), pc, length);
256 break;
257 case XC16X_OPERAND_COND :
258 print_keyword (cd, info, & xc16x_cgen_opval_conditioncode_names, fields->f_condcode, 0);
259 break;
260 case XC16X_OPERAND_DATA8 :
261 print_normal (cd, info, fields->f_data8, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
262 break;
263 case XC16X_OPERAND_DATAHI8 :
264 print_normal (cd, info, fields->f_datahi8, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
265 break;
266 case XC16X_OPERAND_DOT :
267 print_dot (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
268 break;
269 case XC16X_OPERAND_DR :
270 print_keyword (cd, info, & xc16x_cgen_opval_gr_names, fields->f_r1, 0);
271 break;
272 case XC16X_OPERAND_DRB :
273 print_keyword (cd, info, & xc16x_cgen_opval_grb_names, fields->f_r1, 0);
274 break;
275 case XC16X_OPERAND_DRI :
276 print_keyword (cd, info, & xc16x_cgen_opval_gr_names, fields->f_r4, 0);
277 break;
278 case XC16X_OPERAND_EXTCOND :
279 print_keyword (cd, info, & xc16x_cgen_opval_extconditioncode_names, fields->f_extccode, 0);
280 break;
281 case XC16X_OPERAND_GENREG :
282 print_keyword (cd, info, & xc16x_cgen_opval_r8_names, fields->f_regb8, 0);
283 break;
284 case XC16X_OPERAND_HASH :
285 print_hash (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
286 break;
287 case XC16X_OPERAND_ICOND :
288 print_keyword (cd, info, & xc16x_cgen_opval_conditioncode_names, fields->f_icondcode, 0);
289 break;
290 case XC16X_OPERAND_LBIT2 :
291 print_normal (cd, info, fields->f_op_lbit2, 0, pc, length);
292 break;
293 case XC16X_OPERAND_LBIT4 :
294 print_normal (cd, info, fields->f_op_lbit4, 0, pc, length);
295 break;
296 case XC16X_OPERAND_MASK8 :
297 print_normal (cd, info, fields->f_mask8, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
298 break;
299 case XC16X_OPERAND_MASKLO8 :
300 print_normal (cd, info, fields->f_datahi8, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
301 break;
302 case XC16X_OPERAND_MEMGR8 :
303 print_keyword (cd, info, & xc16x_cgen_opval_memgr8_names, fields->f_memgr8, 0);
304 break;
305 case XC16X_OPERAND_MEMORY :
306 print_address (cd, info, fields->f_memory, 0, pc, length);
307 break;
308 case XC16X_OPERAND_PAG :
309 print_pag (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
310 break;
311 case XC16X_OPERAND_PAGENUM :
312 print_normal (cd, info, fields->f_pagenum, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
313 break;
314 case XC16X_OPERAND_POF :
315 print_pof (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
316 break;
317 case XC16X_OPERAND_QBIT :
318 print_with_dot_prefix (cd, info, fields->f_qbit, 0|(1<<CGEN_OPERAND_DOT_PREFIX), pc, length);
319 break;
320 case XC16X_OPERAND_QHIBIT :
321 print_with_dot_prefix (cd, info, fields->f_qhibit, 0|(1<<CGEN_OPERAND_DOT_PREFIX), pc, length);
322 break;
323 case XC16X_OPERAND_QLOBIT :
324 print_with_dot_prefix (cd, info, fields->f_qlobit, 0|(1<<CGEN_OPERAND_DOT_PREFIX), pc, length);
325 break;
326 case XC16X_OPERAND_REG8 :
327 print_keyword (cd, info, & xc16x_cgen_opval_r8_names, fields->f_reg8, 0);
328 break;
329 case XC16X_OPERAND_REGB8 :
330 print_keyword (cd, info, & xc16x_cgen_opval_grb8_names, fields->f_regb8, 0);
331 break;
332 case XC16X_OPERAND_REGBMEM8 :
333 print_keyword (cd, info, & xc16x_cgen_opval_regbmem8_names, fields->f_regmem8, 0);
334 break;
335 case XC16X_OPERAND_REGHI8 :
336 print_keyword (cd, info, & xc16x_cgen_opval_r8_names, fields->f_reghi8, 0);
337 break;
338 case XC16X_OPERAND_REGMEM8 :
339 print_keyword (cd, info, & xc16x_cgen_opval_regmem8_names, fields->f_regmem8, 0);
340 break;
341 case XC16X_OPERAND_REGOFF8 :
342 print_keyword (cd, info, & xc16x_cgen_opval_r8_names, fields->f_regoff8, 0);
343 break;
344 case XC16X_OPERAND_REL :
345 print_normal (cd, info, fields->f_rel8, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
346 break;
347 case XC16X_OPERAND_RELHI :
348 print_normal (cd, info, fields->f_relhi8, 0|(1<<CGEN_OPERAND_SIGNED)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
349 break;
350 case XC16X_OPERAND_SEG :
351 print_normal (cd, info, fields->f_seg8, 0, pc, length);
352 break;
353 case XC16X_OPERAND_SEGHI8 :
354 print_normal (cd, info, fields->f_segnum8, 0, pc, length);
355 break;
356 case XC16X_OPERAND_SEGM :
357 print_seg (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
358 break;
359 case XC16X_OPERAND_SOF :
360 print_sof (cd, info, 0, 0|(1<<CGEN_OPERAND_SIGNED), pc, length);
361 break;
362 case XC16X_OPERAND_SR :
363 print_keyword (cd, info, & xc16x_cgen_opval_gr_names, fields->f_r2, 0);
364 break;
365 case XC16X_OPERAND_SR2 :
366 print_keyword (cd, info, & xc16x_cgen_opval_gr_names, fields->f_r0, 0);
367 break;
368 case XC16X_OPERAND_SRB :
369 print_keyword (cd, info, & xc16x_cgen_opval_grb_names, fields->f_r2, 0);
370 break;
371 case XC16X_OPERAND_SRC1 :
372 print_keyword (cd, info, & xc16x_cgen_opval_gr_names, fields->f_r1, 0);
373 break;
374 case XC16X_OPERAND_SRC2 :
375 print_keyword (cd, info, & xc16x_cgen_opval_gr_names, fields->f_r2, 0);
376 break;
377 case XC16X_OPERAND_SRDIV :
378 print_keyword (cd, info, & xc16x_cgen_opval_regdiv8_names, fields->f_reg8, 0);
379 break;
380 case XC16X_OPERAND_U4 :
381 print_keyword (cd, info, & xc16x_cgen_opval_reg0_name, fields->f_uimm4, 0);
382 break;
383 case XC16X_OPERAND_UIMM16 :
384 print_normal (cd, info, fields->f_uimm16, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
385 break;
386 case XC16X_OPERAND_UIMM2 :
387 print_keyword (cd, info, & xc16x_cgen_opval_ext_names, fields->f_uimm2, 0|(1<<CGEN_OPERAND_HASH_PREFIX));
388 break;
389 case XC16X_OPERAND_UIMM3 :
390 print_keyword (cd, info, & xc16x_cgen_opval_reg0_name1, fields->f_uimm3, 0|(1<<CGEN_OPERAND_HASH_PREFIX));
391 break;
392 case XC16X_OPERAND_UIMM4 :
393 print_normal (cd, info, fields->f_uimm4, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
394 break;
395 case XC16X_OPERAND_UIMM7 :
396 print_normal (cd, info, fields->f_uimm7, 0|(1<<CGEN_OPERAND_HASH_PREFIX)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_PCREL_ADDR), pc, length);
397 break;
398 case XC16X_OPERAND_UIMM8 :
399 print_normal (cd, info, fields->f_uimm8, 0|(1<<CGEN_OPERAND_HASH_PREFIX), pc, length);
400 break;
401 case XC16X_OPERAND_UPAG16 :
402 print_with_pag_prefix (cd, info, fields->f_uimm16, 0|(1<<CGEN_OPERAND_PAG_PREFIX), pc, length);
403 break;
404 case XC16X_OPERAND_UPOF16 :
405 print_with_pof_prefix (cd, info, fields->f_memory, 0|(1<<CGEN_OPERAND_POF_PREFIX), pc, length);
406 break;
407 case XC16X_OPERAND_USEG16 :
408 print_normal (cd, info, fields->f_offset16, 0|(1<<CGEN_OPERAND_SEG_PREFIX)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_ABS_ADDR), pc, length);
409 break;
410 case XC16X_OPERAND_USEG8 :
411 print_normal (cd, info, fields->f_seg8, 0|(1<<CGEN_OPERAND_SEG_PREFIX), pc, length);
412 break;
413 case XC16X_OPERAND_USOF16 :
414 print_normal (cd, info, fields->f_offset16, 0|(1<<CGEN_OPERAND_SOF_PREFIX)|(1<<CGEN_OPERAND_RELOC)|(1<<CGEN_OPERAND_ABS_ADDR), pc, length);
415 break;
416
417 default :
418 /* xgettext:c-format */
419 fprintf (stderr, _("Unrecognized field %d while printing insn.\n"),
420 opindex);
421 abort ();
422 }
423 }
424
425 cgen_print_fn * const xc16x_cgen_print_handlers[] =
426 {
427 print_insn_normal,
428 };
429
430
431 void
xc16x_cgen_init_dis(CGEN_CPU_DESC cd)432 xc16x_cgen_init_dis (CGEN_CPU_DESC cd)
433 {
434 xc16x_cgen_init_opcode_table (cd);
435 xc16x_cgen_init_ibld_table (cd);
436 cd->print_handlers = & xc16x_cgen_print_handlers[0];
437 cd->print_operand = xc16x_cgen_print_operand;
438 }
439
440
441 /* Default print handler. */
442
443 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)444 print_normal (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
445 void *dis_info,
446 long value,
447 unsigned int attrs,
448 bfd_vma pc ATTRIBUTE_UNUSED,
449 int length ATTRIBUTE_UNUSED)
450 {
451 disassemble_info *info = (disassemble_info *) dis_info;
452
453 /* Print the operand as directed by the attributes. */
454 if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SEM_ONLY))
455 ; /* nothing to do */
456 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SIGNED))
457 (*info->fprintf_func) (info->stream, "%ld", value);
458 else
459 (*info->fprintf_func) (info->stream, "0x%lx", value);
460 }
461
462 /* Default address handler. */
463
464 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)465 print_address (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
466 void *dis_info,
467 bfd_vma value,
468 unsigned int attrs,
469 bfd_vma pc ATTRIBUTE_UNUSED,
470 int length ATTRIBUTE_UNUSED)
471 {
472 disassemble_info *info = (disassemble_info *) dis_info;
473
474 /* Print the operand as directed by the attributes. */
475 if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SEM_ONLY))
476 ; /* Nothing to do. */
477 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_PCREL_ADDR))
478 (*info->print_address_func) (value, info);
479 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_ABS_ADDR))
480 (*info->print_address_func) (value, info);
481 else if (CGEN_BOOL_ATTR (attrs, CGEN_OPERAND_SIGNED))
482 (*info->fprintf_func) (info->stream, "%ld", (long) value);
483 else
484 (*info->fprintf_func) (info->stream, "0x%lx", (long) value);
485 }
486
487 /* Keyword print handler. */
488
489 static void
print_keyword(CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,void * dis_info,CGEN_KEYWORD * keyword_table,long value,unsigned int attrs ATTRIBUTE_UNUSED)490 print_keyword (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
491 void *dis_info,
492 CGEN_KEYWORD *keyword_table,
493 long value,
494 unsigned int attrs ATTRIBUTE_UNUSED)
495 {
496 disassemble_info *info = (disassemble_info *) dis_info;
497 const CGEN_KEYWORD_ENTRY *ke;
498
499 ke = cgen_keyword_lookup_value (keyword_table, value);
500 if (ke != NULL)
501 (*info->fprintf_func) (info->stream, "%s", ke->name);
502 else
503 (*info->fprintf_func) (info->stream, "???");
504 }
505
506 /* Default insn printer.
507
508 DIS_INFO is defined as `void *' so the disassembler needn't know anything
509 about disassemble_info. */
510
511 static void
print_insn_normal(CGEN_CPU_DESC cd,void * dis_info,const CGEN_INSN * insn,CGEN_FIELDS * fields,bfd_vma pc,int length)512 print_insn_normal (CGEN_CPU_DESC cd,
513 void *dis_info,
514 const CGEN_INSN *insn,
515 CGEN_FIELDS *fields,
516 bfd_vma pc,
517 int length)
518 {
519 const CGEN_SYNTAX *syntax = CGEN_INSN_SYNTAX (insn);
520 disassemble_info *info = (disassemble_info *) dis_info;
521 const CGEN_SYNTAX_CHAR_TYPE *syn;
522
523 CGEN_INIT_PRINT (cd);
524
525 for (syn = CGEN_SYNTAX_STRING (syntax); *syn; ++syn)
526 {
527 if (CGEN_SYNTAX_MNEMONIC_P (*syn))
528 {
529 (*info->fprintf_func) (info->stream, "%s", CGEN_INSN_MNEMONIC (insn));
530 continue;
531 }
532 if (CGEN_SYNTAX_CHAR_P (*syn))
533 {
534 (*info->fprintf_func) (info->stream, "%c", CGEN_SYNTAX_CHAR (*syn));
535 continue;
536 }
537
538 /* We have an operand. */
539 xc16x_cgen_print_operand (cd, CGEN_SYNTAX_FIELD (*syn), info,
540 fields, CGEN_INSN_ATTRS (insn), pc, length);
541 }
542 }
543
544 /* Subroutine of print_insn. Reads an insn into the given buffers and updates
545 the extract info.
546 Returns 0 if all is well, non-zero otherwise. */
547
548 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)549 read_insn (CGEN_CPU_DESC cd ATTRIBUTE_UNUSED,
550 bfd_vma pc,
551 disassemble_info *info,
552 bfd_byte *buf,
553 int buflen,
554 CGEN_EXTRACT_INFO *ex_info,
555 unsigned long *insn_value)
556 {
557 int status = (*info->read_memory_func) (pc, buf, buflen, info);
558
559 if (status != 0)
560 {
561 (*info->memory_error_func) (status, pc, info);
562 return -1;
563 }
564
565 ex_info->dis_info = info;
566 ex_info->valid = (1 << buflen) - 1;
567 ex_info->insn_bytes = buf;
568
569 *insn_value = bfd_get_bits (buf, buflen * 8, info->endian == BFD_ENDIAN_BIG);
570 return 0;
571 }
572
573 /* Utility to print an insn.
574 BUF is the base part of the insn, target byte order, BUFLEN bytes long.
575 The result is the size of the insn in bytes or zero for an unknown insn
576 or -1 if an error occurs fetching data (memory_error_func will have
577 been called). */
578
579 static int
print_insn(CGEN_CPU_DESC cd,bfd_vma pc,disassemble_info * info,bfd_byte * buf,unsigned int buflen)580 print_insn (CGEN_CPU_DESC cd,
581 bfd_vma pc,
582 disassemble_info *info,
583 bfd_byte *buf,
584 unsigned int buflen)
585 {
586 CGEN_INSN_INT insn_value;
587 const CGEN_INSN_LIST *insn_list;
588 CGEN_EXTRACT_INFO ex_info;
589 int basesize;
590
591 /* Extract base part of instruction, just in case CGEN_DIS_* uses it. */
592 basesize = cd->base_insn_bitsize < buflen * 8 ?
593 cd->base_insn_bitsize : buflen * 8;
594 insn_value = cgen_get_insn_value (cd, buf, basesize);
595
596
597 /* Fill in ex_info fields like read_insn would. Don't actually call
598 read_insn, since the incoming buffer is already read (and possibly
599 modified a la m32r). */
600 ex_info.valid = (1 << buflen) - 1;
601 ex_info.dis_info = info;
602 ex_info.insn_bytes = buf;
603
604 /* The instructions are stored in hash lists.
605 Pick the first one and keep trying until we find the right one. */
606
607 insn_list = CGEN_DIS_LOOKUP_INSN (cd, (char *) buf, insn_value);
608 while (insn_list != NULL)
609 {
610 const CGEN_INSN *insn = insn_list->insn;
611 CGEN_FIELDS fields;
612 int length;
613 unsigned long insn_value_cropped;
614
615 #ifdef CGEN_VALIDATE_INSN_SUPPORTED
616 /* Not needed as insn shouldn't be in hash lists if not supported. */
617 /* Supported by this cpu? */
618 if (! xc16x_cgen_insn_supported (cd, insn))
619 {
620 insn_list = CGEN_DIS_NEXT_INSN (insn_list);
621 continue;
622 }
623 #endif
624
625 /* Basic bit mask must be correct. */
626 /* ??? May wish to allow target to defer this check until the extract
627 handler. */
628
629 /* Base size may exceed this instruction's size. Extract the
630 relevant part from the buffer. */
631 if ((unsigned) (CGEN_INSN_BITSIZE (insn) / 8) < buflen &&
632 (unsigned) (CGEN_INSN_BITSIZE (insn) / 8) <= sizeof (unsigned long))
633 insn_value_cropped = bfd_get_bits (buf, CGEN_INSN_BITSIZE (insn),
634 info->endian == BFD_ENDIAN_BIG);
635 else
636 insn_value_cropped = insn_value;
637
638 if ((insn_value_cropped & CGEN_INSN_BASE_MASK (insn))
639 == CGEN_INSN_BASE_VALUE (insn))
640 {
641 /* Printing is handled in two passes. The first pass parses the
642 machine insn and extracts the fields. The second pass prints
643 them. */
644
645 /* Make sure the entire insn is loaded into insn_value, if it
646 can fit. */
647 if (((unsigned) CGEN_INSN_BITSIZE (insn) > cd->base_insn_bitsize) &&
648 (unsigned) (CGEN_INSN_BITSIZE (insn) / 8) <= sizeof (unsigned long))
649 {
650 unsigned long full_insn_value;
651 int rc = read_insn (cd, pc, info, buf,
652 CGEN_INSN_BITSIZE (insn) / 8,
653 & ex_info, & full_insn_value);
654 if (rc != 0)
655 return rc;
656 length = CGEN_EXTRACT_FN (cd, insn)
657 (cd, insn, &ex_info, full_insn_value, &fields, pc);
658 }
659 else
660 length = CGEN_EXTRACT_FN (cd, insn)
661 (cd, insn, &ex_info, insn_value_cropped, &fields, pc);
662
663 /* Length < 0 -> error. */
664 if (length < 0)
665 return length;
666 if (length > 0)
667 {
668 CGEN_PRINT_FN (cd, insn) (cd, info, insn, &fields, pc, length);
669 /* Length is in bits, result is in bytes. */
670 return length / 8;
671 }
672 }
673
674 insn_list = CGEN_DIS_NEXT_INSN (insn_list);
675 }
676
677 return 0;
678 }
679
680 /* Default value for CGEN_PRINT_INSN.
681 The result is the size of the insn in bytes or zero for an unknown insn
682 or -1 if an error occured fetching bytes. */
683
684 #ifndef CGEN_PRINT_INSN
685 #define CGEN_PRINT_INSN default_print_insn
686 #endif
687
688 static int
default_print_insn(CGEN_CPU_DESC cd,bfd_vma pc,disassemble_info * info)689 default_print_insn (CGEN_CPU_DESC cd, bfd_vma pc, disassemble_info *info)
690 {
691 bfd_byte buf[CGEN_MAX_INSN_SIZE];
692 int buflen;
693 int status;
694
695 /* Attempt to read the base part of the insn. */
696 buflen = cd->base_insn_bitsize / 8;
697 status = (*info->read_memory_func) (pc, buf, buflen, info);
698
699 /* Try again with the minimum part, if min < base. */
700 if (status != 0 && (cd->min_insn_bitsize < cd->base_insn_bitsize))
701 {
702 buflen = cd->min_insn_bitsize / 8;
703 status = (*info->read_memory_func) (pc, buf, buflen, info);
704 }
705
706 if (status != 0)
707 {
708 (*info->memory_error_func) (status, pc, info);
709 return -1;
710 }
711
712 return print_insn (cd, pc, info, buf, buflen);
713 }
714
715 /* Main entry point.
716 Print one instruction from PC on INFO->STREAM.
717 Return the size of the instruction (in bytes). */
718
719 typedef struct cpu_desc_list
720 {
721 struct cpu_desc_list *next;
722 CGEN_BITSET *isa;
723 int mach;
724 int endian;
725 CGEN_CPU_DESC cd;
726 } cpu_desc_list;
727
728 int
print_insn_xc16x(bfd_vma pc,disassemble_info * info)729 print_insn_xc16x (bfd_vma pc, disassemble_info *info)
730 {
731 static cpu_desc_list *cd_list = 0;
732 cpu_desc_list *cl = 0;
733 static CGEN_CPU_DESC cd = 0;
734 static CGEN_BITSET *prev_isa;
735 static int prev_mach;
736 static int prev_endian;
737 int length;
738 CGEN_BITSET *isa;
739 int mach;
740 int endian = (info->endian == BFD_ENDIAN_BIG
741 ? CGEN_ENDIAN_BIG
742 : CGEN_ENDIAN_LITTLE);
743 enum bfd_architecture arch;
744
745 /* ??? gdb will set mach but leave the architecture as "unknown" */
746 #ifndef CGEN_BFD_ARCH
747 #define CGEN_BFD_ARCH bfd_arch_xc16x
748 #endif
749 arch = info->arch;
750 if (arch == bfd_arch_unknown)
751 arch = CGEN_BFD_ARCH;
752
753 /* There's no standard way to compute the machine or isa number
754 so we leave it to the target. */
755 #ifdef CGEN_COMPUTE_MACH
756 mach = CGEN_COMPUTE_MACH (info);
757 #else
758 mach = info->mach;
759 #endif
760
761 #ifdef CGEN_COMPUTE_ISA
762 {
763 static CGEN_BITSET *permanent_isa;
764
765 if (!permanent_isa)
766 permanent_isa = cgen_bitset_create (MAX_ISAS);
767 isa = permanent_isa;
768 cgen_bitset_clear (isa);
769 cgen_bitset_add (isa, CGEN_COMPUTE_ISA (info));
770 }
771 #else
772 isa = info->insn_sets;
773 #endif
774
775 /* If we've switched cpu's, try to find a handle we've used before */
776 if (cd
777 && (cgen_bitset_compare (isa, prev_isa) != 0
778 || mach != prev_mach
779 || endian != prev_endian))
780 {
781 cd = 0;
782 for (cl = cd_list; cl; cl = cl->next)
783 {
784 if (cgen_bitset_compare (cl->isa, isa) == 0 &&
785 cl->mach == mach &&
786 cl->endian == endian)
787 {
788 cd = cl->cd;
789 prev_isa = cd->isas;
790 break;
791 }
792 }
793 }
794
795 /* If we haven't initialized yet, initialize the opcode table. */
796 if (! cd)
797 {
798 const bfd_arch_info_type *arch_type = bfd_lookup_arch (arch, mach);
799 const char *mach_name;
800
801 if (!arch_type)
802 abort ();
803 mach_name = arch_type->printable_name;
804
805 prev_isa = cgen_bitset_copy (isa);
806 prev_mach = mach;
807 prev_endian = endian;
808 cd = xc16x_cgen_cpu_open (CGEN_CPU_OPEN_ISAS, prev_isa,
809 CGEN_CPU_OPEN_BFDMACH, mach_name,
810 CGEN_CPU_OPEN_ENDIAN, prev_endian,
811 CGEN_CPU_OPEN_END);
812 if (!cd)
813 abort ();
814
815 /* Save this away for future reference. */
816 cl = xmalloc (sizeof (struct cpu_desc_list));
817 cl->cd = cd;
818 cl->isa = prev_isa;
819 cl->mach = mach;
820 cl->endian = endian;
821 cl->next = cd_list;
822 cd_list = cl;
823
824 xc16x_cgen_init_dis (cd);
825 }
826
827 /* We try to have as much common code as possible.
828 But at this point some targets need to take over. */
829 /* ??? Some targets may need a hook elsewhere. Try to avoid this,
830 but if not possible try to move this hook elsewhere rather than
831 have two hooks. */
832 length = CGEN_PRINT_INSN (cd, pc, info);
833 if (length > 0)
834 return length;
835 if (length < 0)
836 return -1;
837
838 (*info->fprintf_func) (info->stream, UNKNOWN_INSN_MSG);
839 return cd->default_insn_bitsize / 8;
840 }
841