Coverage Report

Created: 2026-05-11 07:54

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/opcodes/arc-dis.c
Line
Count
Source
1
/* Instruction printing code for the ARC.
2
   Copyright (C) 1994-2026 Free Software Foundation, Inc.
3
4
   Contributed by Claudiu Zissulescu (claziss@synopsys.com)
5
6
   This file is part of libopcodes.
7
8
   This library is free software; you can redistribute it and/or modify
9
   it under the terms of the GNU General Public License as published by
10
   the Free Software Foundation; either version 3, or (at your option)
11
   any later version.
12
13
   It is distributed in the hope that it will be useful, but WITHOUT
14
   ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
15
   or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public
16
   License for more details.
17
18
   You should have received a copy of the GNU General Public License
19
   along with this program; if not, write to the Free Software
20
   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21
   MA 02110-1301, USA.  */
22
23
#include "sysdep.h"
24
#include <stdio.h>
25
#include <assert.h>
26
#include "dis-asm.h"
27
#include "opcode/arc.h"
28
#include "elf/arc.h"
29
#include "arc-dis.h"
30
#include "arc-ext.h"
31
#include "elf-bfd.h"
32
#include "libiberty.h"
33
#include "opintl.h"
34
35
/* Structure used to iterate over, and extract the values for, operands of
36
   an opcode.  */
37
38
struct arc_operand_iterator
39
{
40
  /* The complete instruction value to extract operands from.  */
41
  unsigned long long insn;
42
43
  /* The LIMM if this is being tracked separately.  This field is only
44
     valid if we find the LIMM operand in the operand list.  */
45
  unsigned limm;
46
47
  /* The opcode this iterator is operating on.  */
48
  const struct arc_opcode *opcode;
49
50
  /* The index into the opcodes operand index list.  */
51
  const unsigned char *opidx;
52
};
53
54
/* A private data used by ARC decoder.  */
55
struct arc_disassemble_info
56
{
57
  /* The current disassembled arc opcode.  */
58
  const struct arc_opcode *opcode;
59
60
  /* Instruction length w/o limm field.  */
61
  unsigned insn_len;
62
63
  /* TRUE if we have limm.  */
64
  bool limm_p;
65
66
  /* LIMM value, if exists.  */
67
  unsigned limm;
68
69
  /* Condition code, if exists.  */
70
  unsigned condition_code;
71
72
  /* Writeback mode.  */
73
  unsigned writeback_mode;
74
75
  /* Number of operands.  */
76
  unsigned operands_count;
77
78
  struct arc_insn_operand operands[MAX_INSN_ARGS];
79
80
  /* Classes of extension instructions to be considered.  */
81
#define MAX_DECODES 25
82
  struct
83
  {
84
    insn_class_t insn_class;
85
    insn_subclass_t subclass;
86
  } decode[MAX_DECODES];
87
88
  /* Current size of the above array.  */
89
  unsigned decode_count;
90
91
  /* ISA mask value via disassembler info options, or bfd.  */
92
  unsigned isa_mask;
93
94
  /* True if we want to print using only hex numbers.  */
95
  bool print_hex;
96
};
97
98
/* Globals variables.  */
99
100
static const char * const regnames[64] =
101
{
102
  "r0", "r1", "r2", "r3", "r4", "r5", "r6", "r7",
103
  "r8", "r9", "r10", "r11", "r12", "r13", "r14", "r15",
104
  "r16", "r17", "r18", "r19", "r20", "r21", "r22", "r23",
105
  "r24", "r25", "gp", "fp", "sp", "ilink", "r30", "blink",
106
107
  "r32", "r33", "r34", "r35", "r36", "r37", "r38", "r39",
108
  "r40", "r41", "r42", "r43", "r44", "r45", "r46", "r47",
109
  "r48", "r49", "r50", "r51", "r52", "r53", "r54", "r55",
110
  "r56", "r57", "r58", "r59", "lp_count", "reserved", "LIMM", "pcl"
111
};
112
113
static const char * const addrtypenames[ARC_NUM_ADDRTYPES] =
114
{
115
  "bd", "jid", "lbd", "mbd", "sd", "sm", "xa", "xd",
116
  "cd", "cbd", "cjid", "clbd", "cm", "csd", "cxa", "cxd"
117
};
118
119
/* Macros section.  */
120
121
#ifdef DEBUG
122
# define pr_debug(fmt, args...) fprintf (stderr, fmt, ##args)
123
#else
124
# define pr_debug(fmt, args...)
125
#endif
126
127
#define ARRANGE_ENDIAN(info, buf)         \
128
290k
  (info->endian == BFD_ENDIAN_LITTLE ? bfd_getm32 (bfd_getl32 (buf))  \
129
290k
   : bfd_getb32 (buf))
130
131
667k
#define BITS(word,s,e)  (((word) >> (s)) & ((1ull << ((e) - (s)) << 1) - 1))
132
667k
#define OPCODE_32BIT_INSN(word) (BITS ((word), 27, 31))
133
134
/* Functions implementation.  */
135
136
/* Initialize private data.  */
137
static bool
138
init_arc_disasm_info (struct disassemble_info *info)
139
3.58k
{
140
3.58k
  struct arc_disassemble_info *arc_infop = calloc (1, sizeof (*arc_infop));
141
142
3.58k
  info->private_data = arc_infop;
143
3.58k
  return arc_infop != NULL;
144
3.58k
}
145
146
/* Add a new element to the decode list.  */
147
148
static void
149
add_to_decode (struct arc_disassemble_info *arc_infop,
150
         insn_class_t insn_class,
151
         insn_subclass_t subclass)
152
15.5k
{
153
15.5k
  unsigned int i;
154
131k
  for (i = 0; i < arc_infop->decode_count; i++)
155
125k
    if (arc_infop->decode[i].insn_class == insn_class
156
16.9k
  && arc_infop->decode[i].subclass == subclass)
157
9.48k
      return;
158
159
15.5k
  assert (i < MAX_DECODES);
160
6.02k
  arc_infop->decode[i].insn_class = insn_class;
161
6.02k
  arc_infop->decode[i].subclass = subclass;
162
6.02k
  arc_infop->decode_count = i + 1;
163
6.02k
}
164
165
/* Return TRUE if we need to skip the opcode from being
166
   disassembled.  */
167
168
static bool
169
skip_this_opcode (struct disassemble_info *info, const struct arc_opcode *opcode)
170
208k
{
171
  /* Check opcode for major 0x06, return if it is not in.  */
172
208k
  if (arc_opcode_len (opcode) == 4
173
208k
      && (OPCODE_32BIT_INSN (opcode->opcode) != 0x06
174
    /* Can be an APEX extensions.  */
175
195k
    && OPCODE_32BIT_INSN (opcode->opcode) != 0x07))
176
191k
    return false;
177
178
  /* or not a known truble class.  */
179
16.9k
  switch (opcode->insn_class)
180
16.9k
    {
181
8.84k
    case FLOAT:
182
11.7k
    case DSP:
183
13.7k
    case ARITH:
184
14.1k
    case MPY:
185
14.1k
      break;
186
2.79k
    default:
187
2.79k
      return false;
188
16.9k
    }
189
190
14.1k
  struct arc_disassemble_info *arc_infop = info->private_data;
191
17.8k
  for (unsigned int i = 0; i < arc_infop->decode_count; i++)
192
4.45k
    if (arc_infop->decode[i].insn_class == opcode->insn_class
193
2.35k
  && arc_infop->decode[i].subclass == opcode->subclass)
194
819
      return false;
195
196
13.3k
  return true;
197
14.1k
}
198
199
static bfd_vma
200
bfd_getm32 (unsigned int data)
201
249k
{
202
249k
  bfd_vma value = 0;
203
204
249k
  value = ((data & 0xff00) | (data & 0xff)) << 16;
205
249k
  value |= ((data & 0xff0000) | (data & 0xff000000)) >> 16;
206
249k
  return value;
207
249k
}
208
209
static bool
210
special_flag_p (const char *opname,
211
    const char *flgname)
212
133k
{
213
133k
  const struct arc_flag_special *flg_spec;
214
133k
  unsigned i, j, flgidx;
215
216
1.12M
  for (i = 0; i < arc_num_flag_special; i++)
217
1.02M
    {
218
1.02M
      flg_spec = &arc_flag_special_cases[i];
219
220
1.02M
      if (strcmp (opname, flg_spec->name))
221
933k
  continue;
222
223
      /* Found potential special case instruction.  */
224
1.27M
      for (j=0;; ++j)
225
1.37M
  {
226
1.37M
    flgidx = flg_spec->flags[j];
227
1.37M
    if (flgidx == 0)
228
57.8k
      break; /* End of the array.  */
229
230
1.31M
    if (strcmp (flgname, arc_flag_operands[flgidx].name) == 0)
231
34.4k
      return true;
232
1.31M
  }
233
92.3k
    }
234
99.0k
  return false;
235
133k
}
236
237
/* Find opcode from ARC_TABLE given the instruction described by INSN and
238
   INSNLEN.  The ISA_MASK restricts the possible matches in ARC_TABLE.  */
