Coverage Report

Created: 2026-04-04 08:16

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/opcodes/avr-dis.c
Line
Count
Source
1
/* Disassemble AVR instructions.
2
   Copyright (C) 1999-2026 Free Software Foundation, Inc.
3
4
   Contributed by Denis Chertykov <denisc@overta.ru>
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 <assert.h>
25
#include "disassemble.h"
26
#include "opintl.h"
27
#include "libiberty.h"
28
#include <stdint.h>
29
30
struct avr_opcodes_s
31
{
32
  char *name;
33
  char *constraints;
34
  char *opcode;
35
  int insn_size;    /* In words.  */
36
  int isa;
37
  unsigned int bin_opcode;
38
};
39
40
#define AVR_INSN(NAME, CONSTR, OPCODE, SIZE, ISA, BIN) \
41
{#NAME, CONSTR, OPCODE, SIZE, ISA, BIN},
42
43
const struct avr_opcodes_s avr_opcodes[] =
44
{
45
  #include "opcode/avr.h"
46
  {NULL, NULL, NULL, 0, 0, 0}
47
};
48
49
static const char * comment_start = "0x";
50
51
static int
52
avr_operand (unsigned int        insn,
53
       unsigned int        insn2,
54
       unsigned int        pc,
55
       int                 constraint,
56
             char *              opcode_str,
57
       char *              buf,
58
       char *              comment,
59
       enum disassembler_style *  style,
60
       int                 regs,
61
       int *               sym,
62
       bfd_vma *           sym_addr,
63
       disassemble_info *  info)
64
340k
{
65
340k
  int ok = 1;
66
340k
  *sym = 0;
67
68
340k
  switch (constraint)
69
340k
    {
70
      /* Any register operand.  */
71
119k
    case 'r':
72
119k
      if (regs)
73
39.6k
  insn = (insn & 0xf) | ((insn & 0x0200) >> 5); /* Source register.  */
74
79.6k
      else
75
79.6k
  insn = (insn & 0x01f0) >> 4; /* Destination register.  */
76
77
119k
      sprintf (buf, "r%d", insn);
78
119k
      *style = dis_style_register;
79
119k
      break;
80
81
73.6k
    case 'd':
82
73.6k
      if (regs)
83
2.64k
  sprintf (buf, "r%d", 16 + (insn & 0xf));
84
70.9k
      else
85
70.9k
  sprintf (buf, "r%d", 16 + ((insn & 0xf0) >> 4));
86
73.6k
      *style = dis_style_register;
87
73.6k
      break;
88
89
1.01k
    case 'w':
90
1.01k
      sprintf (buf, "r%d", 24 + ((insn & 0x30) >> 3));
91
1.01k
      *style = dis_style_register;
92
1.01k
      break;
93
94
2.77k
    case 'a':
95
2.77k
      if (regs)
96
1.38k
  sprintf (buf, "r%d", 16 + (insn & 7));
97
1.38k
      else
98
1.38k
  sprintf (buf, "r%d", 16 + ((insn >> 4) & 7));
99
2.77k
      *style = dis_style_register;
100
2.77k
      break;
101
102
6.65k
    case 'v':
103
6.65k
      if (regs)
104
3.32k
  sprintf (buf, "r%d", (insn & 0xf) * 2);
105
3.32k
      else
106
3.32k
  sprintf (buf, "r%d", ((insn & 0xf0) >> 3));
107
6.65k
      *style = dis_style_register;
108
6.65k
      break;
109
110
4.68k
    case 'e':
111
4.68k
      {
112
4.68k
  char *xyz;
113
114
4.68k
  switch (insn & 0x100f)
115
4.68k
    {
116
771
      case 0x0000: xyz = "Z";  break;
117
788
      case 0x1001: xyz = "Z+"; break;
118
576
      case 0x1002: xyz = "-Z"; break;
119
322
      case 0x0008: xyz = "Y";  break;
120
96
      case 0x1009: xyz = "Y+"; break;
121
249
      case 0x100a: xyz = "-Y"; break;
122
297
      case 0x100c: xyz = "X";  break;
123
275
      case 0x100d: xyz = "X+"; break;
124
278
      case 0x100e: xyz = "-X"; break;
125
1.03k
      default: xyz = "??"; ok = 0;
126
4.68k
    }
127
4.68k
  strcpy (buf, xyz);
128
129
4.68k
  if (AVR_UNDEF_P (insn))
130
683
    sprintf (comment, _("undefined"));
131
4.68k
      }
132
0
      *style = dis_style_register;
133
4.68k
      break;
134
135
1.21k
    case 'z':
136
1.21k
      *buf++ = 'Z';
137
138
      /* Check for post-increment. */
139
1.21k
      char *s;
140
19.9k
      for (s = opcode_str; *s; ++s)
141
19.4k
        {
142
19.4k
          if (*s == '+')
143
670
            {
144
670
        if (insn & (1 << (15 - (s - opcode_str))))
145
272
    *buf++ = '+';
146
670
              break;
147
670
            }
148
19.4k
        }
149
150
1.21k
      *buf = '\0';
151
1.21k
      if (AVR_UNDEF_P (insn))
152
18
  sprintf (comment, _("undefined"));
153
1.21k
      *style = dis_style_register;
154
1.21k
      break;
155
156
19.9k
    case 'b':
157
19.9k
      {
158
19.9k
  unsigned int x;
159
160
19.9k
  x = (insn & 7);
161
19.9k
  x |= (insn >> 7) & (3 << 3);
162
19.9k
  x |= (insn >> 8) & (1 << 5);
163
164
19.9k
  if (insn & 0x8)
165
9.03k
    *buf++ = 'Y';
166
10.9k
  else
167
10.9k
    *buf++ = 'Z';
168
19.9k
  sprintf (buf, "+%d", x);
169
19.9k
  sprintf (comment, "0x%02x", x);
170
19.9k
  *style = dis_style_register;
171
19.9k
      }
172
19.9k
      break;
173
174
277
    case 'h':
175
277
      *sym = 1;
176
277
      *sym_addr = ((((insn & 1) | ((insn & 0x1f0) >> 3)) << 16) | insn2) * 2;
177
      /* See PR binutils/2454.  Ideally we would like to display the hex
178
   value of the address only once, but this would mean recoding
179
   objdump_print_address() which would affect many targets.  */
180
277
      sprintf (buf, "%#lx", (unsigned long) *sym_addr);
181
277
      strcpy (comment, comment_start);
182
277
      info->insn_info_valid = 1;
183
277
      info->insn_type = dis_jsr;
184
277
      info->target = *sym_addr;
185
277
      *style = dis_style_address;
186
277
      break;
187
188
18.3k
    case 'L':
189
18.3k
      {
190
18.3k
  int rel_addr = (((insn & 0xfff) ^ 0x800) - 0x800) * 2;
191
18.3k
  sprintf (buf, ".%+-8d", rel_addr);
192
18.3k
        *sym = 1;
193
18.3k
        *sym_addr = pc + 2 + rel_addr;
194
18.3k
  strcpy (comment, comment_start);
195
18.3k
        info->insn_info_valid = 1;
196
18.3k
        info->insn_type = dis_branch;
197
18.3k
        info->target = *sym_addr;
198
18.3k
  *style = dis_style_address_offset;
199
18.3k
      }
200
18.3k
      break;
201
202
4.65k
    case 'l':
203
4.65k
      {
204
4.65k
  int rel_addr = ((((insn >> 3) & 0x7f) ^ 0x40) - 0x40) * 2;
205
206
4.65k
  sprintf (buf, ".%+-8d", rel_addr);
207
4.65k
        *sym = 1;
208
4.65k
        *sym_addr = pc + 2 + rel_addr;
209
4.65k
  strcpy (comment, comment_start);
210
4.65k
        info->insn_info_valid = 1;
211
4.65k
        info->insn_type = dis_condbranch;
212
4.65k
        info->target = *sym_addr;
213
4.65k
  *style = dis_style_address_offset;
214
4.65k
      }
215
4.65k
      break;
216
217
497
    case 'i':
218
497
      {
219
497
        unsigned int val = insn2 | 0x800000;
220
497
        *sym = 1;
221
497
        *sym_addr = val;
222
497
        sprintf (buf, "0x%04X", insn2);
223
497
        strcpy (comment, comment_start);
224
497
  *style = dis_style_immediate;
225
497
      }
226
497
      break;
227
228
1.52k
    case 'j':
229
1.52k
      {
230
1.52k
        unsigned int val = ((insn & 0xf) | ((insn & 0x600) >> 5)
231
1.52k
                                         | ((insn & 0x100) >> 2));
232
1.52k
  if ((insn & 0x100) == 0)
233
1.21k
    val |= 0x80;
234
1.52k
        *sym = 1;
235
1.52k
        *sym_addr = val | 0x800000;
236
1.52k
        sprintf (buf, "0x%02x", val);
237
1.52k
        strcpy (comment, comment_start);
238
1.52k
  *style = dis_style_immediate;
239
1.52k
      }
240
1.52k
      break;
241
242
66.8k
    case 'M':
243
66.8k
      sprintf (buf, "0x%02X", ((insn & 0xf00) >> 4) | (insn & 0xf));
244
66.8k
      sprintf (comment, "%d", ((insn & 0xf00) >> 4) | (insn & 0xf));
245
66.8k
      *style = dis_style_immediate;
246
66.8k
      break;
247
248
0
    case 'n':
249
0
      sprintf (buf, "??");
250
      /* xgettext:c-format */
251
0
      opcodes_error_handler (_("internal disassembler error"));
252
0
      ok = 0;
253
0
      *style = dis_style_immediate;
254
0
      break;
255
256
1.01k
    case 'K':
257
1.01k
      {
258
1.01k
  unsigned int x;
259
260
1.01k
  x = (insn & 0xf) | ((insn >> 2) & 0x30);
261
1.01k
  sprintf (buf, "0x%02x", x);
262
1.01k
  sprintf (comment, "%d", x);
263
1.01k
  *style = dis_style_immediate;
264
1.01k
      }
265
1.01k
      break;
266
267
7.12k
    case 's':
268
7.12k
      sprintf (buf, "%d", insn & 7);
269
7.12k
      *style = dis_style_immediate;
270
7.12k
      break;
271
272
0
    case 'S':
273
0
      sprintf (buf, "%d", (insn >> 4) & 7);
274
0
      *style = dis_style_immediate;
275
0
      break;
276
277
8.81k
    case 'P':
278
8.81k
      {
279
8.81k
  unsigned int x;
280
281
8.81k
  x = (insn & 0xf);
282
8.81k
  x |= (insn >> 5) & 0x30;
283
8.81k
  sprintf (buf, "0x%02x", x);
284
8.81k
  sprintf (comment, "%d", x);
285
8.81k
  *style = dis_style_address;
286
8.81k
      }
287
8.81k
      break;
288
289
1.90k
    case 'p':
290
1.90k
      {
291
1.90k
  unsigned int x;
292
293
1.90k
  x = (insn >> 3) & 0x1f;
294
1.90k
  sprintf (buf, "0x%02x", x);
295
1.90k
  sprintf (comment, "%d", x);
296
1.90k
  *style = dis_style_address;
297
1.90k
      }
298
1.90k
      break;
299
300
55
    case 'E':
301
55
      sprintf (buf, "%d", (insn >> 4) & 15);
302
55
      *style = dis_style_immediate;
303
55
      break;
304
305
0
    case '?':
306
0
      *buf = '\0';
307
0
      break;
308
309
0
    default:
310
0
      sprintf (buf, "??");
311
      /* xgettext:c-format */
312
0
      opcodes_error_handler (_("unknown constraint `%c'"), constraint);
313
0
      ok = 0;
314
340k
    }
315
316
340k
    return ok;
317
340k
}
318
319
/* Read the opcode from ADDR.  Return 0 in success and save opcode
320
   in *INSN, otherwise, return -1.  */
321
322
static int
323
avrdis_opcode (bfd_vma addr, disassemble_info *info, uint16_t *insn)
324
261k
{
325
261k
  bfd_byte buffer[2];
326
261k
  int status;
327
328
261k
  status = info->read_memory_func (addr, buffer, 2, info);
329
330
261k
  if (status == 0)
331
261k
    {
332
261k
      *insn = bfd_getl16 (buffer);
333
261k
      return 0;
334
261k
    }
335
336
151
  info->memory_error_func (status, addr, info);
337
151
  return -1;
338
261k
}
339
340
341
int
342
print_insn_avr (bfd_vma addr, disassemble_info *info)
343
260k
{
344
260k
  uint16_t insn, insn2;
345
260k
  const struct avr_opcodes_s *opcode;
346
260k
  static unsigned int *maskptr;
347
260k
  void *stream = info->stream;
348
260k
  fprintf_styled_ftype prin = info->fprintf_styled_func;
349
260k
  static unsigned int *avr_bin_masks;
350
260k
  static int initialized;
351
260k
  int cmd_len = 2;
352
260k
  int ok = 0;
353
260k
  char op1[20], op2[20], comment1[40], comment2[40];
354
260k
  enum disassembler_style style_op1, style_op2;
355
260k
  int sym_op1 = 0, sym_op2 = 0;
356
260k
  bfd_vma sym_addr1, sym_addr2;
357
358
  /* Clear instruction information field.  */
359
260k
  info->insn_info_valid = 0;
360
260k
  info->branch_delay_insns = 0;
361
260k
  info->data_size = 0;
362
260k
  info->insn_type = dis_noninsn;
363
260k
  info->target = 0;
364
260k
  info->target2 = 0;
365
366
260k
  if (!initialized)
367
2
    {
368
2
      unsigned int nopcodes;
369
370
      /* PR 4045: Try to avoid duplicating the 0x prefix that
371
   objdump_print_addr() will put on addresses when there
372
   is no symbol table available.  */
373
2
      if (info->symtab_size == 0)
374
2
  comment_start = " ";
375
376
2
      nopcodes = sizeof (avr_opcodes) / sizeof (struct avr_opcodes_s);
377
378
2
      avr_bin_masks = xmalloc (nopcodes * sizeof (unsigned int));
379
380
2
      for (opcode = avr_opcodes, maskptr = avr_bin_masks;
381
252
     opcode->name;
382
250
     opcode++, maskptr++)
383
250
  {
384
250
    char * s;
385
250
    unsigned int bin = 0;
386
250
    unsigned int mask = 0;
387
388
4.25k
    for (s = opcode->opcode; *s; ++s)
389
4.00k
      {
390
4.00k
        bin <<= 1;
391
4.00k
        mask <<= 1;
392
4.00k
        bin |= (*s == '1');
393
4.00k
        mask |= (*s == '1' || *s == '0');
394
4.00k
      }
395
250
    assert (s - opcode->opcode == 16);
396
250
    assert (opcode->bin_opcode == bin);
397
250
    *maskptr = mask;
398
250
  }
399
400
2
      initialized = 1;
401
2
    }
402
403
260k
  if (avrdis_opcode (addr, info, &insn)  != 0)
404
150
    return -1;
405
406
260k
  for (opcode = avr_opcodes, maskptr = avr_bin_masks;
407
18.7M
       opcode->name;
408
18.4M
       opcode++, maskptr++)
409
18.7M
    {
410
18.7M
      if ((opcode->isa == AVR_ISA_TINY) && (info->mach != bfd_mach_avrtiny))
411
116k
        continue;
412
18.5M
      if ((insn & *maskptr) == opcode->bin_opcode)
413
215k
        break;
414
18.5M
    }
415
416
  /* Special case: disassemble `ldd r,b+0' as `ld r,b', and
417
     `std b+0,r' as `st b,r' (next entry in the table).  */
418
419
260k
  if (AVR_DISP0_P (insn))
420
1.09k
    opcode++;
421
422
260k
  op1[0] = 0;
423
260k
  op2[0] = 0;
424
260k
  comment1[0] = 0;
425
260k
  comment2[0] = 0;
426
260k
  style_op1 = dis_style_text;
427
260k
  style_op2 = dis_style_text;
428
429
260k
  if (opcode->name)
430
215k
    {
431
215k
      char *constraints = opcode->constraints;
432
215k
      char *opcode_str = opcode->opcode;
433
434
215k
      insn2 = 0;
435
215k
      ok = 1;
436
437
215k
      if (opcode->insn_size > 1)
438
775
  {
439
775
    if (avrdis_opcode (addr + 2, info, &insn2) != 0)
440
1
      return -1;
441
774
    cmd_len = 4;
442
774
  }
443
444
215k
      if (*constraints && *constraints != '?')
445
182k
  {
446
182k
    int regs = REGISTER_P (*constraints);
447
448
182k
    ok = avr_operand (insn, insn2, addr, *constraints, opcode_str, op1,
449
182k
          comment1, &style_op1, 0, &sym_op1, &sym_addr1,
450
182k
          info);
451
452
182k
    if (ok && *(++constraints) == ',')
453
157k
      ok = avr_operand (insn, insn2, addr, *(++constraints), opcode_str,
454
157k
            op2, *comment1 ? comment2 : comment1,
455
157k
            &style_op2, regs, &sym_op2, &sym_addr2,
456
157k
            info);
457
182k
  }
458
215k
    }
459
460
260k
  if (!ok)
461
45.9k
    {
462
      /* Unknown opcode, or invalid combination of operands.  */
463
45.9k
      sprintf (op1, "0x%04x", insn);
464
45.9k
      op2[0] = 0;
465
45.9k
      sprintf (comment1, "????");
466
45.9k
      comment2[0] = 0;
467
45.9k
    }
468
469
260k
  (*prin) (stream, ok ? dis_style_mnemonic : dis_style_assembler_directive,
470
260k
     "%s", ok ? opcode->name : ".word");
471
  
472
260k
  if (*op1)
473
227k
    (*prin) (stream, style_op1, "\t%s", op1);
474
475
260k
  if (*op2)
476
157k
    {
477
157k
      (*prin) (stream, dis_style_text, ", ");
478
157k
      (*prin) (stream, style_op2, "%s", op2);
479
157k
    }
480
481
260k
  if (*comment1)
482
170k
    (*prin) (stream, dis_style_comment_start, "\t; %s", comment1);
483
484
260k
  if (sym_op1)
485
24.2k
    info->print_address_func (sym_addr1, info);
486
487
260k
  if (*comment2)
488
0
    (*prin) (stream, dis_style_comment_start, " %s", comment2);
489
490
260k
  if (sym_op2)
491
1.03k
    info->print_address_func (sym_addr2, info);
492
493
260k
  return cmd_len;
494
260k
}