|
|
@@ -1,12 +1,245 @@
|
|
1
|
1
|
#include <assert.h>
|
|
|
2
|
+#include <inttypes.h>
|
|
|
3
|
+#include <stdio.h>
|
|
|
4
|
+#include <string.h>
|
|
|
5
|
+
|
|
|
6
|
+#include <linux/bpf.h>
|
|
2
|
7
|
|
|
3
|
8
|
#include "ir.h"
|
|
4
|
9
|
|
|
5
|
10
|
const uint16_t vreg_base = 0x8000;
|
|
6
|
11
|
|
|
|
12
|
+static const char *bpf_func_name(enum bpf_func_id id)
|
|
|
13
|
+{
|
|
|
14
|
+ switch (id) {
|
|
|
15
|
+ case BPF_FUNC_get_current_comm:
|
|
|
16
|
+ return "get_current_comm";
|
|
|
17
|
+ case BPF_FUNC_get_current_pid_tgid:
|
|
|
18
|
+ return "get_current_pid_tgid";
|
|
|
19
|
+ case BPF_FUNC_get_current_uid_gid:
|
|
|
20
|
+ return "get_current_uid_gid";
|
|
|
21
|
+ case BPF_FUNC_get_stackid:
|
|
|
22
|
+ return "get_stackid";
|
|
|
23
|
+ case BPF_FUNC_ktime_get_ns:
|
|
|
24
|
+ return "ktime_get_ns";
|
|
|
25
|
+ case BPF_FUNC_map_delete_elem:
|
|
|
26
|
+ return "map_delete_elem";
|
|
|
27
|
+ case BPF_FUNC_map_lookup_elem:
|
|
|
28
|
+ return "map_lookup_elem";
|
|
|
29
|
+ case BPF_FUNC_map_update_elem:
|
|
|
30
|
+ return "map_update_elem";
|
|
|
31
|
+ case BPF_FUNC_perf_event_output:
|
|
|
32
|
+ return "perf_event_output";
|
|
|
33
|
+ case BPF_FUNC_probe_read:
|
|
|
34
|
+ return "probe_read";
|
|
|
35
|
+ case BPF_FUNC_trace_printk:
|
|
|
36
|
+ return "trace_printk";
|
|
|
37
|
+ default:
|
|
|
38
|
+ return NULL;
|
|
|
39
|
+ }
|
|
|
40
|
+}
|
|
|
41
|
+
|
|
|
42
|
+static void reg_name(uint16_t reg, char *name)
|
|
|
43
|
+{
|
|
|
44
|
+ if (reg & vreg_base) {
|
|
|
45
|
+ sprintf(name, "v%u", reg & ~vreg_base);
|
|
|
46
|
+ } else if (reg == BPF_REG_10) {
|
|
|
47
|
+ strcpy(name, "bp");
|
|
|
48
|
+ } else {
|
|
|
49
|
+ sprintf(name, "r%u", reg);
|
|
|
50
|
+ }
|
|
|
51
|
+}
|
|
|
52
|
+
|
|
|
53
|
+static void reg_dump(uint16_t reg, int16_t off, FILE *fp)
|
|
|
54
|
+{
|
|
|
55
|
+ char name[8];
|
|
|
56
|
+
|
|
|
57
|
+ reg_name(reg, name);
|
|
|
58
|
+
|
|
|
59
|
+ if (off < 0)
|
|
|
60
|
+ fprintf(fp, "[%s - 0x%x]", name, -off);
|
|
|
61
|
+ else if (off > 0)
|
|
|
62
|
+ fprintf(fp, "[%s + 0x%x]", name, off);
|
|
|
63
|
+ else
|
|
|
64
|
+ fprintf(fp, "%s", name);
|
|
|
65
|
+}
|
|
|
66
|
+
|
|
|
67
|
+static char size_name(uint8_t code)
|
|
|
68
|
+{
|
|
|
69
|
+ switch (BPF_SIZE(code)) {
|
|
|
70
|
+ case BPF_B: return 'b';
|
|
|
71
|
+ case BPF_H: return 'h';
|
|
|
72
|
+ case BPF_W: return 'w';
|
|
|
73
|
+ case BPF_DW: return 'q';
|
|
|
74
|
+ }
|
|
|
75
|
+
|
|
|
76
|
+ return '?';
|
|
|
77
|
+}
|
|
|
78
|
+
|
|
|
79
|
+static void alu_dump(uint8_t code, FILE *fp)
|
|
|
80
|
+{
|
|
|
81
|
+ switch (BPF_OP(code)) {
|
|
|
82
|
+ case BPF_MOV: fputs("mov", fp); break;
|
|
|
83
|
+ case BPF_ADD: fputs("add", fp); break;
|
|
|
84
|
+ case BPF_SUB: fputs("sub", fp); break;
|
|
|
85
|
+ case BPF_MUL: fputs("mul", fp); break;
|
|
|
86
|
+ case BPF_DIV: fputs("div", fp); break;
|
|
|
87
|
+ case BPF_OR : fputs("or", fp); break;
|
|
|
88
|
+ case BPF_AND: fputs("and", fp); break;
|
|
|
89
|
+ case BPF_LSH: fputs("lsh", fp); break;
|
|
|
90
|
+ case BPF_RSH: fputs("rsh", fp); break;
|
|
|
91
|
+ case BPF_NEG: fputs("neg", fp); break;
|
|
|
92
|
+ case BPF_MOD: fputs("mod", fp); break;
|
|
|
93
|
+ case BPF_XOR: fputs("xor", fp); break;
|
|
|
94
|
+ }
|
|
|
95
|
+
|
|
|
96
|
+ switch (BPF_CLASS(code)) {
|
|
|
97
|
+ case BPF_ALU: fputc(size_name(BPF_W), fp);
|
|
|
98
|
+ case BPF_ALU64: fputc(size_name(BPF_DW), fp);
|
|
|
99
|
+ }
|
|
|
100
|
+}
|
|
|
101
|
+
|
|
|
102
|
+static void offset_dump(int16_t off, FILE *fp)
|
|
|
103
|
+{
|
|
|
104
|
+ if (off < 0)
|
|
|
105
|
+ fprintf(fp, "L%d", -off);
|
|
|
106
|
+ else
|
|
|
107
|
+ fprintf(fp, "+%d", off);
|
|
|
108
|
+}
|
|
|
109
|
+
|
|
|
110
|
+static void __insn_dump(const struct bpf_insn insn, uint16_t dst, uint16_t src,
|
|
|
111
|
+ FILE *fp)
|
|
|
112
|
+{
|
|
|
113
|
+ const char *name;
|
|
|
114
|
+ enum {
|
|
|
115
|
+ OFF_NONE,
|
|
|
116
|
+ OFF_DST,
|
|
|
117
|
+ OFF_SRC,
|
|
|
118
|
+ OFF_EXP,
|
|
|
119
|
+ } off = OFF_NONE;
|
|
|
120
|
+
|
|
|
121
|
+ switch (BPF_CLASS(insn.code)) {
|
|
|
122
|
+ case BPF_LD:
|
|
|
123
|
+ case BPF_LDX:
|
|
|
124
|
+ off = OFF_SRC;
|
|
|
125
|
+ fprintf(fp, "ld%c", size_name(insn.code));
|
|
|
126
|
+ break;
|
|
|
127
|
+
|
|
|
128
|
+ case BPF_ST:
|
|
|
129
|
+ case BPF_STX:
|
|
|
130
|
+ off = OFF_DST;
|
|
|
131
|
+ fprintf(fp, "st%c", size_name(insn.code));
|
|
|
132
|
+ break;
|
|
|
133
|
+
|
|
|
134
|
+ case BPF_ALU:
|
|
|
135
|
+ case BPF_ALU64:
|
|
|
136
|
+ alu_dump(insn.code, fp);
|
|
|
137
|
+ break;
|
|
|
138
|
+
|
|
|
139
|
+ case BPF_JMP:
|
|
|
140
|
+ off = OFF_EXP;
|
|
|
141
|
+
|
|
|
142
|
+ switch (BPF_OP(insn.code)) {
|
|
|
143
|
+ case BPF_EXIT:
|
|
|
144
|
+ fputs("exit", fp);
|
|
|
145
|
+ return;
|
|
|
146
|
+ case BPF_CALL:
|
|
|
147
|
+ fputs("call\t", fp);
|
|
|
148
|
+
|
|
|
149
|
+ name = bpf_func_name(insn.imm);
|
|
|
150
|
+ if (name)
|
|
|
151
|
+ fputs(name, fp);
|
|
|
152
|
+ else
|
|
|
153
|
+ fprintf(fp, "%d", insn.imm);
|
|
|
154
|
+ return;
|
|
|
155
|
+ case BPF_JA:
|
|
|
156
|
+ fputs("ja\t", fp);
|
|
|
157
|
+ offset_dump(insn.off, fp);
|
|
|
158
|
+ return;
|
|
|
159
|
+
|
|
|
160
|
+ case BPF_JEQ: fputs("jeq", fp); break;
|
|
|
161
|
+ case BPF_JNE: fputs("jne", fp); break;
|
|
|
162
|
+ case BPF_JGT: fputs("jgt", fp); break;
|
|
|
163
|
+ case BPF_JGE: fputs("jge", fp); break;
|
|
|
164
|
+ case BPF_JSGE: fputs("jsge", fp); break;
|
|
|
165
|
+ case BPF_JSGT: fputs("jsgt", fp); break;
|
|
|
166
|
+ default:
|
|
|
167
|
+ goto unknown;
|
|
|
168
|
+ }
|
|
|
169
|
+ break;
|
|
|
170
|
+
|
|
|
171
|
+ default:
|
|
|
172
|
+ goto unknown;
|
|
|
173
|
+ }
|
|
|
174
|
+
|
|
|
175
|
+ fputc('\t', fp);
|
|
|
176
|
+ reg_dump(dst, off == OFF_DST ? insn.off : 0, fp);
|
|
|
177
|
+ fputs(", ", fp);
|
|
|
178
|
+
|
|
|
179
|
+ if (BPF_CLASS(insn.code) == BPF_LDX || BPF_CLASS(insn.code) == BPF_STX)
|
|
|
180
|
+ goto reg_src;
|
|
|
181
|
+
|
|
|
182
|
+ switch (BPF_SRC(insn.code)) {
|
|
|
183
|
+ case BPF_K:
|
|
|
184
|
+ fprintf(fp, "#%s0x%x", insn.imm < 0 ? "-" : "",
|
|
|
185
|
+ insn.imm < 0 ? -insn.imm : insn.imm);
|
|
|
186
|
+ break;
|
|
|
187
|
+ case BPF_X:
|
|
|
188
|
+ reg_src:
|
|
|
189
|
+ reg_dump(src, off == OFF_SRC ? insn.off : 0, fp);
|
|
|
190
|
+ break;
|
|
|
191
|
+ }
|
|
|
192
|
+
|
|
|
193
|
+ if (off == OFF_EXP) {
|
|
|
194
|
+ fputs(", ", fp);
|
|
|
195
|
+ offset_dump(insn.off, fp);
|
|
|
196
|
+ }
|
|
|
197
|
+
|
|
|
198
|
+ return;
|
|
|
199
|
+
|
|
|
200
|
+unknown:
|
|
|
201
|
+ fprintf(fp, "data\t0x%16.16" PRIx64 "\n", *((uint64_t *)&insn));
|
|
|
202
|
+}
|
|
|
203
|
+
|
|
|
204
|
+void insn_dump(struct bpf_insn insn, FILE *fp)
|
|
|
205
|
+{
|
|
|
206
|
+ __insn_dump(insn, insn.dst_reg, insn.src_reg, fp);
|
|
|
207
|
+}
|
|
|
208
|
+
|
|
|
209
|
+void vinsn_dump(vinsn_t *vi, FILE *fp)
|
|
|
210
|
+{
|
|
|
211
|
+ switch (vi->vitype) {
|
|
|
212
|
+ case VI_INSN:
|
|
|
213
|
+ __insn_dump(vi->insn.bpf, vi->insn.dst, vi->insn.src, fp);
|
|
|
214
|
+ return;
|
|
|
215
|
+ case VI_LABEL:
|
|
|
216
|
+ offset_dump(vi->label, fp);
|
|
|
217
|
+ fputc(':', fp);
|
|
|
218
|
+ return;
|
|
|
219
|
+ case VI_REG_GET:
|
|
|
220
|
+ case VI_REG_PUT:
|
|
|
221
|
+ fputs((vi->vitype == VI_REG_GET) ? "+ " : "- ", fp);
|
|
|
222
|
+ reg_dump(vi->reg, 0, fp);
|
|
|
223
|
+ return;
|
|
|
224
|
+ }
|
|
|
225
|
+}
|
|
|
226
|
+
|
|
|
227
|
+void ir_dump(ir_t *ir, FILE *fp)
|
|
|
228
|
+{
|
|
|
229
|
+ size_t i;
|
|
|
230
|
+
|
|
|
231
|
+ for (i = 0; i < ir->len; i++) {
|
|
|
232
|
+ if (ir->vi[i].vitype == VI_INSN)
|
|
|
233
|
+ fputc('\t', fp);
|
|
|
234
|
+
|
|
|
235
|
+ vinsn_dump(&ir->vi[i], fp);
|
|
|
236
|
+ fputc('\n', fp);
|
|
|
237
|
+ }
|
|
|
238
|
+}
|
|
|
239
|
+
|
|
7
|
240
|
static void ir_emit(ir_t *ir, vinsn_t *vi)
|
|
8
|
241
|
{
|
|
9
|
|
- ir->vi = realloc(ir->vi, ++ir->len);
|
|
|
242
|
+ ir->vi = realloc(ir->vi, (++ir->len)*sizeof(*vi));
|
|
10
|
243
|
assert(ir->vi);
|
|
11
|
244
|
|
|
12
|
245
|
ir->vi[ir->len - 1] = *vi;
|
|
|
@@ -29,6 +262,7 @@ void ir_emit_label (ir_t *ir, int16_t label)
|
|
29
|
262
|
|
|
30
|
263
|
vi.vitype = VI_LABEL;
|
|
31
|
264
|
vi.label = label;
|
|
|
265
|
+ ir_emit(ir, &vi);
|
|
32
|
266
|
}
|
|
33
|
267
|
|
|
34
|
268
|
void ir_emit_reg_get(ir_t *ir, uint16_t reg)
|
|
|
@@ -37,6 +271,7 @@ void ir_emit_reg_get(ir_t *ir, uint16_t reg)
|
|
37
|
271
|
|
|
38
|
272
|
vi.vitype = VI_REG_GET;
|
|
39
|
273
|
vi.reg = reg;
|
|
|
274
|
+ ir_emit(ir, &vi);
|
|
40
|
275
|
}
|
|
41
|
276
|
|
|
42
|
277
|
void ir_emit_reg_put(ir_t *ir, uint16_t reg)
|
|
|
@@ -45,6 +280,7 @@ void ir_emit_reg_put(ir_t *ir, uint16_t reg)
|
|
45
|
280
|
|
|
46
|
281
|
vi.vitype = VI_REG_PUT;
|
|
47
|
282
|
vi.reg = reg;
|
|
|
283
|
+ ir_emit(ir, &vi);
|
|
48
|
284
|
}
|
|
49
|
285
|
|
|
50
|
286
|
int16_t ir_alloc_label (ir_t *ir)
|