Coverage Report

Created: 2026-05-11 07:54

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/opcodes/bpf-dis.c
Line
Count
Source
1
/* bpf-dis.c - BPF disassembler.
2
   Copyright (C) 2023-2026 Free Software Foundation, Inc.
3
4
   Contributed by Oracle Inc.
5
6
   This file is part of the GNU binutils.
7
8
   This is free software; you can redistribute them and/or modify them
9
   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
   This program is distributed in the hope that it will be useful, but
14
   WITHOUT ANY WARRANTY; without even the implied warranty of
15
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16
   General Public License for more details.
17
18
   You should have received a copy of the GNU General Public License
19
   along with this program; see the file COPYING3. If not,
20
   see <http://www.gnu.org/licenses/>.  */
21
22
#include "sysdep.h"
23
#include "disassemble.h"
24
#include "libiberty.h"
25
#include "opintl.h"
26
#include "opcode/bpf.h"
27
#include "elf-bfd.h"
28
#include "elf/bpf.h"
29
30
#include <string.h>
31
#include <inttypes.h>
32
33
/* This disassembler supports two different syntaxes for BPF assembly.
34
   One is called "normal" and has the typical form for assembly
35
   languages, with mnemonics and the like.  The other is called
36
   "pseudoc" and looks like C.  */