239
240
static const struct arc_opcode *
241
find_format_from_table (struct disassemble_info *info,
242
      const struct arc_opcode *arc_table,
243
                        unsigned long long insn,
244
      unsigned int insn_len,
245
                        unsigned isa_mask,
246
      bool *has_limm,
247
      bool overlaps)
248
648k
{
249
648k
  unsigned int i = 0;
250
648k
  const struct arc_opcode *opcode = NULL;
251
648k
  const struct arc_opcode *t_op = NULL;
252
648k
  const unsigned char *opidx;
253
648k
  const unsigned char *flgidx;
254
648k
  bool warn_p = false;
255
256
648k
  do
257
1.85G
    {
258
1.85G
      bool invalid = false;
259
260
1.85G
      opcode = &arc_table[i++];
261
262
1.85G
      if (!(opcode->cpu & isa_mask))
263
955M
  continue;
264
265
901M
      if (arc_opcode_len (opcode) != (int) insn_len)
266
505M
  continue;
267
268
395M
      if ((insn & opcode->mask) != opcode->opcode)
269
394M
  continue;
270
271
604k
      *has_limm = false;
272
273
      /* Possible candidate, check the operands.  */
274
2.44M
      for (opidx = opcode->operands; *opidx; opidx++)
275
1.85M
  {
276
1.85M
    int value, limmind;
277
1.85M
    const struct arc_operand *operand = &arc_operands[*opidx];
278
279
1.85M
    if (operand->flags & ARC_OPERAND_FAKE)
280
394k
      continue;
281
282
1.45M
    if (operand->extract)
283
1.22M
      value = (*operand->extract) (insn, &invalid);
284
232k
    else
285
232k
      value = (insn >> operand->shift) & ((1ull << operand->bits) - 1);
286
287
    /* Check for LIMM indicator.  If it is there, then make sure
288
       we pick the right format.  */
289
1.45M
    limmind = (isa_mask & ARC_OPCODE_ARCV2) ? 0x1E : 0x3E;
290
1.45M
    if (operand->flags & ARC_OPERAND_IR
291
789k
        && !(operand->flags & ARC_OPERAND_LIMM))
292
789k
      {
293
789k
        if ((value == 0x3E && insn_len == 4)
294
778k
      || (value == limmind && insn_len == 2))
295
11.9k
    {
296
11.9k
      invalid = true;
297
11.9k
      break;
298
11.9k
    }
299
789k
      }
300
301
1.44M
    if (operand->flags & ARC_OPERAND_LIMM
302
9.55k
        && !(operand->flags & ARC_OPERAND_DUPLICATE))
303
7.36k
      *has_limm = true;
304
1.44M
  }
305
306
      /* Check the flags.  */
307
1.07M
      for (flgidx = opcode->flags; *flgidx; flgidx++)
308
479k
  {
309
    /* Get a valid flag class.  */
310
479k
    const struct arc_flag_class *cl_flags = &arc_flag_classes[*flgidx];
311
479k
    const unsigned *flgopridx;
312
479k
    int foundA = 0, foundB = 0;
313
479k
    unsigned int value;
314
315
    /* Check first the extensions.  */
316
479k
    if (cl_flags->flag_class & F_CLASS_EXTEND)
317
84.7k
      {
318
84.7k
        value = (insn & 0x1F);
319
84.7k
        if (arcExtMap_condCodeName (value))
320
0
    continue;
321
84.7k
      }
322
323
    /* Check for the implicit flags.  */
324
479k
    if (cl_flags->flag_class & F_CLASS_IMPLICIT)
325
108k
      continue;
326
327
3.46M
    for (flgopridx = cl_flags->flags; *flgopridx; ++flgopridx)
328
3.09M
      {
329
3.09M
        const struct arc_flag_operand *flg_operand =
330
3.09M
    &arc_flag_operands[*flgopridx];
331
332
3.09M
        value = (insn >> flg_operand->shift)
333
3.09M
    & ((1 << flg_operand->bits) - 1);
334
3.09M
        if (value == flg_operand->code)
335
431k
    foundA = 1;
336
3.09M
        if (value)
337
1.43M
    foundB = 1;
338
3.09M
      }
339
340
370k
    if (!foundA && foundB)
341
7.29k
      {
342
7.29k
        invalid = true;
343
7.29k
        break;
344
7.29k
      }
345
370k
  }
346
347
604k
      if (invalid)
348
25.5k
  continue;
349
350
578k
      if (insn_len == 4
351
208k
    && overlaps)
352
208k
  {
353
208k
    warn_p = true;
354
208k
    t_op = opcode;
355
208k
    if (skip_this_opcode (info, opcode))
356
13.3k
      continue;
357
208k
  }
358
359
      /* The instruction is valid.  */
360
565k
      return opcode;
361
578k
    }
362
1.85G
  while (opcode->mask);
363
364
83.3k
  if (warn_p)
365
11.2k
    {
366
11.2k
      info->fprintf_styled_func
367
11.2k
  (info->stream, dis_style_text,
368
11.2k
   _("\nWarning: disassembly may be wrong due to "
369
11.2k
     "guessed opcode class choice.\n"
370
11.2k
     "Use -M<class[,class]> to select the correct "
371
11.2k
     "opcode class(es).\n\t\t\t\t"));
372
11.2k
      return t_op;
373
11.2k
    }
374
375
72.0k
  return NULL;
376
83.3k
}
377
378
/* Find opcode for INSN, trying various different sources.  The instruction
379
   length in INSN_LEN will be updated if the instruction requires a LIMM
380
   extension.
381
382
   A pointer to the opcode is placed into OPCODE_RESULT, and ITER is
383
   initialised, ready to iterate over the operands of the found opcode.  If
384
   the found opcode requires a LIMM then the LIMM value will be loaded into a
385
   field of ITER.
386
387
   This function returns TRUE in almost all cases, FALSE is reserved to
388
   indicate an error (failing to find an opcode is not an error) a returned
389
   result of FALSE would indicate that the disassembler can't continue.
390
391
   If no matching opcode is found then the returned result will be TRUE, the
392
   value placed into OPCODE_RESULT will be NULL, ITER will be undefined, and
393
   INSN_LEN will be unchanged.
394
395
   If a matching opcode is found, then the returned result will be TRUE, the
396
   opcode pointer is placed into OPCODE_RESULT, INSN_LEN will be increased by
397
   4 if the instruction requires a LIMM, and the LIMM value will have been
398
   loaded into a field of ITER.  Finally, ITER will have been initialised so
399
   that calls to OPERAND_ITERATOR_NEXT will iterate over the opcode's
400
   operands.  */
401
402
static bool
403
find_format (bfd_vma                       memaddr,
404
       unsigned long long            insn,
405
       unsigned int *                insn_len,
406
             unsigned                      isa_mask,
407
       struct disassemble_info *     info,
408
             const struct arc_opcode **    opcode_result,
409
             struct arc_operand_iterator * iter)
