xref: /linux/tools/testing/selftests/bpf/progs/iters_state_safety.c (revision 7b26bc6582b13a52a42a4a9765e8f30d58a81198)
1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright (c) 2022 Facebook */
3 
4 #include <errno.h>
5 #include <string.h>
6 #include <linux/bpf.h>
7 #include <bpf/bpf_helpers.h>
8 #include "bpf_misc.h"
9 
10 char _license[] SEC("license") = "GPL";
11 
12 #define ITER_HELPERS						\
13 	  __imm(bpf_iter_num_new),				\
14 	  __imm(bpf_iter_num_next),				\
15 	  __imm(bpf_iter_num_destroy)
16 
17 SEC("?raw_tp")
18 __success
19 int force_clang_to_emit_btf_for_externs(void *ctx)
20 {
21 	/* we need this as a workaround to enforce compiler emitting BTF
22 	 * information for bpf_iter_num_{new,next,destroy}() kfuncs,
23 	 * as, apparently, it doesn't emit it for symbols only referenced from
24 	 * assembly (or cleanup attribute, for that matter, as well)
25 	 */
26 	bpf_repeat(0);
27 
28 	return 0;
29 }
30 
31 SEC("?raw_tp")
32 __success __log_level(2)
33 __msg("fp-8_w=iter_num(ref_id=1,state=active,depth=0)")
34 int create_and_destroy(void *ctx)
35 {
36 	struct bpf_iter_num iter;
37 
38 	asm volatile (
39 		/* create iterator */
40 		"r1 = %[iter];"
41 		"r2 = 0;"
42 		"r3 = 1000;"
43 		"call %[bpf_iter_num_new];"
44 		/* destroy iterator */
45 		"r1 = %[iter];"
46 		"call %[bpf_iter_num_destroy];"
47 		:
48 		: __imm_ptr(iter), ITER_HELPERS
49 		: __clobber_common
50 	);
51 
52 	return 0;
53 }
54 
55 SEC("?raw_tp")
56 __failure __msg("Unreleased reference id=1")
57 int create_and_forget_to_destroy_fail(void *ctx)
58 {
59 	struct bpf_iter_num iter;
60 
61 	asm volatile (
62 		/* create iterator */
63 		"r1 = %[iter];"
64 		"r2 = 0;"
65 		"r3 = 1000;"
66 		"call %[bpf_iter_num_new];"
67 		:
68 		: __imm_ptr(iter), ITER_HELPERS
69 		: __clobber_common
70 	);
71 
72 	return 0;
73 }
74 
75 SEC("?raw_tp")
76 __failure __msg("expected an initialized iter_num as arg #0")
77 int destroy_without_creating_fail(void *ctx)
78 {
79 	/* init with zeros to stop verifier complaining about uninit stack */
80 	struct bpf_iter_num iter;
81 
82 	asm volatile (
83 		"r1 = %[iter];"
84 		"call %[bpf_iter_num_destroy];"
85 		:
86 		: __imm_ptr(iter), ITER_HELPERS
87 		: __clobber_common
88 	);
89 
90 	return 0;
91 }
92 
93 SEC("?raw_tp")
94 __failure __msg("expected an initialized iter_num as arg #0")
95 int compromise_iter_w_direct_write_fail(void *ctx)
96 {
97 	struct bpf_iter_num iter;
98 
99 	asm volatile (
100 		/* create iterator */
101 		"r1 = %[iter];"
102 		"r2 = 0;"
103 		"r3 = 1000;"
104 		"call %[bpf_iter_num_new];"
105 
106 		/* directly write over first half of iter state */
107 		"*(u64 *)(%[iter] + 0) = r0;"
108 
109 		/* (attempt to) destroy iterator */
110 		"r1 = %[iter];"
111 		"call %[bpf_iter_num_destroy];"
112 		:
113 		: __imm_ptr(iter), ITER_HELPERS
114 		: __clobber_common
115 	);
116 
117 	return 0;
118 }
119 
120 SEC("?raw_tp")
121 __failure __msg("Unreleased reference id=1")
122 int compromise_iter_w_direct_write_and_skip_destroy_fail(void *ctx)
123 {
124 	struct bpf_iter_num iter;
125 
126 	asm volatile (
127 		/* create iterator */
128 		"r1 = %[iter];"
129 		"r2 = 0;"
130 		"r3 = 1000;"
131 		"call %[bpf_iter_num_new];"
132 
133 		/* directly write over first half of iter state */
134 		"*(u64 *)(%[iter] + 0) = r0;"
135 
136 		/* don't destroy iter, leaking ref, which should fail */
137 		:
138 		: __imm_ptr(iter), ITER_HELPERS
139 		: __clobber_common
140 	);
141 
142 	return 0;
143 }
144 
145 SEC("?raw_tp")
146 __failure __msg("expected an initialized iter_num as arg #0")
147 int compromise_iter_w_helper_write_fail(void *ctx)
148 {
149 	struct bpf_iter_num iter;
150 
151 	asm volatile (
152 		/* create iterator */
153 		"r1 = %[iter];"
154 		"r2 = 0;"
155 		"r3 = 1000;"
156 		"call %[bpf_iter_num_new];"
157 
158 		/* overwrite 8th byte with bpf_probe_read_kernel() */
159 		"r1 = %[iter];"
160 		"r1 += 7;"
161 		"r2 = 1;"
162 		"r3 = 0;" /* NULL */
163 		"call %[bpf_probe_read_kernel];"
164 
165 		/* (attempt to) destroy iterator */
166 		"r1 = %[iter];"
167 		"call %[bpf_iter_num_destroy];"
168 		:
169 		: __imm_ptr(iter), ITER_HELPERS, __imm(bpf_probe_read_kernel)
170 		: __clobber_common
171 	);
172 
173 	return 0;
174 }
175 
176 static __noinline void subprog_with_iter(void)
177 {
178 	struct bpf_iter_num iter;
179 
180 	bpf_iter_num_new(&iter, 0, 1);
181 
182 	return;
183 }
184 
185 SEC("?raw_tp")
186 __failure
187 /* ensure there was a call to subprog, which might happen without __noinline */
188 __msg("returning from callee:")
189 __msg("Unreleased reference id=1")
190 int leak_iter_from_subprog_fail(void *ctx)
191 {
192 	subprog_with_iter();
193 
194 	return 0;
195 }
196 
197 SEC("?raw_tp")
198 __success __log_level(2)
199 __msg("fp-8_w=iter_num(ref_id=1,state=active,depth=0)")
200 int valid_stack_reuse(void *ctx)
201 {
202 	struct bpf_iter_num iter;
203 
204 	asm volatile (
205 		/* create iterator */
206 		"r1 = %[iter];"
207 		"r2 = 0;"
208 		"r3 = 1000;"
209 		"call %[bpf_iter_num_new];"
210 		/* destroy iterator */
211 		"r1 = %[iter];"
212 		"call %[bpf_iter_num_destroy];"
213 
214 		/* now reuse same stack slots */
215 
216 		/* create iterator */
217 		"r1 = %[iter];"
218 		"r2 = 0;"
219 		"r3 = 1000;"
220 		"call %[bpf_iter_num_new];"
221 		/* destroy iterator */
222 		"r1 = %[iter];"
223 		"call %[bpf_iter_num_destroy];"
224 		:
225 		: __imm_ptr(iter), ITER_HELPERS
226 		: __clobber_common
227 	);
228 
229 	return 0;
230 }
231 
232 SEC("?raw_tp")
233 __failure __msg("expected uninitialized iter_num as arg #0")
234 int double_create_fail(void *ctx)
235 {
236 	struct bpf_iter_num iter;
237 
238 	asm volatile (
239 		/* create iterator */
240 		"r1 = %[iter];"
241 		"r2 = 0;"
242 		"r3 = 1000;"
243 		"call %[bpf_iter_num_new];"
244 		/* (attempt to) create iterator again */
245 		"r1 = %[iter];"
246 		"r2 = 0;"
247 		"r3 = 1000;"
248 		"call %[bpf_iter_num_new];"
249 		/* destroy iterator */
250 		"r1 = %[iter];"
251 		"call %[bpf_iter_num_destroy];"
252 		:
253 		: __imm_ptr(iter), ITER_HELPERS
254 		: __clobber_common
255 	);
256 
257 	return 0;
258 }
259 
260 SEC("?raw_tp")
261 __failure __msg("expected an initialized iter_num as arg #0")
262 int double_destroy_fail(void *ctx)
263 {
264 	struct bpf_iter_num iter;
265 
266 	asm volatile (
267 		/* create iterator */
268 		"r1 = %[iter];"
269 		"r2 = 0;"
270 		"r3 = 1000;"
271 		"call %[bpf_iter_num_new];"
272 		/* destroy iterator */
273 		"r1 = %[iter];"
274 		"call %[bpf_iter_num_destroy];"
275 		/* (attempt to) destroy iterator again */
276 		"r1 = %[iter];"
277 		"call %[bpf_iter_num_destroy];"
278 		:
279 		: __imm_ptr(iter), ITER_HELPERS
280 		: __clobber_common
281 	);
282 
283 	return 0;
284 }
285 
286 SEC("?raw_tp")
287 __failure __msg("expected an initialized iter_num as arg #0")
288 int next_without_new_fail(void *ctx)
289 {
290 	struct bpf_iter_num iter;
291 
292 	asm volatile (
293 		/* don't create iterator and try to iterate*/
294 		"r1 = %[iter];"
295 		"call %[bpf_iter_num_next];"
296 		/* destroy iterator */
297 		"r1 = %[iter];"
298 		"call %[bpf_iter_num_destroy];"
299 		:
300 		: __imm_ptr(iter), ITER_HELPERS
301 		: __clobber_common
302 	);
303 
304 	return 0;
305 }
306 
307 SEC("?raw_tp")
308 __failure __msg("expected an initialized iter_num as arg #0")
309 int next_after_destroy_fail(void *ctx)
310 {
311 	struct bpf_iter_num iter;
312 
313 	asm volatile (
314 		/* create iterator */
315 		"r1 = %[iter];"
316 		"r2 = 0;"
317 		"r3 = 1000;"
318 		"call %[bpf_iter_num_new];"
319 		/* destroy iterator */
320 		"r1 = %[iter];"
321 		"call %[bpf_iter_num_destroy];"
322 		/* don't create iterator and try to iterate*/
323 		"r1 = %[iter];"
324 		"call %[bpf_iter_num_next];"
325 		:
326 		: __imm_ptr(iter), ITER_HELPERS
327 		: __clobber_common
328 	);
329 
330 	return 0;
331 }
332 
333 SEC("?raw_tp")
334 __failure __msg("invalid read from stack")
335 int __naked read_from_iter_slot_fail(void)
336 {
337 	asm volatile (
338 		/* r6 points to struct bpf_iter_num on the stack */
339 		"r6 = r10;"
340 		"r6 += -24;"
341 
342 		/* create iterator */
343 		"r1 = r6;"
344 		"r2 = 0;"
345 		"r3 = 1000;"
346 		"call %[bpf_iter_num_new];"
347 
348 		/* attemp to leak bpf_iter_num state */
349 		"r7 = *(u64 *)(r6 + 0);"
350 		"r8 = *(u64 *)(r6 + 8);"
351 
352 		/* destroy iterator */
353 		"r1 = r6;"
354 		"call %[bpf_iter_num_destroy];"
355 
356 		/* leak bpf_iter_num state */
357 		"r0 = r7;"
358 		"if r7 > r8 goto +1;"
359 		"r0 = r8;"
360 		"exit;"
361 		:
362 		: ITER_HELPERS
363 		: __clobber_common, "r6", "r7", "r8"
364 	);
365 }
366 
367 int zero;
368 
369 SEC("?raw_tp")
370 __failure
371 __flag(BPF_F_TEST_STATE_FREQ)
372 __msg("Unreleased reference")
373 int stacksafe_should_not_conflate_stack_spill_and_iter(void *ctx)
374 {
375 	struct bpf_iter_num iter;
376 
377 	asm volatile (
378 		/* Create a fork in logic, with general setup as follows:
379 		 *   - fallthrough (first) path is valid;
380 		 *   - branch (second) path is invalid.
381 		 * Then depending on what we do in fallthrough vs branch path,
382 		 * we try to detect bugs in func_states_equal(), regsafe(),
383 		 * refsafe(), stack_safe(), and similar by tricking verifier
384 		 * into believing that branch state is a valid subset of
385 		 * a fallthrough state. Verifier should reject overall
386 		 * validation, unless there is a bug somewhere in verifier
387 		 * logic.
388 		 */
389 		"call %[bpf_get_prandom_u32];"
390 		"r6 = r0;"
391 		"call %[bpf_get_prandom_u32];"
392 		"r7 = r0;"
393 
394 		"if r6 > r7 goto bad;" /* fork */
395 
396 		/* spill r6 into stack slot of bpf_iter_num var */
397 		"*(u64 *)(%[iter] + 0) = r6;"
398 
399 		"goto skip_bad;"
400 
401 	"bad:"
402 		/* create iterator in the same stack slot */
403 		"r1 = %[iter];"
404 		"r2 = 0;"
405 		"r3 = 1000;"
406 		"call %[bpf_iter_num_new];"
407 
408 		/* but then forget about it and overwrite it back to r6 spill */
409 		"*(u64 *)(%[iter] + 0) = r6;"
410 
411 	"skip_bad:"
412 		"goto +0;" /* force checkpoint */
413 
414 		/* corrupt stack slots, if they are really dynptr */
415 		"*(u64 *)(%[iter] + 0) = r6;"
416 		:
417 		: __imm_ptr(iter),
418 		  __imm_addr(zero),
419 		  __imm(bpf_get_prandom_u32),
420 		  __imm(bpf_dynptr_from_mem),
421 		  ITER_HELPERS
422 		: __clobber_common, "r6", "r7"
423 	);
424 
425 	return 0;
426 }
427