xref: /linux/tools/testing/selftests/bpf/progs/timer_failure.c (revision 3ba84ac69b53e6ee07c31d54554e00793d7b144f)
1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright (c) 2023 Meta Platforms, Inc. and affiliates. */
3 
4 #include <linux/bpf.h>
5 #include <time.h>
6 #include <errno.h>
7 #include <bpf/bpf_helpers.h>
8 #include <bpf/bpf_tracing.h>
9 #include "bpf_misc.h"
10 
11 char _license[] SEC("license") = "GPL";
12 
13 struct elem {
14 	struct bpf_timer t;
15 };
16 
17 struct {
18 	__uint(type, BPF_MAP_TYPE_ARRAY);
19 	__uint(max_entries, 1);
20 	__type(key, int);
21 	__type(value, struct elem);
22 } timer_map SEC(".maps");
23 
24 __naked __noinline __used
25 static unsigned long timer_cb_ret_bad()
26 {
27 	asm volatile (
28 		"call %[bpf_get_prandom_u32];"
29 		"if r0 s> 1000 goto 1f;"
30 		"r0 = 0;"
31 	"1:"
32 		"goto +0;" /* checkpoint */
33 		/* async callback is expected to return 0, so branch above
34 		 * skipping r0 = 0; should lead to a failure, but if exit
35 		 * instruction doesn't enforce r0's precision, this callback
36 		 * will be successfully verified
37 		 */
38 		"exit;"
39 		:
40 		: __imm(bpf_get_prandom_u32)
41 		: __clobber_common
42 	);
43 }
44 
45 SEC("fentry/bpf_fentry_test1")
46 __log_level(2)
47 __flag(BPF_F_TEST_STATE_FREQ)
48 __failure
49 /* check that fallthrough code path marks r0 as precise */
50 __msg("mark_precise: frame0: regs=r0 stack= before")
51 __msg(": (85) call bpf_get_prandom_u32#7") /* anchor message */
52 /* check that branch code path marks r0 as precise */
53 __msg("mark_precise: frame0: regs=r0 stack= before ") __msg(": (85) call bpf_get_prandom_u32#7")
54 __msg("should have been in [0, 0]")
55 long BPF_PROG2(test_bad_ret, int, a)
56 {
57 	int key = 0;
58 	struct bpf_timer *timer;
59 
60 	timer = bpf_map_lookup_elem(&timer_map, &key);
61 	if (timer) {
62 		bpf_timer_init(timer, &timer_map, CLOCK_BOOTTIME);
63 		bpf_timer_set_callback(timer, timer_cb_ret_bad);
64 		bpf_timer_start(timer, 1000, 0);
65 	}
66 
67 	return 0;
68 }
69