410
648k
{
411
648k
  const struct arc_opcode *opcode = NULL;
412
648k
  bool needs_limm = false;
413
648k
  const extInstruction_t *einsn, *i;
414
648k
  unsigned limm = 0;
415
648k
  struct arc_disassemble_info *arc_infop = info->private_data;
416
417
  /* First, try the extension instructions.  */
418
648k
  if (*insn_len == 4)
419
263k
    {
420
263k
      einsn = arcExtMap_insn (OPCODE_32BIT_INSN (insn), insn);
421
263k
      for (i = einsn; (i != NULL) && (opcode == NULL); i = i->next)
422
0
  {
423
0
    const char *errmsg = NULL;
424
425
0
    opcode = arcExtMap_genOpcode (i, isa_mask, &errmsg);
426
0
    if (opcode == NULL)
427
0
      {
428
0
        (*info->fprintf_styled_func)
429
0
    (info->stream, dis_style_text,
430
0
     _("An error occurred while generating "
431
0
       "the extension instruction operations"));
432
0
        *opcode_result = NULL;
433
0
        return false;
434
0
      }
435
436
0
    opcode = find_format_from_table (info, opcode, insn, *insn_len,
437
0
             isa_mask, &needs_limm, false);
438
0
  }
439
263k
    }
440
441
  /* Then, try finding the first match in the opcode table.  */
442
648k
  if (opcode == NULL)
443
648k
    opcode = find_format_from_table (info, arc_opcodes, insn, *insn_len,
444
648k
             isa_mask, &needs_limm, true);
445
446
648k
  if (opcode != NULL && needs_limm)
447
4.61k
    {
448
4.61k
      bfd_byte buffer[4];
449
4.61k
      int status;
450
451
4.61k
      status = (*info->read_memory_func) (memaddr + *insn_len, buffer,
452
4.61k
                                          4, info);
453
4.61k
      if (status != 0)
454
170
        {
455
170
          opcode = NULL;
456
170
        }
457
4.44k
      else
458
4.44k
        {
459
4.44k
          limm = ARRANGE_ENDIAN (info, buffer);
460
4.44k
          *insn_len += 4;
461
4.44k
        }
462
4.61k
    }
463
464
648k
  if (opcode != NULL)
465
576k
    {
466
576k
      iter->insn = insn;
467
576k
      iter->limm = limm;
468
576k
      iter->opcode = opcode;
469
576k
      iter->opidx = opcode->operands;
470
576k
    }
471
472
648k
  *opcode_result = opcode;
473
474
  /* Update private data.  */
475
648k
  arc_infop->opcode = opcode;
476
648k
  arc_infop->limm = limm;
477
648k
  arc_infop->limm_p = needs_limm;
478
479
648k
  return true;
480
648k
}
481
482
static void
483
print_flags (const struct arc_opcode *opcode,
484
       unsigned long long *insn,
485
       struct disassemble_info *info)
486
576k
{
487
576k
  const unsigned char *flgidx;
488
576k
  unsigned int value;
489
576k
  struct arc_disassemble_info *arc_infop = info->private_data;
490
491
  /* Now extract and print the flags.  */
492
1.03M
  for (flgidx = opcode->flags; *flgidx; flgidx++)
493
456k
    {
494
      /* Get a valid flag class.  */
495
456k
      const struct arc_flag_class *cl_flags = &arc_flag_classes[*flgidx];
496
456k
      const unsigned *flgopridx;
497
498
      /* Check first the extensions.  */
499
456k
      if (cl_flags->flag_class & F_CLASS_EXTEND)
500
82.2k
  {
501
82.2k
    const char *name;
502
82.2k
    value = (insn[0] & 0x1F);
503
504
82.2k
    name = arcExtMap_condCodeName (value);
505
82.2k
    if (name)
506
0
      {
507
0
        (*info->fprintf_styled_func) (info->stream, dis_style_mnemonic,
508
0
              ".%s", name);
509
0
        continue;
510
0
      }
511
82.2k
  }
512
513
3.54M
      for (flgopridx = cl_flags->flags; *flgopridx; ++flgopridx)
514
3.08M
  {
515
3.08M
    const struct arc_flag_operand *flg_operand =
516
3.08M
      &arc_flag_operands[*flgopridx];
517
518
    /* Implicit flags are only used for the insn decoder.  */
519
3.08M
    if (cl_flags->flag_class & F_CLASS_IMPLICIT)
520
108k
      {
521
108k
        if (cl_flags->flag_class & F_CLASS_COND)
522
31.3k
    arc_infop->condition_code = flg_operand->code;
523
76.9k
        else if (cl_flags->flag_class & F_CLASS_WB)
524
3.09k
    arc_infop->writeback_mode = flg_operand->code;
525
73.8k
        else if (cl_flags->flag_class & F_CLASS_ZZ)
526
73.8k
    info->data_size = flg_operand->code;
527
108k
        continue;
528
108k
      }
529
530
2.97M
    if (!flg_operand->favail)
531
1.15M
      continue;
532
533
1.82M
    value = (insn[0] >> flg_operand->shift)
534
1.82M
      & ((1 << flg_operand->bits) - 1);
535
1.82M
    if (value == flg_operand->code)
536
133k
      {
537
         /* FIXME!: print correctly nt/t flag.  */
538
133k
        if (!special_flag_p (opcode->name, flg_operand->name))
539
99.0k
    (*info->fprintf_styled_func) (info->stream,
540
99.0k
                dis_style_mnemonic, ".");
541
34.4k
        else if (info->insn_type == dis_dref)
542
12.7k
    {
543
12.7k
      switch (flg_operand->name[0])
544
12.7k
        {
545
5.19k
        case 'b':
546
5.19k
          info->data_size = 1;
547
5.19k
          break;
548
7.57k
        case 'h':
549
7.57k
        case 'w':
550
7.57k
          info->data_size = 2;
551
7.57k
          break;
552
0
        default:
553
0
          info->data_size = 4;
554
0
          break;
555
12.7k
        }
556
12.7k
    }
557
133k
        if (flg_operand->name[0] == 'd'
558
38.7k
      && flg_operand->name[1] == 0)
559
29.9k
    info->branch_delay_insns = 1;
560
561
        /* Check if it is a conditional flag.  */
562
133k
        if (cl_flags->flag_class & F_CLASS_COND)
563
25.2k
    {
564
25.2k
      if (info->insn_type == dis_jsr)
565
3.76k
        info->insn_type = dis_condjsr;
566
21.4k
      else if (info->insn_type == dis_branch)
567
18.1k
        info->insn_type = dis_condbranch;
568
25.2k
      arc_infop->condition_code = flg_operand->code;
569
25.2k
    }
570
571
        /* Check for the write back modes.  */
572
133k
        if (cl_flags->flag_class & F_CLASS_WB)
573
15.5k
    arc_infop->writeback_mode = flg_operand->code;
574
575
133k
        (*info->fprintf_styled_func) (info->stream, dis_style_mnemonic,
576
133k
              "%s", flg_operand->name);
577
133k
      }
578
1.82M
  }
579
456k
    }
580
576k
}
581
582
static const char *
583
get_auxreg (const struct arc_opcode *opcode,
584
      int value,
585
      unsigned isa_mask)
586
607k
{
587
607k
  const char *name;
588
607k
  unsigned int i;
589
607k
  const struct arc_aux_reg *auxr = &arc_aux_regs[0];
590
591
607k
  if (opcode->insn_class != AUXREG)
592
601k
    return NULL;
593
594
6.14k
  name = arcExtMap_auxRegName (value);
595
6.14k
  if (name)
596
0
    return name;
597
598
924k
  for (i = 0; i < arc_num_aux_regs; i++, auxr++)
599
923k
    {
600
923k
      if (!(auxr->cpu & isa_mask))
601
348k
  continue;
602
603
575k
      if (auxr->subclass != NONE)
604
1.36k
  return NULL;
605
606
573k
      if (auxr->address == value)
607
3.68k
  return auxr->name;
608
573k
    }
609
1.09k
  return NULL;
610
6.14k
}
611
612
/* Convert a value representing an address type to a string used to refer to
613
   the address type in assembly code.  */
614
615
static const char *
616
get_addrtype (unsigned int value)
617
3.88k
{
618
3.88k
  if (value >= ARC_NUM_ADDRTYPES)
619
0
    return "unknown";
620
621
3.88k
  return addrtypenames[value];
622
3.88k
}
623
624
/* Calculate the instruction length for an instruction starting with MSB
625
   and LSB, the most and least significant byte.  The ISA_MASK is used to
626
   filter the instructions considered to only those that are part of the
627
   current architecture.
628
629
   The instruction lengths are calculated from the ARC_OPCODE table, and
630
   cached for later use.  */
631
632
static unsigned int
633
arc_insn_length (bfd_byte msb, bfd_byte lsb, struct disassemble_info *info)
634
649k
{
635
649k
  bfd_byte major_opcode = msb >> 3;
636
637
649k
  switch (info->mach)
638
649k
    {
639
364k
    case bfd_mach_arc_arc700:
640
      /* The nps400 extension set requires this special casing of the
641
   instruction length calculation.  Right now this is not causing any
642
   problems as none of the known extensions overlap in opcode space,
643
   but, if they ever do then we might need to start carrying
644
   information around in the elf about which extensions are in use.  */
645
364k
      if (major_opcode == 0xb)
646
19.8k
        {
647
19.8k
          bfd_byte minor_opcode = lsb & 0x1f;
648
649
19.8k
    if (minor_opcode < 4)
650
4.02k
      return 6;
651
15.8k
    else if (minor_opcode == 0x10 || minor_opcode == 0x11)
652
4.09k
      return 8;
653
19.8k
        }
654
356k
      if (major_opcode == 0xa)
655
5.09k
        {
656
5.09k
          return 8;
657
5.09k
        }
658
      /* Fall through.  */
659
389k
    case bfd_mach_arc_arc600:
660
389k
      return (major_opcode > 0xb) ? 2 : 4;
661
0
      break;
662
663
246k
    case bfd_mach_arc_arcv2:
664
246k
      return (major_opcode > 0x7) ? 2 : 4;
665
0
      break;
666
667
2
    default:
668
2
      return 0;
669
649k
    }
670
649k
}
671
672
/* Extract and return the value of OPERAND from the instruction whose value
673
   is held in the array INSN.  */
