1*745b7587SStephen Hemminger /* SPDX-License-Identifier: BSD-3-Clause
2*745b7587SStephen Hemminger * Copyright (c) 2021 Stephen Hemminger
3*745b7587SStephen Hemminger * Based on filter2xdp
4*745b7587SStephen Hemminger * Copyright (C) 2017 Tobias Klauser
5*745b7587SStephen Hemminger */
6*745b7587SStephen Hemminger
7*745b7587SStephen Hemminger #include <stdio.h>
8*745b7587SStephen Hemminger #include <stdint.h>
9*745b7587SStephen Hemminger
10*745b7587SStephen Hemminger #include "rte_bpf.h"
11*745b7587SStephen Hemminger
12*745b7587SStephen Hemminger #define BPF_OP_INDEX(x) (BPF_OP(x) >> 4)
13*745b7587SStephen Hemminger #define BPF_SIZE_INDEX(x) (BPF_SIZE(x) >> 3)
14*745b7587SStephen Hemminger
15*745b7587SStephen Hemminger static const char *const class_tbl[] = {
16*745b7587SStephen Hemminger [BPF_LD] = "ld", [BPF_LDX] = "ldx", [BPF_ST] = "st",
17*745b7587SStephen Hemminger [BPF_STX] = "stx", [BPF_ALU] = "alu", [BPF_JMP] = "jmp",
18*745b7587SStephen Hemminger [BPF_RET] = "ret", [BPF_MISC] = "alu64",
19*745b7587SStephen Hemminger };
20*745b7587SStephen Hemminger
21*745b7587SStephen Hemminger static const char *const alu_op_tbl[16] = {
22*745b7587SStephen Hemminger [BPF_ADD >> 4] = "add", [BPF_SUB >> 4] = "sub",
23*745b7587SStephen Hemminger [BPF_MUL >> 4] = "mul", [BPF_DIV >> 4] = "div",
24*745b7587SStephen Hemminger [BPF_OR >> 4] = "or", [BPF_AND >> 4] = "and",
25*745b7587SStephen Hemminger [BPF_LSH >> 4] = "lsh", [BPF_RSH >> 4] = "rsh",
26*745b7587SStephen Hemminger [BPF_NEG >> 4] = "neg", [BPF_MOD >> 4] = "mod",
27*745b7587SStephen Hemminger [BPF_XOR >> 4] = "xor", [EBPF_MOV >> 4] = "mov",
28*745b7587SStephen Hemminger [EBPF_ARSH >> 4] = "arsh", [EBPF_END >> 4] = "endian",
29*745b7587SStephen Hemminger };
30*745b7587SStephen Hemminger
31*745b7587SStephen Hemminger static const char *const size_tbl[] = {
32*745b7587SStephen Hemminger [BPF_W >> 3] = "w",
33*745b7587SStephen Hemminger [BPF_H >> 3] = "h",
34*745b7587SStephen Hemminger [BPF_B >> 3] = "b",
35*745b7587SStephen Hemminger [EBPF_DW >> 3] = "dw",
36*745b7587SStephen Hemminger };
37*745b7587SStephen Hemminger
38*745b7587SStephen Hemminger static const char *const jump_tbl[16] = {
39*745b7587SStephen Hemminger [BPF_JA >> 4] = "ja", [BPF_JEQ >> 4] = "jeq",
40*745b7587SStephen Hemminger [BPF_JGT >> 4] = "jgt", [BPF_JGE >> 4] = "jge",
41*745b7587SStephen Hemminger [BPF_JSET >> 4] = "jset", [EBPF_JNE >> 4] = "jne",
42*745b7587SStephen Hemminger [EBPF_JSGT >> 4] = "jsgt", [EBPF_JSGE >> 4] = "jsge",
43*745b7587SStephen Hemminger [EBPF_CALL >> 4] = "call", [EBPF_EXIT >> 4] = "exit",
44*745b7587SStephen Hemminger };
45*745b7587SStephen Hemminger
rte_bpf_dump(FILE * f,const struct ebpf_insn * buf,uint32_t len)46*745b7587SStephen Hemminger void rte_bpf_dump(FILE *f, const struct ebpf_insn *buf, uint32_t len)
47*745b7587SStephen Hemminger {
48*745b7587SStephen Hemminger uint32_t i;
49*745b7587SStephen Hemminger
50*745b7587SStephen Hemminger for (i = 0; i < len; ++i) {
51*745b7587SStephen Hemminger const struct ebpf_insn *ins = buf + i;
52*745b7587SStephen Hemminger uint8_t cls = BPF_CLASS(ins->code);
53*745b7587SStephen Hemminger const char *op, *postfix = "";
54*745b7587SStephen Hemminger
55*745b7587SStephen Hemminger fprintf(f, " L%u:\t", i);
56*745b7587SStephen Hemminger
57*745b7587SStephen Hemminger switch (cls) {
58*745b7587SStephen Hemminger default:
59*745b7587SStephen Hemminger fprintf(f, "unimp 0x%x // class: %s\n",
60*745b7587SStephen Hemminger ins->code, class_tbl[cls]);
61*745b7587SStephen Hemminger break;
62*745b7587SStephen Hemminger case BPF_ALU:
63*745b7587SStephen Hemminger postfix = "32";
64*745b7587SStephen Hemminger /* fall through */
65*745b7587SStephen Hemminger case EBPF_ALU64:
66*745b7587SStephen Hemminger op = alu_op_tbl[BPF_OP_INDEX(ins->code)];
67*745b7587SStephen Hemminger if (BPF_SRC(ins->code) == BPF_X)
68*745b7587SStephen Hemminger fprintf(f, "%s%s r%u, r%u\n", op, postfix, ins->dst_reg,
69*745b7587SStephen Hemminger ins->src_reg);
70*745b7587SStephen Hemminger else
71*745b7587SStephen Hemminger fprintf(f, "%s%s r%u, #0x%x\n", op, postfix,
72*745b7587SStephen Hemminger ins->dst_reg, ins->imm);
73*745b7587SStephen Hemminger break;
74*745b7587SStephen Hemminger case BPF_LD:
75*745b7587SStephen Hemminger op = "ld";
76*745b7587SStephen Hemminger postfix = size_tbl[BPF_SIZE_INDEX(ins->code)];
77*745b7587SStephen Hemminger if (ins->code == (BPF_LD | BPF_IMM | EBPF_DW)) {
78*745b7587SStephen Hemminger uint64_t val;
79*745b7587SStephen Hemminger
80*745b7587SStephen Hemminger val = (uint32_t)ins[0].imm |
81*745b7587SStephen Hemminger (uint64_t)(uint32_t)ins[1].imm << 32;
82*745b7587SStephen Hemminger fprintf(f, "%s%s r%d, #0x%"PRIx64"\n",
83*745b7587SStephen Hemminger op, postfix, ins->dst_reg, val);
84*745b7587SStephen Hemminger i++;
85*745b7587SStephen Hemminger } else if (BPF_MODE(ins->code) == BPF_IMM)
86*745b7587SStephen Hemminger fprintf(f, "%s%s r%d, #0x%x\n", op, postfix,
87*745b7587SStephen Hemminger ins->dst_reg, ins->imm);
88*745b7587SStephen Hemminger else if (BPF_MODE(ins->code) == BPF_ABS)
89*745b7587SStephen Hemminger fprintf(f, "%s%s r%d, [%d]\n", op, postfix,
90*745b7587SStephen Hemminger ins->dst_reg, ins->imm);
91*745b7587SStephen Hemminger else if (BPF_MODE(ins->code) == BPF_IND)
92*745b7587SStephen Hemminger fprintf(f, "%s%s r%d, [r%u + %d]\n", op, postfix,
93*745b7587SStephen Hemminger ins->dst_reg, ins->src_reg, ins->imm);
94*745b7587SStephen Hemminger else
95*745b7587SStephen Hemminger fprintf(f, "// BUG: LD opcode 0x%02x in eBPF insns\n",
96*745b7587SStephen Hemminger ins->code);
97*745b7587SStephen Hemminger break;
98*745b7587SStephen Hemminger case BPF_LDX:
99*745b7587SStephen Hemminger op = "ldx";
100*745b7587SStephen Hemminger postfix = size_tbl[BPF_SIZE_INDEX(ins->code)];
101*745b7587SStephen Hemminger fprintf(f, "%s%s r%d, [r%u + %d]\n", op, postfix, ins->dst_reg,
102*745b7587SStephen Hemminger ins->src_reg, ins->off);
103*745b7587SStephen Hemminger break;
104*745b7587SStephen Hemminger case BPF_ST:
105*745b7587SStephen Hemminger op = "st";
106*745b7587SStephen Hemminger postfix = size_tbl[BPF_SIZE_INDEX(ins->code)];
107*745b7587SStephen Hemminger if (BPF_MODE(ins->code) == BPF_MEM)
108*745b7587SStephen Hemminger fprintf(f, "%s%s [r%d + %d], #0x%x\n", op, postfix,
109*745b7587SStephen Hemminger ins->dst_reg, ins->off, ins->imm);
110*745b7587SStephen Hemminger else
111*745b7587SStephen Hemminger fprintf(f, "// BUG: ST opcode 0x%02x in eBPF insns\n",
112*745b7587SStephen Hemminger ins->code);
113*745b7587SStephen Hemminger break;
114*745b7587SStephen Hemminger case BPF_STX:
115*745b7587SStephen Hemminger op = "stx";
116*745b7587SStephen Hemminger postfix = size_tbl[BPF_SIZE_INDEX(ins->code)];
117*745b7587SStephen Hemminger fprintf(f, "%s%s [r%d + %d], r%u\n", op, postfix,
118*745b7587SStephen Hemminger ins->dst_reg, ins->off, ins->src_reg);
119*745b7587SStephen Hemminger break;
120*745b7587SStephen Hemminger #define L(pc, off) ((int)(pc) + 1 + (off))
121*745b7587SStephen Hemminger case BPF_JMP:
122*745b7587SStephen Hemminger op = jump_tbl[BPF_OP_INDEX(ins->code)];
123*745b7587SStephen Hemminger if (op == NULL)
124*745b7587SStephen Hemminger fprintf(f, "invalid jump opcode: %#x\n", ins->code);
125*745b7587SStephen Hemminger else if (BPF_OP(ins->code) == BPF_JA)
126*745b7587SStephen Hemminger fprintf(f, "%s L%d\n", op, L(i, ins->off));
127*745b7587SStephen Hemminger else if (BPF_OP(ins->code) == EBPF_EXIT)
128*745b7587SStephen Hemminger fprintf(f, "%s\n", op);
129*745b7587SStephen Hemminger else
130*745b7587SStephen Hemminger fprintf(f, "%s r%u, #0x%x, L%d\n", op, ins->dst_reg,
131*745b7587SStephen Hemminger ins->imm, L(i, ins->off));
132*745b7587SStephen Hemminger break;
133*745b7587SStephen Hemminger case BPF_RET:
134*745b7587SStephen Hemminger fprintf(f, "// BUG: RET opcode 0x%02x in eBPF insns\n",
135*745b7587SStephen Hemminger ins->code);
136*745b7587SStephen Hemminger break;
137*745b7587SStephen Hemminger }
138*745b7587SStephen Hemminger }
139*745b7587SStephen Hemminger }
140