Coverage Report

Created: 2026-05-11 07:54

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/opcodes/visium-dis.c
Line
Count
Source
1
/* Single instruction disassembler for the Visium.
2
3
   Copyright (C) 2002-2026 Free Software Foundation, Inc.
4
5
   This file is part of the GNU opcodes library.
6
7
   This library is free software; you can redistribute it and/or modify
8
   it under the terms of the GNU General Public License as published by
9
   the Free Software Foundation; either version 3, or (at your option)
10
   any later version.
11
12
   It is distributed in the hope that it will be useful, but WITHOUT
13
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
14
   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
15
   License for more details.
16
17
   You should have received a copy of the GNU General Public License
18
   along with this program; if not, write to the Free Software
19
   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
20
   MA 02110-1301, USA.  */
21
22
#include "sysdep.h"
23
#include "disassemble.h"
24
#include "opcode/visium.h"
25
26
#include <string.h>
27
#include <stdlib.h>
28
#include <stdio.h>
29
#include <ctype.h>
30
#include <setjmp.h>
31
32
/* Maximum length of an instruction.  */
33
#define MAXLEN 4
34
35
struct private
36
{
37
  /* Points to first byte not fetched.  */
38
  bfd_byte *max_fetched;
39
  bfd_byte the_buffer[MAXLEN];
40
  bfd_vma insn_start;
41
  jmp_buf bailout;
42
};
43
44
/* Make sure that bytes from INFO->PRIVATE_DATA->BUFFER (inclusive)
45
   to ADDR (exclusive) are valid.  Returns 1 for success, longjmps
46
   on error.  */
47
#define FETCH_DATA(info, addr) \
48
542k
  ((addr) <= ((struct private *)(info->private_data))->max_fetched \
49
542k
   ? 1 : fetch_data ((info), (addr)))