674
675
static int
676
extract_operand_value (const struct arc_operand *operand,
677
           unsigned long long insn,
678
           unsigned limm)
679
1.72M
{
680
1.72M
  int value;
681
682
  /* Read the limm operand, if required.  */
683
1.72M
  if (operand->flags & ARC_OPERAND_LIMM)
684
    /* The second part of the instruction value will have been loaded as
685
       part of the find_format call made earlier.  */
686
6.37k
    value = limm;
687
1.72M
  else
688
1.72M
    {
689
1.72M
      if (operand->extract)
690
1.15M
  value = (*operand->extract) (insn, (bool *) NULL);
691
562k
      else
692
562k
        {
693
562k
          if (operand->flags & ARC_OPERAND_ALIGNED32)
694
0
            {
695
0
              value = (insn >> operand->shift)
696
0
                & ((1 << (operand->bits - 2)) - 1);
697
0
              value = value << 2;
698
0
            }
699
562k
          else
700
562k
            {
701
562k
              value = (insn >> operand->shift) & ((1 << operand->bits) - 1);
702
562k
            }
703
562k
          if (operand->flags & ARC_OPERAND_SIGNED)
704
1.72k
            {
705
1.72k
              int signbit = 1 << (operand->bits - 1);
706
1.72k
              value = (value ^ signbit) - signbit;
707
1.72k
            }
708
562k
        }
709
1.72M
    }
710
711
1.72M
  return value;
712
1.72M
}
713
714
/* Find the next operand, and the operands value from ITER.  Return TRUE if
715
   there is another operand, otherwise return FALSE.  If there is an
716
   operand returned then the operand is placed into OPERAND, and the value
717
   into VALUE.  If there is no operand returned then OPERAND and VALUE are
718
   unchanged.  */
719
720
static bool
721
operand_iterator_next (struct arc_operand_iterator *iter,
722
                       const struct arc_operand **operand,
723
                       int *value)
