1 // SPDX-License-Identifier: GPL-2.0
3 #include <bpf/bpf_tracing.h>
4 #include <bpf/bpf_helpers.h>
5 #include "bpf_experimental.h"
8 struct bpf_spin_lock lock;
13 __uint(type, BPF_MAP_TYPE_ARRAY);
15 __type(value, struct foo);
16 __uint(max_entries, 1);
17 } array_map SEC(".maps");
20 __uint(type, BPF_MAP_TYPE_ARRAY_OF_MAPS);
21 __uint(max_entries, 1);
24 __array(values, struct array_map);
25 } map_of_maps SEC(".maps") = {
31 static struct bpf_spin_lock lockA SEC(".data.A");
32 static struct bpf_spin_lock lockB SEC(".data.B");
35 int lock_id_kptr_preserve(void *ctx)
39 f = bpf_obj_new(typeof(*f));
47 int lock_id_global_zero(void *ctx)
49 bpf_this_cpu_ptr(&lockA);
54 int lock_id_mapval_preserve(void *ctx)
59 f = bpf_map_lookup_elem(&array_map, &key);
67 int lock_id_innermapval_preserve(void *ctx)
73 map = bpf_map_lookup_elem(&map_of_maps, &key);
76 f = bpf_map_lookup_elem(map, &key);
83 #define CHECK(test, A, B) \
85 int lock_id_mismatch_##test(void *ctx) \
87 struct foo *f1, *f2, *v, *iv; \
91 map = bpf_map_lookup_elem(&map_of_maps, &key); \
94 iv = bpf_map_lookup_elem(map, &key); \
97 v = bpf_map_lookup_elem(&array_map, &key); \
100 f1 = bpf_obj_new(typeof(*f1)); \
103 f2 = bpf_obj_new(typeof(*f2)); \
109 bpf_spin_unlock(B); \
113 CHECK(kptr_kptr, &f1->lock, &f2->lock);
114 CHECK(kptr_global, &f1->lock, &lockA);
115 CHECK(kptr_mapval, &f1->lock, &v->lock);
116 CHECK(kptr_innermapval, &f1->lock, &iv->lock);
118 CHECK(global_global, &lockA, &lockB);
119 CHECK(global_kptr, &lockA, &f1->lock);
120 CHECK(global_mapval, &lockA, &v->lock);
121 CHECK(global_innermapval, &lockA, &iv->lock);
124 int lock_id_mismatch_mapval_mapval(void *ctx)
129 f1 = bpf_map_lookup_elem(&array_map, &key);
132 f2 = bpf_map_lookup_elem(&array_map, &key);
136 bpf_spin_lock(&f1->lock);
138 bpf_spin_unlock(&f2->lock);
143 CHECK(mapval_kptr, &v->lock, &f1->lock);
144 CHECK(mapval_global, &v->lock, &lockB);
145 CHECK(mapval_innermapval, &v->lock, &iv->lock);
148 int lock_id_mismatch_innermapval_innermapval1(void *ctx)
154 map = bpf_map_lookup_elem(&map_of_maps, &key);
157 f1 = bpf_map_lookup_elem(map, &key);
160 f2 = bpf_map_lookup_elem(map, &key);
164 bpf_spin_lock(&f1->lock);
166 bpf_spin_unlock(&f2->lock);
172 int lock_id_mismatch_innermapval_innermapval2(void *ctx)
178 map = bpf_map_lookup_elem(&map_of_maps, &key);
181 f1 = bpf_map_lookup_elem(map, &key);
184 map = bpf_map_lookup_elem(&map_of_maps, &key);
187 f2 = bpf_map_lookup_elem(map, &key);
191 bpf_spin_lock(&f1->lock);
193 bpf_spin_unlock(&f2->lock);
198 CHECK(innermapval_kptr, &iv->lock, &f1->lock);
199 CHECK(innermapval_global, &iv->lock, &lockA);
200 CHECK(innermapval_mapval, &iv->lock, &v->lock);
205 int global_subprog(struct __sk_buff *ctx)
207 volatile int ret = 0;
210 ret += ctx->protocol;
211 return ret + ctx->mark;
215 static int static_subprog_call_global(struct __sk_buff *ctx)
217 volatile int ret = 0;
221 return ret + ctx->len + global_subprog(ctx);
225 int lock_global_subprog_call1(struct __sk_buff *ctx)
229 bpf_spin_lock(&lockA);
231 ret = global_subprog(ctx);
232 bpf_spin_unlock(&lockA);
237 int lock_global_subprog_call2(struct __sk_buff *ctx)
241 bpf_spin_lock(&lockA);
243 ret = static_subprog_call_global(ctx);
244 bpf_spin_unlock(&lockA);
248 char _license[] SEC("license") = "GPL";