37
38
enum bpf_dialect
39
{
40
  BPF_DIALECT_NORMAL,
41
  BPF_DIALECT_PSEUDOC
42
};
43
44
/* Global configuration for the disassembler.  */
45
46
static enum bpf_dialect asm_dialect = BPF_DIALECT_NORMAL;
47
static int asm_bpf_version = -1;
48
static int asm_obase = 10;
49
50
/* Print BPF specific command-line options.  */
51
52
void
53
print_bpf_disassembler_options (FILE *stream)
54
0
{
55
0
  fprintf (stream, _("\n\
56
0
The following BPF specific disassembler options are supported for use\n\
57
0
with the -M switch (multiple options should be separated by commas):\n"));
58
0
  fprintf (stream, "\n");
59
0
  fprintf (stream, _("\
60
0
      pseudoc                  Use pseudo-c syntax.\n\
61
0
      v1,v2,v3,v4,xbpf         Version of the BPF ISA to use.\n\
62
0
      hex,oct,dec              Output numerical base for immediates.\n"));
63
0
}
64
65
/* Parse BPF specific command-line options.  */
66
67
static void
68
parse_bpf_dis_option (const char *option)
69
0
{
70
0
  if (strcmp (option, "pseudoc") == 0)
71
0
    asm_dialect = BPF_DIALECT_PSEUDOC;
72
0
  else if (strcmp (option, "v1") == 0)
73
0
    asm_bpf_version = BPF_V1;
74
0
  else if (strcmp (option, "v2") == 0)
75
0
    asm_bpf_version = BPF_V2;
76
0
  else if (strcmp (option, "v3") == 0)
77
0
    asm_bpf_version = BPF_V3;
78
0
  else if (strcmp (option, "v4") == 0)
79
0
    asm_bpf_version = BPF_V4;
80
0
  else if (strcmp (option, "xbpf") == 0)
81
0
    asm_bpf_version = BPF_XBPF;
82
0
  else if (strcmp (option, "hex") == 0)
83
0
    asm_obase = 16;
84
0
  else if (strcmp (option, "oct") == 0)
85
0
    asm_obase = 8;
86
0
  else if (strcmp (option, "dec") == 0)
87
0
    asm_obase = 10;
88
0
  else
89
    /* xgettext:c-format */
90
0
    opcodes_error_handler (_("unrecognized disassembler option: %s"), option);
91
0
}
92
93
static void
94
parse_bpf_dis_options (const char *opts_in)
95
0
{
96
0
  char *opts = xstrdup (opts_in), *opt = opts, *opt_end = opts;
97
98
0
  for ( ; opt_end != NULL; opt = opt_end + 1)
99
0
    {
100
0
      if ((opt_end = strchr (opt, ',')) != NULL)
101
0
  *opt_end = 0;
102
0
      parse_bpf_dis_option (opt);
103
0
    }
104
105
0
  free (opts);
106
0
}
107
108
/* Auxiliary function used in print_insn_bpf below.  */
109
110
static void
111
print_register (disassemble_info *info,
112
                const char *tag, uint8_t regno)
113
231
{
114
231
  const char *fmt
115
231
    = (asm_dialect == BPF_DIALECT_NORMAL
116
231
       ? "%%r%d"
117
231
       : ((*(tag + 2) == 'w')
118
0
          ? "w%d"
119
0
          : "r%d"));
120
121
231
  (*info->fprintf_styled_func) (info->stream, dis_style_register, fmt, regno);
122
231
}
123
124
/* Main entry point.
125
   Print one instruction from PC on INFO->STREAM.
126
   Return the size of the instruction (in bytes).  */
127
128
int
129
print_insn_bpf (bfd_vma pc, disassemble_info *info)
130
1.07k
{
131
1.07k
  int insn_size = 8, status;
132
1.07k
  bfd_byte insn_bytes[16];
133
1.07k
  bpf_insn_word word = 0;
134
1.07k
  const struct bpf_opcode *insn = NULL;
135
1.07k
  enum bpf_endian endian = (info->endian == BFD_ENDIAN_LITTLE
136
1.07k
                            ? BPF_ENDIAN_LITTLE : BPF_ENDIAN_BIG);
137
138
  /* Handle bpf-specific command-line options.  */
139
1.07k
  if (info->disassembler_options != NULL)
140
0
    {
141
0
      parse_bpf_dis_options (info->disassembler_options);
142
      /* Avoid repeteadly parsing the options.  */
143
0
      info->disassembler_options = NULL;
144
0
    }
145
146
  /* Determine what version of the BPF ISA to use when disassembling.
147
     If the user didn't explicitly specify an ISA version, then derive
148
     it from the CPU Version flag in the ELF header.  A CPU version of
149
     0 in the header means "latest version".  */
150
1.07k
  if (asm_bpf_version == -1 && info->section && info->section->owner)
151
1
    {
152
1
      struct bfd *abfd = info->section->owner;
153
1
      Elf_Internal_Ehdr *header = elf_elfheader (abfd);
154
1
      int cpu_version = header->e_flags & EF_BPF_CPUVER;
155
156
1
      switch (cpu_version)
157
1
        {
158
1
        case 0: asm_bpf_version = BPF_V4; break;
159
0
        case 1: asm_bpf_version = BPF_V1; break;
160
0
        case 2: asm_bpf_version = BPF_V2; break;
161
0
        case 3: asm_bpf_version = BPF_V3; break;
162
0
        case 4: asm_bpf_version = BPF_V4; break;
163
0
        case 0xf: asm_bpf_version = BPF_XBPF; break;
164
0
        default:
165
          /* xgettext:c-format */
166
0
          opcodes_error_handler (_("unknown BPF CPU version %u\n"),
167
0
                                 cpu_version);
168
0
          break;
169
1
        }
170
1
    }
171
172
  /* Print eight bytes per line.  */
173
1.07k
  info->bytes_per_chunk = 1;
174
1.07k
  info->bytes_per_line = 8;
175
176
  /* Read an instruction word.  */
177
1.07k
  status = (*info->read_memory_func) (pc, insn_bytes, 8, info);
178
1.07k
  if (status != 0)
179
11
    {
180
11
      (*info->memory_error_func) (status, pc, info);
181
11
      return -1;
182
11
    }
183
1.06k
  word = (bpf_insn_word) bfd_getb64 (insn_bytes);
184
185
  /* Try to match an instruction with it.  */
186
1.06k
  insn = bpf_match_insn (word, endian, asm_bpf_version);
187
188
  /* Print it out.  */
189
1.06k
  if (insn)
190
215
    {
191
215
      const char *insn_tmpl
192
215
        = asm_dialect == BPF_DIALECT_NORMAL ? insn->normal : insn->pseudoc;
193
215
      const char *p = insn_tmpl;
194
195
      /* Print the template contents completed with the instruction
196
         operands.  */
197
2.53k
      for (p = insn_tmpl; *p != '\0';)
198
2.32k
        {
199
2.32k
          switch (*p)
200
2.32k
            {
201
462
            case ' ':
202
              /* Single space prints to nothing.  */
203
462
              p += 1;
204
462
              break;
205
659
            case '%':
206
659
              if (*(p + 1) == '%')
207
0
                {
208
0
                  (*info->fprintf_styled_func) (info->stream, dis_style_text, "%%");
209
0
                  p += 2;
210
0
                }
211
659
              else if (*(p + 1) == 'w' || *(p + 1) == 'W')
212
215
                {
213
                  /* %W prints to a single space.  */
214
215
                  (*info->fprintf_styled_func) (info->stream, dis_style_text, " ");
215
215
                  p += 2;
216
215
                }
217
444
              else if (strncmp (p, "%dr", 3) == 0
218
276
                       || strncmp (p, "%dw", 3) == 0
219
276
                       || strncmp (p, "%dR", 3) == 0)
220
168
                {
221
168
                  print_register (info, p, bpf_extract_dst (word, endian));
222
168
                  p += 3;
223
168
                }
224
276
              else if (strncmp (p, "%sr", 3) == 0
225
213
                       || strncmp (p, "%sw", 3) == 0
226
213
                       || strncmp (p, "%sR", 3) == 0)
227
63
                {
228
63
                  print_register (info, p, bpf_extract_src (word, endian));
229
63
                  p += 3;
230
63
                }
231
213
              else if (strncmp (p, "%i32", 4) == 0
232
83
                       || strncmp (p, "%d32", 4) == 0
233
49
                       || strncmp (p, "%I32", 4) == 0)
234
164
                {
235
164
                  int32_t imm32 = bpf_extract_imm32 (word, endian);
236
237
164
                  if (p[1] == 'I')
238
0
                    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
239
0
                                                  "%s",
240
0
              asm_obase != 10 || imm32 >= 0 ? "+" : "");
241
164
                  (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
242
164
                                                asm_obase == 10 ? "%" PRIi32
243
164
                                                : asm_obase == 8 ? "%" PRIo32
244
0
                                                : "0x%" PRIx32,
245
164
                                                imm32);
246
164
                  p += 4;
247
164
                }
248
49
              else if (strncmp (p, "%o16", 4) == 0
249
42
                       || strncmp (p, "%d16", 4) == 0)
250
49
                {
251
49
                  int16_t offset16 = bpf_extract_offset16 (word, endian);
252
253
49
                  if (p[1] == 'o')
254
7
                    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
255
7
                                                  "%s",
256
7
              asm_obase != 10 || offset16 >= 0 ? "+" : "");
257
49
                  if (asm_obase == 16 || asm_obase == 8)
258
0
                    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
259
0
                                                  asm_obase == 8 ? "0%" PRIo16 : "0x%" PRIx16,
260
0
                                                  (uint16_t) offset16);
261
49
                  else
262
49
                    (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
263
49
                                                  "%" PRIi16, offset16);
264
49
                  p += 4;
265
49
                }
266
0
              else if (strncmp (p, "%i64", 4) == 0)
267
0
                {
268
0
                  bpf_insn_word word2 = 0;
269
270
0
                  status = (*info->read_memory_func) (pc + 8, insn_bytes + 8,
271
0
                                                          8, info);
272
0
                  if (status != 0)
273
0
                    {
274
0
                      (*info->memory_error_func) (status, pc + 8, info);
275
0
                      return -1;
276
0
                    }
277
0
                  word2 = (bpf_insn_word) bfd_getb64 (insn_bytes + 8);
278
279
0
                  (*info->fprintf_styled_func) (info->stream, dis_style_immediate,
280
0
                                                asm_obase == 10 ? "%" PRIi64
281
0
                                                : asm_obase == 8 ? "0%" PRIo64
282
0
                                                : "0x%" PRIx64,
283
0
                                                bpf_extract_imm64 (word, word2, endian));
284
0
                  insn_size = 16;
285
0
                  p += 4;
286
0
                }
287
0
              else
288
0
                {
289
                  /* xgettext:c-format */
290
0
                  opcodes_error_handler (_("# internal error, unknown tag in opcode template (%s)"),
291
0
                                         insn_tmpl);
292
0
                  return -1;
293
0
                }
294
659
              break;
295
1.19k
            default:
296
              /* Any other character is printed literally.  */
297
1.19k
              (*info->fprintf_styled_func) (info->stream, dis_style_text, "%c", *p);
298
1.19k
              p += 1;
299
2.32k
            }
300
2.32k
        }
301
215
    }
302
845
  else
303
845
    (*info->fprintf_styled_func) (info->stream, dis_style_text, "<unknown>");
304
305
1.06k
  return insn_size;
306
1.06k
}