724
2.30M
{
725
2.30M
  if (*iter->opidx == 0)
726
576k
    {
727
576k
      *operand = NULL;
728
576k
      return false;
729
576k
    }
730
731
1.72M
  *operand = &arc_operands[*iter->opidx];
732
1.72M
  *value = extract_operand_value (*operand, iter->insn, iter->limm);
733
1.72M
  iter->opidx++;
734
735
1.72M
  return true;
736
2.30M
}
737
738
/* Helper for parsing the options.  */
739
740
static void
741
parse_option (struct arc_disassemble_info *arc_infop, const char *option)
742
14.5k
{
743
14.5k
  if (strcmp (option, "dsp") == 0)
744
392
    add_to_decode (arc_infop, DSP, NONE);
745
746
14.1k
  else if (strcmp (option, "spfp") == 0)
747
344
    add_to_decode (arc_infop, FLOAT, SPX);
748
749
13.7k
  else if (strcmp (option, "dpfp") == 0)
750
396
    add_to_decode (arc_infop, FLOAT, DPX);
751
752
13.3k
  else if (strcmp (option, "quarkse_em") == 0)
753
280
    {
754
280
      add_to_decode (arc_infop, FLOAT, DPX);
755
280
      add_to_decode (arc_infop, FLOAT, SPX);
756
280
      add_to_decode (arc_infop, FLOAT, QUARKSE1);
757
280
      add_to_decode (arc_infop, FLOAT, QUARKSE2);
758
280
    }
759
760
13.1k
  else if (strcmp (option, "fpuda") == 0)
761
292
    add_to_decode (arc_infop, FLOAT, DPA);
762
763
12.8k
  else if (strcmp (option, "nps400") == 0)
764
836
    {
765
836
      add_to_decode (arc_infop, ACL, NPS400);
766
836
      add_to_decode (arc_infop, ARITH, NPS400);
767
836
      add_to_decode (arc_infop, BITOP, NPS400);
768
836
      add_to_decode (arc_infop, BMU, NPS400);
769
836
      add_to_decode (arc_infop, CONTROL, NPS400);
770
836
      add_to_decode (arc_infop, DMA, NPS400);
771
836
      add_to_decode (arc_infop, DPI, NPS400);
772
836
      add_to_decode (arc_infop, MEMORY, NPS400);
773
836
      add_to_decode (arc_infop, MISC, NPS400);
774
836
      add_to_decode (arc_infop, NET, NPS400);
775
836
      add_to_decode (arc_infop, PMU, NPS400);
776
836
      add_to_decode (arc_infop, PROTOCOL_DECODE, NPS400);
777
836
      add_to_decode (arc_infop, ULTRAIP, NPS400);
778
836
    }
779
780
11.9k
  else if (strcmp (option, "fpus") == 0)
781
348
    {
782
348
      add_to_decode (arc_infop, FLOAT, SP);
783
348
      add_to_decode (arc_infop, FLOAT, CVT);
784
348
    }
785
786
11.6k
  else if (strcmp (option, "fpud") == 0)
787
388
    {
788
388
      add_to_decode (arc_infop, FLOAT, DP);
789
388
      add_to_decode (arc_infop, FLOAT, CVT);
790
388
    }
791
11.2k
  else if (startswith (option, "hex"))
792
214
    arc_infop->print_hex = true;
793
11.0k
  else
794
    /* xgettext:c-format */
795
11.0k
    opcodes_error_handler (_("unrecognised disassembler option: %s"), option);
796
14.5k
}
797
798
#define ARC_CPU_TYPE_A6xx(NAME,EXTRA)     \
799
  { #NAME, ARC_OPCODE_ARC600, "ARC600" }
800
#define ARC_CPU_TYPE_A7xx(NAME,EXTRA)     \
801
  { #NAME, ARC_OPCODE_ARC700, "ARC700" }
802
#define ARC_CPU_TYPE_AV2EM(NAME,EXTRA)      \
803
  { #NAME,  ARC_OPCODE_ARCv2EM, "ARC EM" }
804
#define ARC_CPU_TYPE_AV2HS(NAME,EXTRA)      \
805
  { #NAME,  ARC_OPCODE_ARCv2HS, "ARC HS" }
806
#define ARC_CPU_TYPE_NONE       \
807
  { 0, 0, 0 }
808
809
/* A table of CPU names and opcode sets.  */
810
static const struct cpu_type
811
{
812
  const char *name;
813
  unsigned flags;
814
  const char *isa;
815
}
816
  cpu_types[] =
817
{
818
  #include "elf/arc-cpu.def"
819
};
820
821
/* Helper for parsing the CPU options.  Accept any of the ARC architectures
822
   values.  OPTION should be a value passed to cpu=.  */
823
824
static unsigned
825
parse_cpu_option (const char *option)
826
1.05k
{
827
1.05k
  int i;
828
829
24.0k
  for (i = 0; cpu_types[i].name; ++i)
830
23.2k
    if (strcmp (cpu_types[i].name, option) == 0)
831
266
      return cpu_types[i].flags;
832
833
  /* xgettext:c-format */
834
786
  opcodes_error_handler (_("unrecognised disassembler CPU option: %s"), option);
835
786
  return ARC_OPCODE_NONE;
836
1.05k
}
837
838
static bool
839
arc_parse_option (const char *option, void *data)
840
15.5k
{
841
15.5k
  struct arc_disassemble_info *arc_infop = data;
842
843
15.5k
  if (strncmp (option, "cpu=", 4) == 0)
844
    /* Strip leading `cpu=`.  */
845
1.05k
    arc_infop->isa_mask = parse_cpu_option (option + 4);
846
14.5k
  else
847
14.5k
    parse_option (arc_infop, option);
848
15.5k
  return true;
849
15.5k
}
850
851
/* Go over the options list and parse it.  */
852
853
static void
854
parse_disassembler_options (struct disassemble_info *info)
855
3.58k
{
856
3.58k
  struct arc_disassemble_info *arc_infop = info->private_data;
857
858
3.58k
  arc_infop->isa_mask = ARC_OPCODE_NONE;
859
860
3.58k
  for_each_disassembler_option (info, arc_parse_option, arc_infop);
861
862
  /* Figure out CPU type, unless it was enforced via disassembler options.  */
863
3.58k
  if (arc_infop->isa_mask == ARC_OPCODE_NONE)
864
3.37k
    {
865
3.37k
      switch (info->mach)
866
3.37k
  {
867
1.65k
  case bfd_mach_arc_arc700:
868
1.65k
    arc_infop->isa_mask = ARC_OPCODE_ARC700;
869
1.65k
    break;
870
871
243
  case bfd_mach_arc_arc600:
872
243
    arc_infop->isa_mask = ARC_OPCODE_ARC600;
873
243
    break;
874
875
1.13k
  case bfd_mach_arc_arcv2:
876
1.47k
  default:
877
1.47k
    {
878
1.47k
      Elf_Internal_Ehdr *header = NULL;
879
880
1.47k
      if (info->section && info->section->owner)
881
33
        header = elf_elfheader (info->section->owner);
882
1.47k
      arc_infop->isa_mask = ARC_OPCODE_ARCv2EM;
883
1.47k
      if (header != NULL
884
33
    && (header->e_flags & EF_ARC_MACH_MSK) == EF_ARC_CPU_ARCV2HS)
885
19
        arc_infop->isa_mask = ARC_OPCODE_ARCv2HS;
886
1.47k
    }
887
1.47k
    break;
888
3.37k
  }
889
3.37k
    }
890
891
3.58k
  if (arc_infop->isa_mask == ARC_OPCODE_ARCv2HS)
892
211
    {
893
      /* FPU instructions are not extensions for HS.  */
894
211
      add_to_decode (arc_infop, FLOAT, SP);
895
211
      add_to_decode (arc_infop, FLOAT, DP);
896
211
      add_to_decode (arc_infop, FLOAT, CVT);
897
211
    }
898
3.58k
}
899
900
/* Return the instruction type for an instruction described by OPCODE.  */
901
902
static enum dis_insn_type
903
arc_opcode_to_insn_type (const struct arc_opcode *opcode)
904
576k
{
905
576k
  enum dis_insn_type insn_type;
906
907
576k
  switch (opcode->insn_class)
908
576k
    {
909
189k
    case BRANCH:
910
191k
    case BBIT0:
911
195k
    case BBIT1:
912
196k
    case BI:
913
197k
    case BIH:
914
218k
    case BRCC:
915
218k
    case DBNZ:
916
227k
    case EI:
917
234k
    case JLI:
918
237k
    case JUMP:
919
238k
    case LOOP:
920
238k
      if (!strncmp (opcode->name, "bl", 2)
921
159k
    || !strncmp (opcode->name, "jl", 2))
922
85.5k
  {
923
85.5k
    if (opcode->subclass == COND)
924
2.34k
      insn_type = dis_condjsr;
925
83.1k
    else
926
83.1k
      insn_type = dis_jsr;
927
85.5k
  }
928
152k
      else
929
152k
  {
930
152k
    if (opcode->subclass == COND)
931
34.2k
      insn_type = dis_condbranch;
932
118k
    else
933
118k
      insn_type = dis_branch;
934
152k
  }
935
238k
      break;
936
114k
    case LOAD:
937
165k
    case STORE:
938
167k
    case MEMORY:
939
168k
    case ENTER:
940
168k
    case PUSH:
941
171k
    case POP:
942
171k
      insn_type = dis_dref;
943
171k
      break;
944
2.56k
    case LEAVE:
945
2.56k
      insn_type = dis_branch;
946
2.56k
      break;
947
164k
    default:
948
164k
      insn_type = dis_nonbranch;
949
164k
      break;
950
576k
    }
951
952
576k
  return insn_type;
953
576k
}
954
955
/* Disassemble ARC instructions.  */
956
957
static int
958
print_insn_arc (bfd_vma memaddr,
959
    struct disassemble_info *info)
960
650k
{
961
650k
  bfd_byte buffer[8];
962
650k
  unsigned int highbyte, lowbyte;
963
650k
  int status;
964
650k
  unsigned int insn_len;
965
650k
  unsigned long long insn = 0;
966
650k
  const struct arc_opcode *opcode;
967
650k
  bool need_comma;
968
650k
  bool open_braket;
969
650k
  int size;
970
650k
  const struct arc_operand *operand;
971
650k
  int value, vpcl;
972
650k
  struct arc_operand_iterator iter;
973
650k
  struct arc_disassemble_info *arc_infop;
974
650k
  bool rpcl = false, rset = false;
975
976
650k
  if (info->private_data == NULL)
977
3.58k
    {
978
3.58k
      if (!init_arc_disasm_info (info))
979
0
  return -1;
980
981
3.58k
      parse_disassembler_options (info);
982
3.58k
    }
983
984
650k
  memset (&iter, 0, sizeof (iter));
985
650k
  highbyte = info->endian == BFD_ENDIAN_LITTLE ? 1 : 0;
986
650k
  lowbyte = info->endian == BFD_ENDIAN_LITTLE ? 0 : 1;
987
988
  /* This variable may be set by the instruction decoder.  It suggests
989
     the number of bytes objdump should display on a single line.  If
990
     the instruction decoder sets this, it should always set it to
991
     the same value in order to get reasonable looking output.  */
992
650k
  info->bytes_per_line  = 8;
993
994
  /* In the next lines, we set two info variables control the way
995
     objdump displays the raw data.  For example, if bytes_per_line is
996
     8 and bytes_per_chunk is 4, the output will look like this:
997
     00:   00000000 00000000
998
     with the chunks displayed according to "display_endian".  */
999
650k
  if (info->section
1000
82.8k
      && !(info->section->flags & SEC_CODE))
1001
72
    {
1002
      /* This is not a CODE section.  */
1003
72
      switch (info->section->size)
1004
72
  {
1005
0
  case 1:
1006
0
  case 2:
1007
0
  case 4:
1008
0
    size = info->section->size;
1009
0
    break;
1010
72
  default:
1011
72
    size = (info->section->size & 0x01) ? 1 : 4;
1012
72
    break;
1013
72
  }
1014
72
      info->bytes_per_chunk = 1;
1015
72
      info->display_endian = info->endian;
1016
72
    }
1017
650k
  else
1018
650k
    {
1019
650k
      size = 2;
1020
650k
      info->bytes_per_chunk = 2;
1021
650k
      info->display_endian = info->endian;
1022
650k
    }
1023
1024
  /* Read the insn into a host word.  */
1025
650k
  status = (*info->read_memory_func) (memaddr, buffer, size, info);
1026
1027
650k
  if (status != 0)
1028
956
    {
1029
956
      (*info->memory_error_func) (status, memaddr, info);
1030
956
      return -1;
1031
956
    }
1032
1033
649k
  if (info->section
1034
82.8k
      && !(info->section->flags & SEC_CODE))
1035
72
    {
1036
      /* Data section.  */
1037
72
      unsigned long data;
1038
1039
72
      data = bfd_get_bits (buffer, size * 8,
1040
72
         info->display_endian == BFD_ENDIAN_BIG);
1041
72
      switch (size)
1042
72
  {
1043
3
  case 1:
1044
3
    (*info->fprintf_styled_func) (info->stream,
1045
3
          dis_style_assembler_directive,
1046
3
          ".byte");
1047
3
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1048
3
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1049
3
          "0x%02lx", data);
1050
3
    break;
1051
0
  case 2:
1052
0
    (*info->fprintf_styled_func) (info->stream,
1053
0
          dis_style_assembler_directive,
1054
0
          ".short");
1055
0
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1056
0
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1057
0
          "0x%04lx", data);
1058
0
    break;
1059
69
  case 4:
1060
69
    (*info->fprintf_styled_func) (info->stream,
1061
69
          dis_style_assembler_directive,
1062
69
          ".word");
1063
69
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1064
69
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1065
69
          "0x%08lx", data);
1066
69
    break;
1067
0
  default:
1068
0
    return -1;
1069
72
  }
1070
72
      return size;
1071
72
    }
1072
1073
649k
  insn_len = arc_insn_length (buffer[highbyte], buffer[lowbyte], info);
1074
649k
  pr_debug ("instruction length = %d bytes\n", insn_len);
1075
649k
  if (insn_len == 0)
1076
2
    return -1;
1077
1078
649k
  arc_infop = info->private_data;
1079
649k
  arc_infop->insn_len = insn_len;
1080
1081
649k
  switch (insn_len)
1082
649k
    {
1083
371k
    case 2:
1084
371k
      insn = (buffer[highbyte] << 8) | buffer[lowbyte];
1085
371k
      break;
1086
1087
264k
    case 4:
1088
264k
      {
1089
  /* This is a long instruction: Read the remaning 2 bytes.  */
1090
264k
  status = (*info->read_memory_func) (memaddr + 2, &buffer[2], 2, info);
1091
264k
  if (status != 0)
1092
281
    {
1093
281
      (*info->memory_error_func) (status, memaddr + 2, info);
1094
281
      return -1;
1095
281
    }
1096
263k
  insn = (unsigned long long) ARRANGE_ENDIAN (info, buffer);
1097
263k
      }
1098
0
      break;
1099
1100
4.02k
    case 6:
1101
4.02k
      {
1102
4.02k
  status = (*info->read_memory_func) (memaddr + 2, &buffer[2], 4, info);
1103
4.02k
  if (status != 0)
1104
25
    {
1105
25
      (*info->memory_error_func) (status, memaddr + 2, info);
1106
25
      return -1;
1107
25
    }
1108
3.99k
  insn = (unsigned long long) ARRANGE_ENDIAN (info, &buffer[2]);
1109
3.99k
  insn |= ((unsigned long long) buffer[highbyte] << 40)
1110
3.99k
    | ((unsigned long long) buffer[lowbyte] << 32);
1111
3.99k
      }
1112
0
      break;
1113
1114
9.18k
    case 8:
1115
9.18k
      {
1116
9.18k
  status = (*info->read_memory_func) (memaddr + 2, &buffer[2], 6, info);
1117
9.18k
  if (status != 0)
1118
59
    {
1119
59
      (*info->memory_error_func) (status, memaddr + 2, info);
1120
59
      return -1;
1121
59
    }
1122
9.12k
  insn =
1123
9.12k
    ((((unsigned long long) ARRANGE_ENDIAN (info, buffer)) << 32)
1124
9.12k
     | ((unsigned long long) ARRANGE_ENDIAN (info, &buffer[4])));
1125
9.12k
      }
1126
0
      break;
1127
1128
0
    default:
1129
      /* There is no instruction whose length is not 2, 4, 6, or 8.  */
1130
0
      return -1;
1131
649k
    }
1132
1133
648k
  pr_debug ("instruction value = %llx\n", insn);
1134
1135
  /* Set some defaults for the insn info.  */
1136
648k
  info->insn_info_valid    = 1;
1137
648k
  info->branch_delay_insns = 0;
1138
648k
  info->data_size    = 4;
1139
648k
  info->insn_type    = dis_nonbranch;
1140
648k
  info->target       = 0;
1141
648k
  info->target2      = 0;
1142
1143
  /* FIXME to be moved in dissasemble_init_for_target.  */
1144
648k
  info->disassembler_needs_relocs = true;
1145
1146
  /* Find the first match in the opcode table.  */
1147
648k
  if (!find_format (memaddr, insn, &insn_len, arc_infop->isa_mask, info,
1148
648k
        &opcode, &iter))
1149
0
    return -1;
1150
1151
648k
  if (!opcode)
1152
72.2k
    {
1153
72.2k
      switch (insn_len)
1154
72.2k
  {
1155
9.09k
  case 2:
1156
9.09k
    (*info->fprintf_styled_func) (info->stream,
1157
9.09k
          dis_style_assembler_directive,
1158
9.09k
          ".short");
1159
9.09k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1160
9.09k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1161
9.09k
          "0x%04llx", insn & 0xffff);
1162
9.09k
    break;
1163
1164
57.6k
  case 4:
1165
57.6k
    (*info->fprintf_styled_func) (info->stream,
1166
57.6k
          dis_style_assembler_directive,
1167
57.6k
          ".word");
1168
57.6k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1169
57.6k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1170
57.6k
          "0x%08llx", insn & 0xffffffff);
1171
57.6k
    break;
1172
1173
623
  case 6:
1174
623
    (*info->fprintf_styled_func) (info->stream,
1175
623
          dis_style_assembler_directive,
1176
623
          ".long");
1177
623
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1178
623
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1179
623
               "0x%08llx", insn & 0xffffffff);
1180
623
    (*info->fprintf_styled_func) (info->stream, dis_style_text, " ");
1181
623
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1182
623
          "0x%04llx", (insn >> 32) & 0xffff);
