1*73087489SDave Marchevsky // SPDX-License-Identifier: GPL-2.0
2*73087489SDave Marchevsky /* Copyright (c) 2022 Meta Platforms, Inc. and affiliates. */
3*73087489SDave Marchevsky 
4*73087489SDave Marchevsky #include "vmlinux.h"
5*73087489SDave Marchevsky #include <bpf/bpf_helpers.h>
6*73087489SDave Marchevsky #include "bpf_misc.h"
7*73087489SDave Marchevsky 
8*73087489SDave Marchevsky #define HASHMAP_SZ 4194304
9*73087489SDave Marchevsky 
10*73087489SDave Marchevsky struct {
11*73087489SDave Marchevsky 	__uint(type, BPF_MAP_TYPE_ARRAY_OF_MAPS);
12*73087489SDave Marchevsky 	__uint(max_entries, 1000);
13*73087489SDave Marchevsky 	__type(key, int);
14*73087489SDave Marchevsky 	__type(value, int);
15*73087489SDave Marchevsky 	__array(values, struct {
16*73087489SDave Marchevsky 		__uint(type, BPF_MAP_TYPE_TASK_STORAGE);
17*73087489SDave Marchevsky 		__uint(map_flags, BPF_F_NO_PREALLOC);
18*73087489SDave Marchevsky 		__type(key, int);
19*73087489SDave Marchevsky 		__type(value, int);
20*73087489SDave Marchevsky 	});
21*73087489SDave Marchevsky } array_of_local_storage_maps SEC(".maps");
22*73087489SDave Marchevsky 
23*73087489SDave Marchevsky struct {
24*73087489SDave Marchevsky 	__uint(type, BPF_MAP_TYPE_ARRAY_OF_MAPS);
25*73087489SDave Marchevsky 	__uint(max_entries, 1000);
26*73087489SDave Marchevsky 	__type(key, int);
27*73087489SDave Marchevsky 	__type(value, int);
28*73087489SDave Marchevsky 	__array(values, struct {
29*73087489SDave Marchevsky 		__uint(type, BPF_MAP_TYPE_HASH);
30*73087489SDave Marchevsky 		__uint(max_entries, HASHMAP_SZ);
31*73087489SDave Marchevsky 		__type(key, int);
32*73087489SDave Marchevsky 		__type(value, int);
33*73087489SDave Marchevsky 	});
34*73087489SDave Marchevsky } array_of_hash_maps SEC(".maps");
35*73087489SDave Marchevsky 
36*73087489SDave Marchevsky long important_hits;
37*73087489SDave Marchevsky long hits;
38*73087489SDave Marchevsky 
39*73087489SDave Marchevsky /* set from user-space */
40*73087489SDave Marchevsky const volatile unsigned int use_hashmap;
41*73087489SDave Marchevsky const volatile unsigned int hashmap_num_keys;
42*73087489SDave Marchevsky const volatile unsigned int num_maps;
43*73087489SDave Marchevsky const volatile unsigned int interleave;
44*73087489SDave Marchevsky 
45*73087489SDave Marchevsky struct loop_ctx {
46*73087489SDave Marchevsky 	struct task_struct *task;
47*73087489SDave Marchevsky 	long loop_hits;
48*73087489SDave Marchevsky 	long loop_important_hits;
49*73087489SDave Marchevsky };
50*73087489SDave Marchevsky 
do_lookup(unsigned int elem,struct loop_ctx * lctx)51*73087489SDave Marchevsky static int do_lookup(unsigned int elem, struct loop_ctx *lctx)
52*73087489SDave Marchevsky {
53*73087489SDave Marchevsky 	void *map, *inner_map;
54*73087489SDave Marchevsky 	int idx = 0;
55*73087489SDave Marchevsky 
56*73087489SDave Marchevsky 	if (use_hashmap)
57*73087489SDave Marchevsky 		map = &array_of_hash_maps;
58*73087489SDave Marchevsky 	else
59*73087489SDave Marchevsky 		map = &array_of_local_storage_maps;
60*73087489SDave Marchevsky 
61*73087489SDave Marchevsky 	inner_map = bpf_map_lookup_elem(map, &elem);
62*73087489SDave Marchevsky 	if (!inner_map)
63*73087489SDave Marchevsky 		return -1;
64*73087489SDave Marchevsky 
65*73087489SDave Marchevsky 	if (use_hashmap) {
66*73087489SDave Marchevsky 		idx = bpf_get_prandom_u32() % hashmap_num_keys;
67*73087489SDave Marchevsky 		bpf_map_lookup_elem(inner_map, &idx);
68*73087489SDave Marchevsky 	} else {
69*73087489SDave Marchevsky 		bpf_task_storage_get(inner_map, lctx->task, &idx,
70*73087489SDave Marchevsky 				     BPF_LOCAL_STORAGE_GET_F_CREATE);
71*73087489SDave Marchevsky 	}
72*73087489SDave Marchevsky 
73*73087489SDave Marchevsky 	lctx->loop_hits++;
74*73087489SDave Marchevsky 	if (!elem)
75*73087489SDave Marchevsky 		lctx->loop_important_hits++;
76*73087489SDave Marchevsky 	return 0;
77*73087489SDave Marchevsky }
78*73087489SDave Marchevsky 
loop(u32 index,void * ctx)79*73087489SDave Marchevsky static long loop(u32 index, void *ctx)
80*73087489SDave Marchevsky {
81*73087489SDave Marchevsky 	struct loop_ctx *lctx = (struct loop_ctx *)ctx;
82*73087489SDave Marchevsky 	unsigned int map_idx = index % num_maps;
83*73087489SDave Marchevsky 
84*73087489SDave Marchevsky 	do_lookup(map_idx, lctx);
85*73087489SDave Marchevsky 	if (interleave && map_idx % 3 == 0)
86*73087489SDave Marchevsky 		do_lookup(0, lctx);
87*73087489SDave Marchevsky 	return 0;
88*73087489SDave Marchevsky }
89*73087489SDave Marchevsky 
90*73087489SDave Marchevsky SEC("fentry/" SYS_PREFIX "sys_getpgid")
get_local(void * ctx)91*73087489SDave Marchevsky int get_local(void *ctx)
92*73087489SDave Marchevsky {
93*73087489SDave Marchevsky 	struct loop_ctx lctx;
94*73087489SDave Marchevsky 
95*73087489SDave Marchevsky 	lctx.task = bpf_get_current_task_btf();
96*73087489SDave Marchevsky 	lctx.loop_hits = 0;
97*73087489SDave Marchevsky 	lctx.loop_important_hits = 0;
98*73087489SDave Marchevsky 	bpf_loop(10000, &loop, &lctx, 0);
99*73087489SDave Marchevsky 	__sync_add_and_fetch(&hits, lctx.loop_hits);
100*73087489SDave Marchevsky 	__sync_add_and_fetch(&important_hits, lctx.loop_important_hits);
101*73087489SDave Marchevsky 	return 0;
102*73087489SDave Marchevsky }
103*73087489SDave Marchevsky 
104*73087489SDave Marchevsky char _license[] SEC("license") = "GPL";
105