50
51
static int fetch_data (struct disassemble_info *info, bfd_byte * addr);
52
53
static int
54
fetch_data (struct disassemble_info *info, bfd_byte *addr)
55
542k
{
56
542k
  int status;
57
542k
  struct private *priv = (struct private *) info->private_data;
58
542k
  bfd_vma start = priv->insn_start + (priv->max_fetched - priv->the_buffer);
59
60
542k
  status = (*info->read_memory_func) (start,
61
542k
              priv->max_fetched,
62
542k
              addr - priv->max_fetched, info);
63
542k
  if (status != 0)
64
272
    {
65
272
      (*info->memory_error_func) (status, start, info);
66
272
      longjmp (priv->bailout, 1);
67
272
    }
68
542k
  else
69
542k
    priv->max_fetched = addr;
70
542k
  return 1;
71
542k
}
72
73
static char *size_names[] = { "?", "b", "w", "?", "l", "?", "?", "?" };
74
75
static char *cc_names[] =
76
{
77
  "fa", "eq", "cs", "os", "ns", "ne", "cc", "oc",
78
  "nc", "ge", "gt", "hi", "le", "ls", "lt", "tr"
79
};
80
81
/* Disassemble non-storage relative instructions.  */
82
83
static int
84
disassem_class0 (disassemble_info *info, unsigned int ins)
85
130k
{
86
130k
  int opcode = (ins >> 21) & 0x000f;
87
88
130k
  if (ins & CLASS0_UNUSED_MASK)
89
71.7k
    goto illegal_opcode;
90
91
59.1k
  switch (opcode)
92
59.1k
    {
93
50.1k
    case 0:
94
      /* BRR instruction.  */
95
50.1k
      {
96
50.1k
  unsigned cbf = (ins >> 27) & 0x000f;
97
50.1k
  int displacement = ((ins & 0xffff) ^ 0x8000) - 0x8000;
98
99
50.1k
  if (ins == 0)
100
29.8k
    (*info->fprintf_func) (info->stream, "nop");
101
20.3k
  else
102
20.3k
    (*info->fprintf_func) (info->stream, "brr     %s,%+d",
103
20.3k
         cc_names[cbf], displacement);
104
50.1k
      }
105
50.1k
      break;
106
1.01k
    case 1:
107
      /* Illegal opcode.  */
108
1.01k
      goto illegal_opcode;
109
0
      break;
110
596
    case 2:
111
      /* Illegal opcode.  */
112
596
      goto illegal_opcode;
113
0
      break;
114
896
    case 3:
115
      /* Illegal opcode.  */
116
896
      goto illegal_opcode;
117
0
      break;
118
712
    case 4:
119
      /* Illegal opcode.  */
120
712
      goto illegal_opcode;
121
0
      break;
122
887
    case 5:
123
      /* Illegal opcode.  */
124
887
      goto illegal_opcode;
125
0
      break;
126
1.39k
    case 6:
127
      /* Illegal opcode.  */
128
1.39k
      goto illegal_opcode;
129
0
      break;
130
278
    case 7:
131
      /* Illegal opcode.  */
132
278
      goto illegal_opcode;
133
0
      break;
134
2.51k
    case 8:
135
      /* Illegal opcode.  */
136
2.51k
      goto illegal_opcode;
137
0
      break;
138
110
    case 9:
139
      /* Illegal opcode.  */
140
110
      goto illegal_opcode;
141
0
      break;
142
59
    case 10:
143
      /* Illegal opcode.  */
144
59
      goto illegal_opcode;
145
0
      break;
146
252
    case 11:
147
      /* Illegal opcode.  */
148
252
      goto illegal_opcode;
149
0
      break;
150
98
    case 12:
151
      /* Illegal opcode.  */
152
98
      goto illegal_opcode;
153
0
      break;
154
35
    case 13:
155
      /* Illegal opcode.  */
156
35
      goto illegal_opcode;
157
0
      break;
158
63
    case 14:
159
      /* Illegal opcode.  */
160
63
      goto illegal_opcode;
161
0
      break;
162
92
    case 15:
163
      /* Illegal opcode.  */
164
92
      goto illegal_opcode;
165
0
      break;
166
59.1k
    }
167
50.1k
  return 0;
168
169
80.8k
 illegal_opcode:
170
80.8k
  return -1;
171
59.1k
}
172
173
/* Disassemble non-storage register class instructions.   */
174
175
static int
176
disassem_class1 (disassemble_info *info, unsigned int ins)
177
58.4k
{
178
58.4k
  int opcode = (ins >> 21) & 0xf;
179
58.4k
  int source_a = (ins >> 16) & 0x1f;
180
58.4k
  int source_b = (ins >> 4) & 0x1f;
181
58.4k
  int indx = (ins >> 10) & 0x1f;
182
183
58.4k
  int size = ins & 0x7;
184
185
58.4k
  if (ins & CLASS1_UNUSED_MASK)
186
32.8k
    goto illegal_opcode;
187
188
25.6k
  switch (opcode)
189
25.6k
    {
190
4.15k
    case 0:
191
      /* Stop.  */
192
4.15k
      (*info->fprintf_func) (info->stream, "stop    %d,r%d", indx, source_a);
193
4.15k
      break;
194
974
    case 1:
195
      /* BMI - Block Move Indirect.  */
196
974
      if (ins != BMI)
197
905
  goto illegal_opcode;
198
199
69
      (*info->fprintf_func) (info->stream, "bmi     r1,r2,r3");
200
69
      break;
201
914
    case 2:
202
      /* Illegal opcode.  */
203
914
      goto illegal_opcode;
204
0
      break;
205
2.00k
    case 3:
206
      /* BMD - Block Move Direct.  */
207
2.00k
      if (ins != BMD)
208
1.96k
  goto illegal_opcode;
209
210
38
      (*info->fprintf_func) (info->stream, "bmd     r1,r2,r3");
211
38
      break;
212
1.00k
    case 4:
213
      /* DSI - Disable Interrupts.  */
214
1.00k
      if (ins != DSI)
215
992
  goto illegal_opcode;
216
217
17
      (*info->fprintf_func) (info->stream, "dsi");
218
17
      break;
219
220
1.07k
    case 5:
221
      /* ENI - Enable Interrupts.  */
222
1.07k
      if (ins != ENI)
223
991
  goto illegal_opcode;
224
225
87
      (*info->fprintf_func) (info->stream, "eni");
226
87
      break;
227
228
1.16k
    case 6:
229
      /* Illegal opcode (was EUT).  */
230
1.16k
      goto illegal_opcode;
231
0
      break;
232
1.02k
    case 7:
233
      /* RFI - Return from Interrupt.  */
234
1.02k
      if (ins != RFI)
235
991
  goto illegal_opcode;
236
237
38
      (*info->fprintf_func) (info->stream, "rfi");
238
38
      break;
239
3.14k
    case 8:
240
      /* Illegal opcode.  */
241
3.14k
      goto illegal_opcode;
242
0
      break;
243
1.59k
    case 9:
244
      /* Illegal opcode.  */
245
1.59k
      goto illegal_opcode;
246
0
      break;
247
1.10k
    case 10:
248
      /* Illegal opcode.  */
249
1.10k
      goto illegal_opcode;
250
0
      break;
251
1.88k
    case 11:
252
      /* Illegal opcode.  */
253
1.88k
      goto illegal_opcode;
254
0
      break;
255
1.04k
    case 12:
256
      /* Illegal opcode.  */
257
1.04k
      goto illegal_opcode;
258
0
      break;
259
787
    case 13:
260
787
      goto illegal_opcode;
261
0
      break;
262
1.04k
    case 14:
263
1.04k
      goto illegal_opcode;
264
0
      break;
265
2.66k
    case 15:
266
2.66k
      if (ins & EAM_SELECT_MASK)
267
1.93k
  {
268
    /* Extension arithmetic module write */
269
1.93k
    int fp_ins = (ins >> 27) & 0xf;
270
271
1.93k
    if (size != 4)
272
489
      goto illegal_opcode;
273
274
1.45k
    if (ins & FP_SELECT_MASK)
275
770
      {
276
        /* Which floating point instructions don't need a fsrcB
277
           register.  */
278
770
        const int no_fsrcb[16] = { 1, 0, 0, 0, 0, 1, 1, 1,
279
770
    1, 1, 0, 0, 1, 0, 0, 0
280
770
        };
281
770
        if (no_fsrcb[fp_ins] && source_b)
282
68
    goto illegal_opcode;
283
284
        /* Check that none of the floating register register numbers
285
           is higher than 15. (If this is fload, then srcA is a
286
           general register.  */
287
702
        if (ins & ((1 << 14) | (1 << 8)) || (fp_ins && ins & (1 << 20)))
288
128
    goto illegal_opcode;
289
290
574
        switch (fp_ins)
291
574
    {
292
29
    case 0:
293
29
      (*info->fprintf_func) (info->stream, "fload   f%d,r%d",
294
29
           indx, source_a);
295
29
      break;
296
44
    case 1:
297
44
      (*info->fprintf_func) (info->stream, "fadd    f%d,f%d,f%d",
298
44
           indx, source_a, source_b);
299
44
      break;
300
50
    case 2:
301
50
      (*info->fprintf_func) (info->stream, "fsub    f%d,f%d,f%d",
302
50
           indx, source_a, source_b);
303
50
      break;
304
2
    case 3:
305
2
      (*info->fprintf_func) (info->stream, "fmult   f%d,f%d,f%d",
306
2
           indx, source_a, source_b);
307
2
      break;
308
53
    case 4:
309
53
      (*info->fprintf_func) (info->stream, "fdiv    f%d,f%d,f%d",
310
53
           indx, source_a, source_b);
311
53
      break;
312
135
    case 5:
313
135
      (*info->fprintf_func) (info->stream, "fsqrt   f%d,f%d",
314
135
           indx, source_a);
315
135
      break;
316
84
    case 6:
317
84
      (*info->fprintf_func) (info->stream, "fneg    f%d,f%d",
318
84
           indx, source_a);
319
84
      break;
320
27
    case 7:
321
27
      (*info->fprintf_func) (info->stream, "fabs    f%d,f%d",
322
27
           indx, source_a);
323
27
      break;
324
2
    case 8:
325
2
      (*info->fprintf_func) (info->stream, "ftoi    f%d,f%d",
326
2
           indx, source_a);
327
2
      break;
328
21
    case 9:
329
21
      (*info->fprintf_func) (info->stream, "itof    f%d,f%d",
330
21
           indx, source_a);
331
21
      break;
332
66
    case 12:
333
66
      (*info->fprintf_func) (info->stream, "fmove   f%d,f%d",
334
66
           indx, source_a);
335
66
      break;
336
61
    default:
337
61
      (*info->fprintf_func) (info->stream,
338
61
           "fpinst  %d,f%d,f%d,f%d", fp_ins,
339
61
           indx, source_a, source_b);
340
61
      break;
341
574
    }
342
574
      }
343
680
    else
344
680
      {
345
        /* Which EAM operations do not need a srcB register.  */
346
680
        const int no_srcb[32] =
347
680
        { 0, 0, 1, 1, 0, 1, 1, 1,
348
680
    0, 1, 1, 1, 0, 0, 0, 0,
349
680
    0, 0, 0, 0, 0, 0, 0, 0,
350
680
    0, 0, 0, 0, 0, 0, 0, 0
351
680
        };
352
353
680
        if (no_srcb[indx] && source_b)
354
15
    goto illegal_opcode;
355
356
665
        if (fp_ins)
357
51
    goto illegal_opcode;
358
359
614
        switch (indx)
360
614
    {
361
17
    case 0:
362
17
      (*info->fprintf_func) (info->stream, "mults   r%d,r%d",
363
17
           source_a, source_b);
364
17
      break;
365
41
    case 1:
366
41
      (*info->fprintf_func) (info->stream, "multu   r%d,r%d",
367
41
           source_a, source_b);
368
41
      break;
369
43
    case 2:
370
43
      (*info->fprintf_func) (info->stream, "divs    r%d",
371
43
           source_a);
372
43
      break;
373
0
    case 3:
374
0
      (*info->fprintf_func) (info->stream, "divu    r%d",
375
0
           source_a);
376
0
      break;
377
44
    case 4:
378
44
      (*info->fprintf_func) (info->stream, "writemd r%d,r%d",
379
44
           source_a, source_b);
380
44
      break;
381
40
    case 5:
382
40
      (*info->fprintf_func) (info->stream, "writemdc r%d",
383
40
           source_a);
384
40
      break;
385
0
    case 6:
386
0
      (*info->fprintf_func) (info->stream, "divds   r%d",
387
0
           source_a);
388
0
      break;
389
33
    case 7:
390
33
      (*info->fprintf_func) (info->stream, "divdu   r%d",
391
33
           source_a);
392
33
      break;
393
0
    case 9:
394
0
      (*info->fprintf_func) (info->stream, "asrd    r%d",
395
0
           source_a);
396
0
      break;
397
47
    case 10:
398
47
      (*info->fprintf_func) (info->stream, "lsrd    r%d",
399
47
           source_a);
400
47
      break;
401
30
    case 11:
402
30
      (*info->fprintf_func) (info->stream, "asld    r%d",
403
30
           source_a);
404
30
      break;
405
319
    default:
406
319
      (*info->fprintf_func) (info->stream,
407
319
           "eamwrite %d,r%d,r%d", indx,
408
319
           source_a, source_b);
409
319
      break;
410
614
    }
411
614
      }
412
1.45k
  }
413
723
      else
414
723
  {
415
    /* WRITE - write to memory.  */
416
723
    (*info->fprintf_func) (info->stream, "write.%s %d(r%d),r%d",
417
723
         size_names[size], indx, source_a, source_b);
418
723
  }
419
1.91k
      break;
420
25.6k
    }
421
422
6.31k
  return 0;
423
424
52.1k
 illegal_opcode:
425
52.1k
  return -1;
426
25.6k
}
427
428
/* Disassemble storage immediate class instructions.   */
429
430
static int
431
disassem_class2 (disassemble_info *info, unsigned int ins)
432
58.0k
{
433
58.0k
  int opcode = (ins >> 21) & 0xf;
434
58.0k
  int source_a = (ins >> 16) & 0x1f;
435
58.0k
  unsigned immediate = ins & 0x0000ffff;
436
437
58.0k
  if (ins & CC_MASK)
438
48.3k
    goto illegal_opcode;
439
440
9.73k
  switch (opcode)
441
9.73k
    {
442
1.18k
    case 0:
443
      /* ADDI instruction.  */
444
1.18k
      (*info->fprintf_func) (info->stream, "addi    r%d,%d", source_a,
445
1.18k
           immediate);
446
1.18k
      break;
447
463
    case 1:
448
      /* Illegal opcode.  */
449
463
      goto illegal_opcode;
450
0
      break;
451
416
    case 2:
452
      /* SUBI instruction.  */
453
416
      (*info->fprintf_func) (info->stream, "subi    r%d,%d", source_a,
454
416
           immediate);
455
416
      break;
456
335
    case 3:
457
      /* Illegal opcode.  */
458
335
      goto illegal_opcode;
459
0
      break;
460
578
    case 4:
461
      /* MOVIL instruction.  */
462
578
      (*info->fprintf_func) (info->stream, "movil   r%d,0x%04X", source_a,
463
578
           immediate);
464
578
      break;
465
166
    case 5:
466
      /* MOVIU instruction.  */
467
166
      (*info->fprintf_func) (info->stream, "moviu   r%d,0x%04X", source_a,
468
166
           immediate);
469
166
      break;
470
238
    case 6:
471
      /* MOVIQ instruction.  */
472
238
      (*info->fprintf_func) (info->stream, "moviq   r%d,%u", source_a,
473
238
           immediate);
474
238
      break;
475
499
    case 7:
476
      /* Illegal opcode.  */
477
499
      goto illegal_opcode;
478
0
      break;
479
2.25k
    case 8:
480
      /* WRTL instruction.  */
481
2.25k
      if (source_a != 0)
482
1.61k
  goto illegal_opcode;
483
484
644
      (*info->fprintf_func) (info->stream, "wrtl    0x%04X", immediate);
485
644
      break;
486
309
    case 9:
487
      /* WRTU instruction.  */
488
309
      if (source_a != 0)
489
218
  goto illegal_opcode;
490
491
91
      (*info->fprintf_func) (info->stream, "wrtu    0x%04X", immediate);
492
91
      break;
493
502
    case 10:
494
      /* Illegal opcode.  */
495
502
      goto illegal_opcode;
496
0
      break;
497
130
    case 11:
498
      /* Illegal opcode.  */
499
130
      goto illegal_opcode;
500
0
      break;
501
1.71k
    case 12:
502
      /* Illegal opcode.  */
503
1.71k
      goto illegal_opcode;
504
0
      break;
505
470
    case 13:
506
      /* Illegal opcode.  */
507
470
      goto illegal_opcode;
508
0
      break;
509
133
    case 14:
510
      /* Illegal opcode.  */
511
133
      goto illegal_opcode;
512
0
      break;
513
336
    case 15:
514
      /* Illegal opcode.  */
515
336
      goto illegal_opcode;
516
0
      break;
517
9.73k
    }
518
519
3.32k
  return 0;
520
521
54.7k
 illegal_opcode:
522
54.7k
  return -1;
523
9.73k
}
524
525
/* Disassemble storage register class instructions.  */
526
527
static int
528
disassem_class3 (disassemble_info *info, unsigned int ins)
529
90.8k
{
530
90.8k
  int opcode = (ins >> 21) & 0xf;
531
90.8k
  int source_b = (ins >> 4) & 0x1f;
532
90.8k
  int source_a = (ins >> 16) & 0x1f;
533
90.8k
  int size = ins & 0x7;
534
90.8k
  int dest = (ins >> 10) & 0x1f;
535
536
  /* Those instructions that don't have a srcB register.  */
537
90.8k
  const int no_srcb[16] =
538
90.8k
  { 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 0 };
539
540
  /* These are instructions which can take an immediate srcB value.  */
541
90.8k
  const int srcb_immed[16] =
542
90.8k
  { 0, 0, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 1 };
543
544
  /* User opcodes should not provide a non-zero srcB register
545
     when none is required. Only a BRA or floating point
546
     instruction should have a non-zero condition code field.
547
     Only a WRITE or EAMWRITE (opcode 15) should select an EAM
548
     or floating point operation.  Note that FP_SELECT_MASK is
549
     the same bit (bit 3) as the interrupt bit which
550
     distinguishes SYS1 from BRA and SYS2 from RFLAG.  */
551
90.8k
  if ((no_srcb[opcode] && source_b)
552
77.1k
      || (!srcb_immed[opcode] && ins & CLASS3_SOURCEB_IMMED)
553
57.1k
      || (opcode != 12 && opcode != 15 && ins & CC_MASK)
554
36.6k
      || (opcode != 15 && ins & (EAM_SELECT_MASK | FP_SELECT_MASK)))
555
55.6k
    goto illegal_opcode;
556
557
558
35.1k
  switch (opcode)
559
35.1k
    {
560
521
    case 0:
561
      /* ADD instruction.  */
562
521
      (*info->fprintf_func) (info->stream, "add.%s   r%d,r%d,r%d",
563
521
           size_names[size], dest, source_a, source_b);
564
521
      break;
565
134
    case 1:
566
      /* ADC instruction.  */
567
134
      (*info->fprintf_func) (info->stream, "adc.%s   r%d,r%d,r%d",
568
134
           size_names[size], dest, source_a, source_b);
569
134
      break;
570
125
    case 2:
571
      /* SUB instruction.  */
572
125
      if (dest == 0)
573
58
  (*info->fprintf_func) (info->stream, "cmp.%s   r%d,r%d",
574
58
             size_names[size], source_a, source_b);
575
67
      else
576
67
  (*info->fprintf_func) (info->stream, "sub.%s   r%d,r%d,r%d",
577
67
             size_names[size], dest, source_a, source_b);
578
125
      break;
579
365
    case 3:
580
      /* SUBC instruction.  */
581
365
      if (dest == 0)
582
186
  (*info->fprintf_func) (info->stream, "cmpc.%s  r%d,r%d",
583
186
             size_names[size], source_a, source_b);
584
179
      else
585
179
  (*info->fprintf_func) (info->stream, "subc.%s  r%d,r%d,r%d",
586
179
             size_names[size], dest, source_a, source_b);
587
365
      break;
588
120
    case 4:
589
      /* EXTW instruction.  */
590
120
      if (size == 1)
591
18
  goto illegal_opcode;
592
593
102
      (*info->fprintf_func) (info->stream, "extw.%s  r%d,r%d",
594
102
           size_names[size], dest, source_a);
595
102
      break;
596
262
    case 5:
597
      /* ASR instruction.  */
598
262
      if (ins & CLASS3_SOURCEB_IMMED)
599
33
  (*info->fprintf_func) (info->stream, "asr.%s   r%d,r%d,%d",
600
33
             size_names[size], dest, source_a, source_b);
601
229
      else
602
229
  (*info->fprintf_func) (info->stream, "asr.%s   r%d,r%d,r%d",
603
229
             size_names[size], dest, source_a, source_b);
604
262
      break;
605
389
    case 6:
606
      /* LSR instruction.  */
607
389
      if (ins & CLASS3_SOURCEB_IMMED)
608
280
  (*info->fprintf_func) (info->stream, "lsr.%s   r%d,r%d,%d",
609
280
             size_names[size], dest, source_a, source_b);
610
109
      else
611
109
  (*info->fprintf_func) (info->stream, "lsr.%s   r%d,r%d,r%d",
612
109
             size_names[size], dest, source_a, source_b);
613
389
      break;
614
118
    case 7:
615
      /* ASL instruction.  */
616
118
      if (ins & CLASS3_SOURCEB_IMMED)
617
89
  (*info->fprintf_func) (info->stream, "asl.%s   r%d,r%d,%d",
618
89
             size_names[size], dest, source_a, source_b);
619
29
      else
620
29
  (*info->fprintf_func) (info->stream, "asl.%s   r%d,r%d,r%d",
621
29
             size_names[size], dest, source_a, source_b);
622
118
      break;
623
448
    case 8:
624
      /* XOR instruction.  */
625
448
      (*info->fprintf_func) (info->stream, "xor.%s   r%d,r%d,r%d",
626
448
           size_names[size], dest, source_a, source_b);
627
448
      break;
628
231
    case 9:
629
      /* OR instruction.  */
630
231
      if (source_b == 0)
631
141
  (*info->fprintf_func) (info->stream, "move.%s  r%d,r%d",
632
141
             size_names[size], dest, source_a);
633
90
      else
634
90
  (*info->fprintf_func) (info->stream, "or.%s    r%d,r%d,r%d",
635
90
             size_names[size], dest, source_a, source_b);
636
231
      break;
637
71
    case 10:
638
      /* AND instruction.  */
639
71
      (*info->fprintf_func) (info->stream, "and.%s   r%d,r%d,r%d",
640
71
           size_names[size], dest, source_a, source_b);
641
71
      break;
642
87
    case 11:
643
      /* NOT instruction.  */
644
87
      (*info->fprintf_func) (info->stream, "not.%s   r%d,r%d",
645
87
           size_names[size], dest, source_a);
646
87
      break;
647
297
    case 12:
648
      /* BRA instruction.  */
649
297
      {
650
297
  unsigned cbf = (ins >> 27) & 0x000f;
651
652
297
  if (size != 4)
653
291
    goto illegal_opcode;
654
655
6
  (*info->fprintf_func) (info->stream, "bra     %s,r%d,r%d",
656
6
             cc_names[cbf], source_a, dest);
657
6
      }
658
0
      break;
659
160
    case 13:
660
      /* RFLAG instruction.  */
661
160
      if (source_a || size != 4)
662
105
  goto illegal_opcode;
663
664
55
      (*info->fprintf_func) (info->stream, "rflag   r%d", dest);
665
55
      break;
666
111
    case 14:
667
      /* EXTB instruction.  */
668
111
      (*info->fprintf_func) (info->stream, "extb.%s  r%d,r%d",
669
111
           size_names[size], dest, source_a);
670
111
      break;
671
31.7k
    case 15:
672
31.7k
      if (!(ins & CLASS3_SOURCEB_IMMED))
673
2.98k
  goto illegal_opcode;
674
675
28.7k
      if (ins & EAM_SELECT_MASK)
676
27.5k
  {
677
    /* Extension arithmetic module read.  */
678
27.5k
    int fp_ins = (ins >> 27) & 0xf;
679
680
27.5k
    if (size != 4)
681
27.1k
      goto illegal_opcode;
682
683
417
    if (ins & FP_SELECT_MASK)
684
223
      {
685
        /* Check fsrcA <= 15 and fsrcB <= 15.  */
686
223
        if (ins & ((1 << 20) | (1 << 8)))
687
196
    goto illegal_opcode;
688
689
27
        switch (fp_ins)
690
27
    {
691
0
    case 0:
692
0
      if (source_b)
693
0
        goto illegal_opcode;
694
695
0
      (*info->fprintf_func) (info->stream, "fstore  r%d,f%d",
696
0
           dest, source_a);
697
0
      break;
698
0
    case 10:
699
0
      (*info->fprintf_func) (info->stream, "fcmp    r%d,f%d,f%d",
700
0
           dest, source_a, source_b);
701
0
      break;
702
6
    case 11:
703
6
      (*info->fprintf_func) (info->stream, "fcmpe   r%d,f%d,f%d",
704
6
           dest, source_a, source_b);
705
6
      break;
706
21
    default:
707
21
      (*info->fprintf_func) (info->stream,
708
21
           "fpuread %d,r%d,f%d,f%d", fp_ins,
709
21
           dest, source_a, source_b);
710
21
      break;
711
27
    }
712
27
      }
713
194
    else
714
194
      {
715
194
        if (fp_ins || source_a)
716
194
    goto illegal_opcode;
717
718
0
        switch (source_b)
719
0
    {
720
0
    case 0:
721
0
      (*info->fprintf_func) (info->stream, "readmda r%d", dest);
722
0
      break;
723
0
    case 1:
724
0
      (*info->fprintf_func) (info->stream, "readmdb r%d", dest);
725
0
      break;
726
0
    case 2:
727
0
      (*info->fprintf_func) (info->stream, "readmdc r%d", dest);
728
0
      break;
729
0
    default:
730
0
      (*info->fprintf_func) (info->stream, "eamread r%d,%d",
731
0
           dest, source_b);
732
0
      break;
733
0
    }
734
0
      }
735
417
  }
736
1.19k
      else
737
1.19k
  {
738
1.19k
    if (ins & FP_SELECT_MASK)
739
431
      goto illegal_opcode;
740
741
    /* READ instruction.  */
742
761
    (*info->fprintf_func) (info->stream, "read.%s  r%d,%d(r%d)",
743
761
         size_names[size], dest, source_b, source_a);
744
761
  }
745
788
      break;
746
35.1k
    }
747
748
3.81k
  return 0;
749
750
87.0k
 illegal_opcode:
751
87.0k
  return -1;
752
753
35.1k
}
754
755
/* Print the visium instruction at address addr in debugged memory,
756
   on info->stream. Return length of the instruction, in bytes.  */