1183
623
    break;
1184
1185
4.89k
  case 8:
1186
4.89k
    (*info->fprintf_styled_func) (info->stream,
1187
4.89k
          dis_style_assembler_directive,
1188
4.89k
          ".long");
1189
4.89k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1190
4.89k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1191
4.89k
          "0x%08llx", insn & 0xffffffff);
1192
4.89k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, " ");
1193
4.89k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1194
4.89k
          "0x%08llx", (insn >> 32));
1195
4.89k
    break;
1196
1197
0
  default:
1198
0
    return -1;
1199
72.2k
  }
1200
1201
72.2k
      info->insn_type = dis_noninsn;
1202
72.2k
      return insn_len;
1203
72.2k
    }
1204
1205
  /* Print the mnemonic.  */
1206
576k
  (*info->fprintf_styled_func) (info->stream, dis_style_mnemonic,
1207
576k
        "%s", opcode->name);
1208
1209
  /* Preselect the insn class.  */
1210
576k
  info->insn_type = arc_opcode_to_insn_type (opcode);
1211
1212
576k
  pr_debug ("%s: 0x%08llx\n", opcode->name, opcode->opcode);
1213
1214
576k
  print_flags (opcode, &insn, info);
1215
1216
576k
  if (opcode->operands[0] != 0)
1217
575k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "\t");
1218
1219
576k
  need_comma = false;
1220
576k
  open_braket = false;
1221
576k
  arc_infop->operands_count = 0;
1222
1223
  /* Now extract and print the operands.  */
1224
576k
  operand = NULL;
1225
576k
  vpcl = 0;
1226
2.30M
  while (operand_iterator_next (&iter, &operand, &value))
1227
1.72M
    {
1228
1.72M
      if (open_braket && (operand->flags & ARC_OPERAND_BRAKET))
1229
185k
  {
1230
185k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "]");
1231
185k
    open_braket = false;
1232
185k
    continue;
1233
185k
  }
1234
1235
      /* Only take input from real operands.  */
1236
1.54M
      if (ARC_OPERAND_IS_FAKE (operand))
1237
0
  continue;
1238
1239
1.54M
      if ((operand->flags & ARC_OPERAND_IGNORE)
1240
144k
    && (operand->flags & ARC_OPERAND_IR)
1241
8.54k
    && value == -1)
1242
5.49k
  continue;
1243
1244
1.53M
      if (operand->flags & ARC_OPERAND_COLON)
1245
3.88k
  {
1246
3.88k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, ":");
1247
3.88k
    continue;
1248
3.88k
  }
1249
1250
1.53M
      if (need_comma)
1251
767k
  (*info->fprintf_styled_func) (info->stream, dis_style_text,",");
1252
1253
1.53M
      if (!open_braket && (operand->flags & ARC_OPERAND_BRAKET))
1254
185k
  {
1255
185k
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "[");
1256
185k
    open_braket = true;
1257
185k
    need_comma = false;
1258
185k
    continue;
1259
185k
  }
1260
1261
1.34M
      need_comma = true;
1262
1263
1.34M
      if (operand->flags & ARC_OPERAND_PCREL)
1264
229k
  {
1265
229k
    rpcl = true;
1266
229k
    vpcl = value;
1267
229k
    rset = true;
1268
1269
229k
    info->target = (bfd_vma) (memaddr & ~3) + value;
1270
229k
  }
1271
1.11M
      else if (!(operand->flags & ARC_OPERAND_IR))
1272
384k
  {
1273
384k
    vpcl = value;
1274
384k
    rset = true;
1275
384k
  }
1276
1277
      /* Print the operand as directed by the flags.  */
1278
1.34M
      if (operand->flags & ARC_OPERAND_IR)
1279
732k
  {
1280
732k
    const char *rname;
1281
1282
732k
    assert (value >=0 && value < 64);
1283
732k
    rname = arcExtMap_coreRegName (value);
1284
732k
    if (!rname)
1285
732k
      rname = regnames[value];
1286
732k
    (*info->fprintf_styled_func) (info->stream, dis_style_register,
1287
732k
          "%s", rname);
1288
1289
    /* Check if we have a double register to print.  */
1290
732k
    if (operand->flags & ARC_OPERAND_TRUNCATE)
1291
1.32k
      {
1292
1.32k
        if ((value & 0x01) == 0)
1293
755
    {
1294
755
      rname = arcExtMap_coreRegName (value + 1);
1295
755
      if (!rname)
1296
755
        rname = regnames[value + 1];
1297
755
    }
1298
567
        else
1299
567
    rname = _("\nWarning: illegal use of double register "
1300
1.32k
        "pair.\n");
1301
1.32k
        (*info->fprintf_styled_func) (info->stream, dis_style_register,
1302
1.32k
              "%s", rname);
1303
1.32k
      }
1304
732k
    if (value == 63)
1305
16.4k
      rpcl = true;
1306
716k
    else
1307
716k
      rpcl = false;
1308
732k
  }
1309
614k
      else if (operand->flags & ARC_OPERAND_LIMM)
1310
6.37k
  {
1311
6.37k
    const char *rname = get_auxreg (opcode, value, arc_infop->isa_mask);
1312
1313
6.37k
    if (rname && open_braket)
1314
870
      (*info->fprintf_styled_func) (info->stream, dis_style_register,
1315
870
            "%s", rname);
1316
5.50k
    else
1317
5.50k
      {
1318
5.50k
        (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1319
5.50k
              "%#x", value);
1320
5.50k
        if (info->insn_type == dis_branch
1321
5.10k
      || info->insn_type == dis_jsr)
1322
691
    info->target = (bfd_vma) value;
1323
5.50k
      }
1324
6.37k
  }
1325
608k
      else if (operand->flags & ARC_OPERAND_SIGNED)
1326
268k
  {
1327
268k
    const char *rname = get_auxreg (opcode, value, arc_infop->isa_mask);
1328
268k
    if (rname && open_braket)
1329
967
      (*info->fprintf_styled_func) (info->stream, dis_style_register,
1330
967
            "%s", rname);
1331
267k
    else
1332
267k
      {
1333
267k
        if (arc_infop->print_hex)
1334
3.53k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1335
3.53k
                "%#x", value);
1336
263k
        else
1337
263k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1338
263k
                "%d", value);
1339
267k
      }
1340
268k
  }
1341
339k
      else if (operand->flags & ARC_OPERAND_ADDRTYPE)
1342
3.88k
  {
1343
3.88k
    const char *addrtype = get_addrtype (value);
1344
3.88k
    (*info->fprintf_styled_func) (info->stream, dis_style_address,
1345
3.88k
          "%s", addrtype);
1346
    /* A colon follow an address type.  */
1347
3.88k
    need_comma = false;
1348
3.88k
  }
1349
336k
      else
1350
336k
  {
1351
336k
    if (operand->flags & ARC_OPERAND_TRUNCATE
1352
93.9k
        && !(operand->flags & ARC_OPERAND_ALIGNED32)
1353
42.0k
        && !(operand->flags & ARC_OPERAND_ALIGNED16)
1354
2.99k
        && value >= 0 && value <= 14)
1355
2.70k
      {
1356
        /* Leave/Enter mnemonics.  */
1357
2.70k
        switch (value)
1358
2.70k
    {
1359
897
    case 0:
1360
897
      need_comma = false;
1361
897
      break;
1362
748
    case 1:
1363
748
      (*info->fprintf_styled_func) (info->stream,
1364
748
            dis_style_register, "r13");
1365
748
      break;
1366
1.06k
    default:
1367
1.06k
      (*info->fprintf_styled_func) (info->stream,
1368
1.06k
            dis_style_register, "r13");
1369
1.06k
      (*info->fprintf_styled_func) (info->stream,
1370
1.06k
            dis_style_text, "-");
1371
1.06k
      (*info->fprintf_styled_func) (info->stream,
1372
1.06k
            dis_style_register, "%s",
1373
1.06k
            regnames[13 + value - 1]);
1374
1.06k
      break;
1375
2.70k
    }
1376
2.70k
        rpcl = false;
1377
2.70k
        rset = false;
1378
2.70k
      }
1379
333k
    else
1380
333k
      {
1381
333k
        const char *rname = get_auxreg (opcode, value,
1382
333k
                arc_infop->isa_mask);
1383
333k
        if (rname && open_braket)
1384
575
    (*info->fprintf_styled_func) (info->stream, dis_style_register,
1385
575
                "%s", rname);
1386
332k
        else
1387
332k
    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
1388
332k
                "%#x", value);
1389
333k
      }
1390
336k
  }
1391
1392
1.34M
      if (operand->flags & ARC_OPERAND_LIMM)
1393
6.37k
  {
1394
6.37k
    arc_infop->operands[arc_infop->operands_count].kind
1395
6.37k
      = ARC_OPERAND_KIND_LIMM;
1396
    /* It is not important to have exactly the LIMM indicator
1397
       here.  */
1398
6.37k
    arc_infop->operands[arc_infop->operands_count].value = 63;
1399
6.37k
  }
1400
1.34M
      else
1401
1.34M
  {
1402
1.34M
    arc_infop->operands[arc_infop->operands_count].value = value;
1403
1.34M
    arc_infop->operands[arc_infop->operands_count].kind
1404
1.34M
      = (operand->flags & ARC_OPERAND_IR
1405
1.34M
         ? ARC_OPERAND_KIND_REG
1406
1.34M
         : ARC_OPERAND_KIND_SHIMM);
1407
1.34M
  }
1408
1.34M
      arc_infop->operands_count ++;
1409
1.34M
    }
1410
1411
  /* Pretty print extra info for pc-relative operands.  */
1412
576k
  if (rpcl && rset)
1413
230k
    {
1414
230k
      if (info->flags & INSN_HAS_RELOC)
1415
  /* If the instruction has a reloc associated with it, then the
1416
     offset field in the instruction will actually be the addend
1417
     for the reloc.  (We are using REL type relocs).  In such
1418
     cases, we can ignore the pc when computing addresses, since
1419
     the addend is not currently pc-relative.  */
1420
0
  memaddr = 0;
1421
1422
230k
      (*info->fprintf_styled_func) (info->stream,
1423
230k
            dis_style_comment_start, "\t;");
1424
230k
      (*info->print_address_func) ((memaddr & ~3) + vpcl, info);
1425
230k
    }
1426
1427
576k
  return insn_len;
1428
576k
}
1429
1430
1431
disassembler_ftype
1432
arc_get_disassembler (bfd *abfd)
1433
3.58k
{
1434
  /* BFD my be absent, if opcodes is invoked from the debugger that
1435
     has connected to remote target and doesn't have an ELF file.  */
1436
3.58k
  if (abfd != NULL)
1437
100
    {
1438
      /* Read the extension insns and registers, if any.  */
1439
100
      build_ARC_extmap (abfd);
1440
#ifdef DEBUG
1441
      dump_ARC_extmap ();
1442
#endif
1443
100
    }
1444
1445
3.58k
  return print_insn_arc;
1446
3.58k
}
1447
1448
/* Indices into option argument vector for options that do require
1449
   an argument.  Use ARC_OPTION_ARG_NONE for options that don't
1450
   expect an argument.  */
1451
typedef enum
1452
{
1453
  ARC_OPTION_ARG_NONE = -1,
1454
  ARC_OPTION_ARG_ARCH,
1455
  ARC_OPTION_ARG_SIZE
1456
} arc_option_arg_t;
1457
1458
/* Valid ARC disassembler options.  */
1459
static struct
1460
{
1461
  const char *name;
1462
  const char *description;
1463
  arc_option_arg_t arg;
1464
} arc_options[] =
1465
{
1466
  { "cpu=",       N_("Enforce the designated architecture while decoding."),
1467
      ARC_OPTION_ARG_ARCH },
1468
  { "dsp",    N_("Recognize DSP instructions."),
1469
      ARC_OPTION_ARG_NONE },
1470
  { "spfp",   N_("Recognize FPX SP instructions."),
1471
      ARC_OPTION_ARG_NONE },
1472
  { "dpfp",   N_("Recognize FPX DP instructions."),
1473
      ARC_OPTION_ARG_NONE },
1474
  { "quarkse_em", N_("Recognize FPU QuarkSE-EM instructions."),
1475
      ARC_OPTION_ARG_NONE },
1476
  { "fpuda",    N_("Recognize double assist FPU instructions."),
1477
      ARC_OPTION_ARG_NONE },
1478
  { "fpus",   N_("Recognize single precision FPU instructions."),
1479
      ARC_OPTION_ARG_NONE },
1480
  { "fpud",   N_("Recognize double precision FPU instructions."),
1481
      ARC_OPTION_ARG_NONE },
1482
  { "nps400",   N_("Recognize NPS400 instructions."),
1483
      ARC_OPTION_ARG_NONE },
1484
  { "hex",    N_("Use only hexadecimal number to print immediates."),
1485
      ARC_OPTION_ARG_NONE }
1486
};
1487
1488
/* Populate the structure for representing ARC's disassembly options.
1489
   Such a dynamic initialization is desired, because it makes the maintenance
1490
   easier and also gdb uses this to enable the "disassembler-option".  */