757
758
int
759
print_insn_visium (bfd_vma addr, disassemble_info *info)
760
542k
{
761
542k
  unsigned ins;
762
542k
  unsigned p1, p2;
763
542k
  int ans;
764
542k
  int i;
765
766
  /* Stuff copied from m68k-dis.c.  */
767
542k
  struct private priv;
768
542k
  bfd_byte *buffer = priv.the_buffer;
769
542k
  info->private_data = &priv;
770
542k
  priv.max_fetched = priv.the_buffer;
771
542k
  priv.insn_start = addr;
772
542k
  if (setjmp (priv.bailout) != 0)
773
272
    {
774
      /* Error return.  */
775
272
      return -1;
776
272
    }
777
778
  /* We do return this info.  */
779
542k
  info->insn_info_valid = 1;
780
781
  /* Assume non branch insn.  */
782
542k
  info->insn_type = dis_nonbranch;
783
784
  /* Assume no delay.  */
785
542k
  info->branch_delay_insns = 0;
786
787
  /* Assume no target known.  */
788
542k
  info->target = 0;
789
790
  /* Get 32-bit instruction word.  */
791
542k
  FETCH_DATA (info, buffer + 4);
792
542k
  ins = (unsigned) buffer[0] << 24;
793
542k
  ins |= buffer[1] << 16;
794
542k
  ins |= buffer[2] << 8;
795
542k
  ins |= buffer[3];
796
797
542k
  ans = 0;
798
799
542k
  p1 = buffer[0] ^ buffer[1] ^ buffer[2] ^ buffer[3];
800
542k
  p2 = 0;
801
4.87M
  for (i = 0; i < 8; i++)
802
4.33M
    {
803
4.33M
      p2 += p1 & 1;
804
4.33M
      p1 >>= 1;
805
4.33M
    }
806
807
  /* Decode the instruction.  */
808
542k
  if (p2 & 1)
809
203k
    ans = -1;
810
338k
  else
811
338k
    {
812
338k
      switch ((ins >> 25) & 0x3)
813
338k
  {
814
130k
  case 0:
815
130k
    ans = disassem_class0 (info, ins);
816
130k
    break;
817
58.4k
  case 1:
818
58.4k
    ans = disassem_class1 (info, ins);
819
58.4k
    break;
820
58.0k
  case 2:
821
58.0k
    ans = disassem_class2 (info, ins);
822
58.0k
    break;
823
90.8k
  case 3:
824
90.8k
    ans = disassem_class3 (info, ins);
825
90.8k
    break;
826
338k
  }
827
338k
    }
828
829
542k
  if (ans != 0)
830
478k
    (*info->fprintf_func) (info->stream, "err");
831
832
  /* Return number of bytes consumed (always 4 for the Visium).  */
833
542k
  return 4;
834
542k
}