1491
1492
const disasm_options_and_args_t *
1493
disassembler_options_arc (void)
1494
0
{
1495
0
  static disasm_options_and_args_t *opts_and_args;
1496
1497
0
  if (opts_and_args == NULL)
1498
0
    {
1499
0
      disasm_option_arg_t *args;
1500
0
      disasm_options_t *opts;
1501
0
      size_t i;
1502
0
      const size_t nr_of_options = ARRAY_SIZE (arc_options);
1503
      /* There is a null element at the end of CPU_TYPES, therefore
1504
   NR_OF_CPUS is actually 1 more and that is desired here too.  */
1505
0
      const size_t nr_of_cpus = ARRAY_SIZE (cpu_types);
1506
1507
0
      opts_and_args = XNEW (disasm_options_and_args_t);
1508
0
      opts_and_args->args
1509
0
  = XNEWVEC (disasm_option_arg_t, ARC_OPTION_ARG_SIZE + 1);
1510
0
      opts_and_args->options.name
1511
0
  = XNEWVEC (const char *, nr_of_options + 1);
1512
0
      opts_and_args->options.description
1513
0
  = XNEWVEC (const char *, nr_of_options + 1);
1514
0
      opts_and_args->options.arg
1515
0
  = XNEWVEC (const disasm_option_arg_t *, nr_of_options + 1);
1516
1517
      /* Populate the arguments for "cpu=" option.  */
1518
0
      args = opts_and_args->args;
1519
0
      args[ARC_OPTION_ARG_ARCH].name = "ARCH";
1520
0
      args[ARC_OPTION_ARG_ARCH].values = XNEWVEC (const char *, nr_of_cpus);
1521
0
      for (i = 0; i < nr_of_cpus; ++i)
1522
0
  args[ARC_OPTION_ARG_ARCH].values[i] = cpu_types[i].name;
1523
0
      args[ARC_OPTION_ARG_SIZE].name = NULL;
1524
0
      args[ARC_OPTION_ARG_SIZE].values = NULL;
1525
1526
      /* Populate the options.  */
1527
0
      opts = &opts_and_args->options;
1528
0
      for (i = 0; i < nr_of_options; ++i)
1529
0
  {
1530
0
    opts->name[i] = arc_options[i].name;
1531
0
    opts->description[i] = arc_options[i].description;
1532
0
    if (arc_options[i].arg != ARC_OPTION_ARG_NONE)
1533
0
      opts->arg[i] = &args[arc_options[i].arg];
1534
0
    else
1535
0
      opts->arg[i] = NULL;
1536
0
  }
1537
0
      opts->name[nr_of_options] = NULL;
1538
0
      opts->description[nr_of_options] = NULL;
1539
0
      opts->arg[nr_of_options] = NULL;
1540
0
    }
1541
1542
0
  return opts_and_args;
1543
0
}
1544
1545
1546
void
1547
print_arc_disassembler_options (FILE *stream)
1548
0
{
1549
0
  const disasm_options_and_args_t *opts_and_args;
1550
0
  const disasm_option_arg_t *args;
1551
0
  const disasm_options_t *opts;
1552
0
  size_t i, j;
1553
0
  size_t max_len = 0;
1554
1555
0
  opts_and_args = disassembler_options_arc ();
1556
0
  opts = &opts_and_args->options;
1557
0
  args = opts_and_args->args;
1558
1559
0
  fprintf (stream, _("\nThe following ARC specific disassembler options are"
1560
0
         " supported for use \nwith the -M switch (multiple"
1561
0
         " options should be separated by commas):\n"));
1562
1563
  /* Find the maximum length for printing options (and their arg name).  */
1564
0
  for (i = 0; opts->name[i] != NULL; ++i)
1565
0
    {
1566
0
      size_t len = strlen (opts->name[i]);
1567
0
      len += (opts->arg[i]) ? strlen (opts->arg[i]->name) : 0;
1568
0
      max_len = (len > max_len) ? len : max_len;
1569
0
    }
1570
1571
  /* Print the options, their arg and description, if any.  */
1572
0
  for (i = 0, ++max_len; opts->name[i] != NULL; ++i)
1573
0
    {
1574
0
      fprintf (stream, "  %s", opts->name[i]);
1575
0
      if (opts->arg[i] != NULL)
1576
0
  fprintf (stream, "%s", opts->arg[i]->name);
1577
0
      if (opts->description[i] != NULL)
1578
0
  {
1579
0
    size_t len = strlen (opts->name[i]);
1580
0
    len += (opts->arg[i]) ? strlen (opts->arg[i]->name) : 0;
1581
0
    fprintf (stream,
1582
0
       "%*c %s", (int) (max_len - len), ' ', opts->description[i]);
1583
0
  }
1584
0
      fprintf (stream, _("\n"));
1585
0
    }
1586
1587
  /* Print the possible values of an argument.  */
1588
0
  for (i = 0; args[i].name != NULL; ++i)
1589
0
    {
1590
0
      size_t len = 3;
1591
0
      if (args[i].values == NULL)
1592
0
  continue;
1593
0
      fprintf (stream, _("\n\
1594
0
  For the options above, the following values are supported for \"%s\":\n   "),
1595
0
         args[i].name);
1596
0
      for (j = 0; args[i].values[j] != NULL; ++j)
1597
0
  {
1598
0
    fprintf (stream, " %s", args[i].values[j]);
1599
0
    len += strlen (args[i].values[j]) + 1;
1600
    /* reset line if printed too long.  */
1601
0
    if (len >= 78)
1602
0
      {
1603
0
        fprintf (stream, _("\n   "));
1604
0
        len = 3;
1605
0
      }
1606
0
  }
1607
0
      fprintf (stream, _("\n"));
1608
0
    }
1609
1610
0
  fprintf (stream, _("\n"));
1611
0
}
1612
1613
void arc_insn_decode (bfd_vma addr,
1614
          struct disassemble_info *info,
1615
          disassembler_ftype disasm_func,
1616
          struct arc_instruction *insn)
1617
0
{
1618
0
  const struct arc_opcode *opcode;
1619
0
  struct arc_disassemble_info *arc_infop;
1620
1621
  /* Ensure that insn would be in the reset state.  */
1622
0
  memset (insn, 0, sizeof (struct arc_instruction));
1623
1624
  /* There was an error when disassembling, for example memory read error.  */
1625
0
  if (disasm_func (addr, info) < 0)
1626
0
    {
1627
0
      insn->valid = false;
1628
0
      return;
1629
0
    }
1630
1631
0
  assert (info->private_data != NULL);
1632
0
  arc_infop = info->private_data;
1633
1634
0
  insn->length  = arc_infop->insn_len;;
1635
0
  insn->address = addr;
1636
1637
  /* Quick exit if memory at this address is not an instruction.  */
1638
0
  if (info->insn_type == dis_noninsn)
1639
0
    {
1640
0
      insn->valid = false;
1641
0
      return;
1642
0
    }
1643
1644
0
  insn->valid = true;
1645
1646
0
  opcode = (const struct arc_opcode *) arc_infop->opcode;
1647
0
  insn->insn_class = opcode->insn_class;
1648
0
  insn->limm_value = arc_infop->limm;
1649
0
  insn->limm_p     = arc_infop->limm_p;
1650
1651
0
  insn->is_control_flow = (info->insn_type == dis_branch
1652
0
         || info->insn_type == dis_condbranch
1653
0
         || info->insn_type == dis_jsr
1654
0
         || info->insn_type == dis_condjsr);
1655
1656
0
  insn->has_delay_slot = info->branch_delay_insns;
1657
0
  insn->writeback_mode
1658
0
    = (enum arc_ldst_writeback_mode) arc_infop->writeback_mode;
1659
0
  insn->data_size_mode = info->data_size;
1660
0
  insn->condition_code = arc_infop->condition_code;
1661
0
  memcpy (insn->operands, arc_infop->operands,
1662
0
    sizeof (struct arc_insn_operand) * MAX_INSN_ARGS);
1663
0
  insn->operands_count = arc_infop->operands_count;
1664
0
}
1665
1666
/* Local variables:
1667
   eval: (c-set-style "gnu")
1668
   indent-tabs-mode: t
1669
   End:  */