]> Git Repo - J-linux.git/blob - tools/lib/bpf/features.c
Merge tag 'vfs-6.13-rc7.fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs
[J-linux.git] / tools / lib / bpf / features.c
1 // SPDX-License-Identifier: (LGPL-2.1 OR BSD-2-Clause)
2 /* Copyright (c) 2023 Meta Platforms, Inc. and affiliates. */
3 #include <linux/kernel.h>
4 #include <linux/filter.h>
5 #include "bpf.h"
6 #include "libbpf.h"
7 #include "libbpf_common.h"
8 #include "libbpf_internal.h"
9 #include "str_error.h"
10
11 static inline __u64 ptr_to_u64(const void *ptr)
12 {
13         return (__u64)(unsigned long)ptr;
14 }
15
16 int probe_fd(int fd)
17 {
18         if (fd >= 0)
19                 close(fd);
20         return fd >= 0;
21 }
22
23 static int probe_kern_prog_name(int token_fd)
24 {
25         const size_t attr_sz = offsetofend(union bpf_attr, prog_token_fd);
26         struct bpf_insn insns[] = {
27                 BPF_MOV64_IMM(BPF_REG_0, 0),
28                 BPF_EXIT_INSN(),
29         };
30         union bpf_attr attr;
31         int ret;
32
33         memset(&attr, 0, attr_sz);
34         attr.prog_type = BPF_PROG_TYPE_SOCKET_FILTER;
35         attr.license = ptr_to_u64("GPL");
36         attr.insns = ptr_to_u64(insns);
37         attr.insn_cnt = (__u32)ARRAY_SIZE(insns);
38         attr.prog_token_fd = token_fd;
39         if (token_fd)
40                 attr.prog_flags |= BPF_F_TOKEN_FD;
41         libbpf_strlcpy(attr.prog_name, "libbpf_nametest", sizeof(attr.prog_name));
42
43         /* make sure loading with name works */
44         ret = sys_bpf_prog_load(&attr, attr_sz, PROG_LOAD_ATTEMPTS);
45         return probe_fd(ret);
46 }
47
48 static int probe_kern_global_data(int token_fd)
49 {
50         struct bpf_insn insns[] = {
51                 BPF_LD_MAP_VALUE(BPF_REG_1, 0, 16),
52                 BPF_ST_MEM(BPF_DW, BPF_REG_1, 0, 42),
53                 BPF_MOV64_IMM(BPF_REG_0, 0),
54                 BPF_EXIT_INSN(),
55         };
56         LIBBPF_OPTS(bpf_map_create_opts, map_opts,
57                 .token_fd = token_fd,
58                 .map_flags = token_fd ? BPF_F_TOKEN_FD : 0,
59         );
60         LIBBPF_OPTS(bpf_prog_load_opts, prog_opts,
61                 .token_fd = token_fd,
62                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
63         );
64         int ret, map, insn_cnt = ARRAY_SIZE(insns);
65
66         map = bpf_map_create(BPF_MAP_TYPE_ARRAY, "libbpf_global", sizeof(int), 32, 1, &map_opts);
67         if (map < 0) {
68                 ret = -errno;
69                 pr_warn("Error in %s(): %s. Couldn't create simple array map.\n",
70                         __func__, errstr(ret));
71                 return ret;
72         }
73
74         insns[0].imm = map;
75
76         ret = bpf_prog_load(BPF_PROG_TYPE_SOCKET_FILTER, NULL, "GPL", insns, insn_cnt, &prog_opts);
77         close(map);
78         return probe_fd(ret);
79 }
80
81 static int probe_kern_btf(int token_fd)
82 {
83         static const char strs[] = "\0int";
84         __u32 types[] = {
85                 /* int */
86                 BTF_TYPE_INT_ENC(1, BTF_INT_SIGNED, 0, 32, 4),
87         };
88
89         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
90                                              strs, sizeof(strs), token_fd));
91 }
92
93 static int probe_kern_btf_func(int token_fd)
94 {
95         static const char strs[] = "\0int\0x\0a";
96         /* void x(int a) {} */
97         __u32 types[] = {
98                 /* int */
99                 BTF_TYPE_INT_ENC(1, BTF_INT_SIGNED, 0, 32, 4),  /* [1] */
100                 /* FUNC_PROTO */                                /* [2] */
101                 BTF_TYPE_ENC(0, BTF_INFO_ENC(BTF_KIND_FUNC_PROTO, 0, 1), 0),
102                 BTF_PARAM_ENC(7, 1),
103                 /* FUNC x */                                    /* [3] */
104                 BTF_TYPE_ENC(5, BTF_INFO_ENC(BTF_KIND_FUNC, 0, 0), 2),
105         };
106
107         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
108                                              strs, sizeof(strs), token_fd));
109 }
110
111 static int probe_kern_btf_func_global(int token_fd)
112 {
113         static const char strs[] = "\0int\0x\0a";
114         /* static void x(int a) {} */
115         __u32 types[] = {
116                 /* int */
117                 BTF_TYPE_INT_ENC(1, BTF_INT_SIGNED, 0, 32, 4),  /* [1] */
118                 /* FUNC_PROTO */                                /* [2] */
119                 BTF_TYPE_ENC(0, BTF_INFO_ENC(BTF_KIND_FUNC_PROTO, 0, 1), 0),
120                 BTF_PARAM_ENC(7, 1),
121                 /* FUNC x BTF_FUNC_GLOBAL */                    /* [3] */
122                 BTF_TYPE_ENC(5, BTF_INFO_ENC(BTF_KIND_FUNC, 0, BTF_FUNC_GLOBAL), 2),
123         };
124
125         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
126                                              strs, sizeof(strs), token_fd));
127 }
128
129 static int probe_kern_btf_datasec(int token_fd)
130 {
131         static const char strs[] = "\0x\0.data";
132         /* static int a; */
133         __u32 types[] = {
134                 /* int */
135                 BTF_TYPE_INT_ENC(0, BTF_INT_SIGNED, 0, 32, 4),  /* [1] */
136                 /* VAR x */                                     /* [2] */
137                 BTF_TYPE_ENC(1, BTF_INFO_ENC(BTF_KIND_VAR, 0, 0), 1),
138                 BTF_VAR_STATIC,
139                 /* DATASEC val */                               /* [3] */
140                 BTF_TYPE_ENC(3, BTF_INFO_ENC(BTF_KIND_DATASEC, 0, 1), 4),
141                 BTF_VAR_SECINFO_ENC(2, 0, 4),
142         };
143
144         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
145                                              strs, sizeof(strs), token_fd));
146 }
147
148 static int probe_kern_btf_qmark_datasec(int token_fd)
149 {
150         static const char strs[] = "\0x\0?.data";
151         /* static int a; */
152         __u32 types[] = {
153                 /* int */
154                 BTF_TYPE_INT_ENC(0, BTF_INT_SIGNED, 0, 32, 4),  /* [1] */
155                 /* VAR x */                                     /* [2] */
156                 BTF_TYPE_ENC(1, BTF_INFO_ENC(BTF_KIND_VAR, 0, 0), 1),
157                 BTF_VAR_STATIC,
158                 /* DATASEC ?.data */                            /* [3] */
159                 BTF_TYPE_ENC(3, BTF_INFO_ENC(BTF_KIND_DATASEC, 0, 1), 4),
160                 BTF_VAR_SECINFO_ENC(2, 0, 4),
161         };
162
163         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
164                                              strs, sizeof(strs), token_fd));
165 }
166
167 static int probe_kern_btf_float(int token_fd)
168 {
169         static const char strs[] = "\0float";
170         __u32 types[] = {
171                 /* float */
172                 BTF_TYPE_FLOAT_ENC(1, 4),
173         };
174
175         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
176                                              strs, sizeof(strs), token_fd));
177 }
178
179 static int probe_kern_btf_decl_tag(int token_fd)
180 {
181         static const char strs[] = "\0tag";
182         __u32 types[] = {
183                 /* int */
184                 BTF_TYPE_INT_ENC(0, BTF_INT_SIGNED, 0, 32, 4),  /* [1] */
185                 /* VAR x */                                     /* [2] */
186                 BTF_TYPE_ENC(1, BTF_INFO_ENC(BTF_KIND_VAR, 0, 0), 1),
187                 BTF_VAR_STATIC,
188                 /* attr */
189                 BTF_TYPE_DECL_TAG_ENC(1, 2, -1),
190         };
191
192         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
193                                              strs, sizeof(strs), token_fd));
194 }
195
196 static int probe_kern_btf_type_tag(int token_fd)
197 {
198         static const char strs[] = "\0tag";
199         __u32 types[] = {
200                 /* int */
201                 BTF_TYPE_INT_ENC(0, BTF_INT_SIGNED, 0, 32, 4),          /* [1] */
202                 /* attr */
203                 BTF_TYPE_TYPE_TAG_ENC(1, 1),                            /* [2] */
204                 /* ptr */
205                 BTF_TYPE_ENC(0, BTF_INFO_ENC(BTF_KIND_PTR, 0, 0), 2),   /* [3] */
206         };
207
208         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
209                                              strs, sizeof(strs), token_fd));
210 }
211
212 static int probe_kern_array_mmap(int token_fd)
213 {
214         LIBBPF_OPTS(bpf_map_create_opts, opts,
215                 .map_flags = BPF_F_MMAPABLE | (token_fd ? BPF_F_TOKEN_FD : 0),
216                 .token_fd = token_fd,
217         );
218         int fd;
219
220         fd = bpf_map_create(BPF_MAP_TYPE_ARRAY, "libbpf_mmap", sizeof(int), sizeof(int), 1, &opts);
221         return probe_fd(fd);
222 }
223
224 static int probe_kern_exp_attach_type(int token_fd)
225 {
226         LIBBPF_OPTS(bpf_prog_load_opts, opts,
227                 .expected_attach_type = BPF_CGROUP_INET_SOCK_CREATE,
228                 .token_fd = token_fd,
229                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
230         );
231         struct bpf_insn insns[] = {
232                 BPF_MOV64_IMM(BPF_REG_0, 0),
233                 BPF_EXIT_INSN(),
234         };
235         int fd, insn_cnt = ARRAY_SIZE(insns);
236
237         /* use any valid combination of program type and (optional)
238          * non-zero expected attach type (i.e., not a BPF_CGROUP_INET_INGRESS)
239          * to see if kernel supports expected_attach_type field for
240          * BPF_PROG_LOAD command
241          */
242         fd = bpf_prog_load(BPF_PROG_TYPE_CGROUP_SOCK, NULL, "GPL", insns, insn_cnt, &opts);
243         return probe_fd(fd);
244 }
245
246 static int probe_kern_probe_read_kernel(int token_fd)
247 {
248         LIBBPF_OPTS(bpf_prog_load_opts, opts,
249                 .token_fd = token_fd,
250                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
251         );
252         struct bpf_insn insns[] = {
253                 BPF_MOV64_REG(BPF_REG_1, BPF_REG_10),   /* r1 = r10 (fp) */
254                 BPF_ALU64_IMM(BPF_ADD, BPF_REG_1, -8),  /* r1 += -8 */
255                 BPF_MOV64_IMM(BPF_REG_2, 8),            /* r2 = 8 */
256                 BPF_MOV64_IMM(BPF_REG_3, 0),            /* r3 = 0 */
257                 BPF_RAW_INSN(BPF_JMP | BPF_CALL, 0, 0, 0, BPF_FUNC_probe_read_kernel),
258                 BPF_EXIT_INSN(),
259         };
260         int fd, insn_cnt = ARRAY_SIZE(insns);
261
262         fd = bpf_prog_load(BPF_PROG_TYPE_TRACEPOINT, NULL, "GPL", insns, insn_cnt, &opts);
263         return probe_fd(fd);
264 }
265
266 static int probe_prog_bind_map(int token_fd)
267 {
268         struct bpf_insn insns[] = {
269                 BPF_MOV64_IMM(BPF_REG_0, 0),
270                 BPF_EXIT_INSN(),
271         };
272         LIBBPF_OPTS(bpf_map_create_opts, map_opts,
273                 .token_fd = token_fd,
274                 .map_flags = token_fd ? BPF_F_TOKEN_FD : 0,
275         );
276         LIBBPF_OPTS(bpf_prog_load_opts, prog_opts,
277                 .token_fd = token_fd,
278                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
279         );
280         int ret, map, prog, insn_cnt = ARRAY_SIZE(insns);
281
282         map = bpf_map_create(BPF_MAP_TYPE_ARRAY, "libbpf_det_bind", sizeof(int), 32, 1, &map_opts);
283         if (map < 0) {
284                 ret = -errno;
285                 pr_warn("Error in %s(): %s. Couldn't create simple array map.\n",
286                         __func__, errstr(ret));
287                 return ret;
288         }
289
290         prog = bpf_prog_load(BPF_PROG_TYPE_SOCKET_FILTER, NULL, "GPL", insns, insn_cnt, &prog_opts);
291         if (prog < 0) {
292                 close(map);
293                 return 0;
294         }
295
296         ret = bpf_prog_bind_map(prog, map, NULL);
297
298         close(map);
299         close(prog);
300
301         return ret >= 0;
302 }
303
304 static int probe_module_btf(int token_fd)
305 {
306         static const char strs[] = "\0int";
307         __u32 types[] = {
308                 /* int */
309                 BTF_TYPE_INT_ENC(1, BTF_INT_SIGNED, 0, 32, 4),
310         };
311         struct bpf_btf_info info;
312         __u32 len = sizeof(info);
313         char name[16];
314         int fd, err;
315
316         fd = libbpf__load_raw_btf((char *)types, sizeof(types), strs, sizeof(strs), token_fd);
317         if (fd < 0)
318                 return 0; /* BTF not supported at all */
319
320         memset(&info, 0, sizeof(info));
321         info.name = ptr_to_u64(name);
322         info.name_len = sizeof(name);
323
324         /* check that BPF_OBJ_GET_INFO_BY_FD supports specifying name pointer;
325          * kernel's module BTF support coincides with support for
326          * name/name_len fields in struct bpf_btf_info.
327          */
328         err = bpf_btf_get_info_by_fd(fd, &info, &len);
329         close(fd);
330         return !err;
331 }
332
333 static int probe_perf_link(int token_fd)
334 {
335         struct bpf_insn insns[] = {
336                 BPF_MOV64_IMM(BPF_REG_0, 0),
337                 BPF_EXIT_INSN(),
338         };
339         LIBBPF_OPTS(bpf_prog_load_opts, opts,
340                 .token_fd = token_fd,
341                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
342         );
343         int prog_fd, link_fd, err;
344
345         prog_fd = bpf_prog_load(BPF_PROG_TYPE_TRACEPOINT, NULL, "GPL",
346                                 insns, ARRAY_SIZE(insns), &opts);
347         if (prog_fd < 0)
348                 return -errno;
349
350         /* use invalid perf_event FD to get EBADF, if link is supported;
351          * otherwise EINVAL should be returned
352          */
353         link_fd = bpf_link_create(prog_fd, -1, BPF_PERF_EVENT, NULL);
354         err = -errno; /* close() can clobber errno */
355
356         if (link_fd >= 0)
357                 close(link_fd);
358         close(prog_fd);
359
360         return link_fd < 0 && err == -EBADF;
361 }
362
363 static int probe_uprobe_multi_link(int token_fd)
364 {
365         LIBBPF_OPTS(bpf_prog_load_opts, load_opts,
366                 .expected_attach_type = BPF_TRACE_UPROBE_MULTI,
367                 .token_fd = token_fd,
368                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
369         );
370         LIBBPF_OPTS(bpf_link_create_opts, link_opts);
371         struct bpf_insn insns[] = {
372                 BPF_MOV64_IMM(BPF_REG_0, 0),
373                 BPF_EXIT_INSN(),
374         };
375         int prog_fd, link_fd, err;
376         unsigned long offset = 0;
377
378         prog_fd = bpf_prog_load(BPF_PROG_TYPE_KPROBE, NULL, "GPL",
379                                 insns, ARRAY_SIZE(insns), &load_opts);
380         if (prog_fd < 0)
381                 return -errno;
382
383         /* Creating uprobe in '/' binary should fail with -EBADF. */
384         link_opts.uprobe_multi.path = "/";
385         link_opts.uprobe_multi.offsets = &offset;
386         link_opts.uprobe_multi.cnt = 1;
387
388         link_fd = bpf_link_create(prog_fd, -1, BPF_TRACE_UPROBE_MULTI, &link_opts);
389         err = -errno; /* close() can clobber errno */
390
391         if (link_fd >= 0 || err != -EBADF) {
392                 if (link_fd >= 0)
393                         close(link_fd);
394                 close(prog_fd);
395                 return 0;
396         }
397
398         /* Initial multi-uprobe support in kernel didn't handle PID filtering
399          * correctly (it was doing thread filtering, not process filtering).
400          * So now we'll detect if PID filtering logic was fixed, and, if not,
401          * we'll pretend multi-uprobes are not supported, if not.
402          * Multi-uprobes are used in USDT attachment logic, and we need to be
403          * conservative here, because multi-uprobe selection happens early at
404          * load time, while the use of PID filtering is known late at
405          * attachment time, at which point it's too late to undo multi-uprobe
406          * selection.
407          *
408          * Creating uprobe with pid == -1 for (invalid) '/' binary will fail
409          * early with -EINVAL on kernels with fixed PID filtering logic;
410          * otherwise -ESRCH would be returned if passed correct binary path
411          * (but we'll just get -BADF, of course).
412          */
413         link_opts.uprobe_multi.pid = -1; /* invalid PID */
414         link_opts.uprobe_multi.path = "/"; /* invalid path */
415         link_opts.uprobe_multi.offsets = &offset;
416         link_opts.uprobe_multi.cnt = 1;
417
418         link_fd = bpf_link_create(prog_fd, -1, BPF_TRACE_UPROBE_MULTI, &link_opts);
419         err = -errno; /* close() can clobber errno */
420
421         if (link_fd >= 0)
422                 close(link_fd);
423         close(prog_fd);
424
425         return link_fd < 0 && err == -EINVAL;
426 }
427
428 static int probe_kern_bpf_cookie(int token_fd)
429 {
430         struct bpf_insn insns[] = {
431                 BPF_RAW_INSN(BPF_JMP | BPF_CALL, 0, 0, 0, BPF_FUNC_get_attach_cookie),
432                 BPF_EXIT_INSN(),
433         };
434         LIBBPF_OPTS(bpf_prog_load_opts, opts,
435                 .token_fd = token_fd,
436                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
437         );
438         int ret, insn_cnt = ARRAY_SIZE(insns);
439
440         ret = bpf_prog_load(BPF_PROG_TYPE_TRACEPOINT, NULL, "GPL", insns, insn_cnt, &opts);
441         return probe_fd(ret);
442 }
443
444 static int probe_kern_btf_enum64(int token_fd)
445 {
446         static const char strs[] = "\0enum64";
447         __u32 types[] = {
448                 BTF_TYPE_ENC(1, BTF_INFO_ENC(BTF_KIND_ENUM64, 0, 0), 8),
449         };
450
451         return probe_fd(libbpf__load_raw_btf((char *)types, sizeof(types),
452                                              strs, sizeof(strs), token_fd));
453 }
454
455 static int probe_kern_arg_ctx_tag(int token_fd)
456 {
457         static const char strs[] = "\0a\0b\0arg:ctx\0";
458         const __u32 types[] = {
459                 /* [1] INT */
460                 BTF_TYPE_INT_ENC(1 /* "a" */, BTF_INT_SIGNED, 0, 32, 4),
461                 /* [2] PTR -> VOID */
462                 BTF_TYPE_ENC(0, BTF_INFO_ENC(BTF_KIND_PTR, 0, 0), 0),
463                 /* [3] FUNC_PROTO `int(void *a)` */
464                 BTF_TYPE_ENC(0, BTF_INFO_ENC(BTF_KIND_FUNC_PROTO, 0, 1), 1),
465                 BTF_PARAM_ENC(1 /* "a" */, 2),
466                 /* [4] FUNC 'a' -> FUNC_PROTO (main prog) */
467                 BTF_TYPE_ENC(1 /* "a" */, BTF_INFO_ENC(BTF_KIND_FUNC, 0, BTF_FUNC_GLOBAL), 3),
468                 /* [5] FUNC_PROTO `int(void *b __arg_ctx)` */
469                 BTF_TYPE_ENC(0, BTF_INFO_ENC(BTF_KIND_FUNC_PROTO, 0, 1), 1),
470                 BTF_PARAM_ENC(3 /* "b" */, 2),
471                 /* [6] FUNC 'b' -> FUNC_PROTO (subprog) */
472                 BTF_TYPE_ENC(3 /* "b" */, BTF_INFO_ENC(BTF_KIND_FUNC, 0, BTF_FUNC_GLOBAL), 5),
473                 /* [7] DECL_TAG 'arg:ctx' -> func 'b' arg 'b' */
474                 BTF_TYPE_DECL_TAG_ENC(5 /* "arg:ctx" */, 6, 0),
475         };
476         const struct bpf_insn insns[] = {
477                 /* main prog */
478                 BPF_CALL_REL(+1),
479                 BPF_EXIT_INSN(),
480                 /* global subprog */
481                 BPF_EMIT_CALL(BPF_FUNC_get_func_ip), /* needs PTR_TO_CTX */
482                 BPF_EXIT_INSN(),
483         };
484         const struct bpf_func_info_min func_infos[] = {
485                 { 0, 4 }, /* main prog -> FUNC 'a' */
486                 { 2, 6 }, /* subprog -> FUNC 'b' */
487         };
488         LIBBPF_OPTS(bpf_prog_load_opts, opts,
489                 .token_fd = token_fd,
490                 .prog_flags = token_fd ? BPF_F_TOKEN_FD : 0,
491         );
492         int prog_fd, btf_fd, insn_cnt = ARRAY_SIZE(insns);
493
494         btf_fd = libbpf__load_raw_btf((char *)types, sizeof(types), strs, sizeof(strs), token_fd);
495         if (btf_fd < 0)
496                 return 0;
497
498         opts.prog_btf_fd = btf_fd;
499         opts.func_info = &func_infos;
500         opts.func_info_cnt = ARRAY_SIZE(func_infos);
501         opts.func_info_rec_size = sizeof(func_infos[0]);
502
503         prog_fd = bpf_prog_load(BPF_PROG_TYPE_KPROBE, "det_arg_ctx",
504                                 "GPL", insns, insn_cnt, &opts);
505         close(btf_fd);
506
507         return probe_fd(prog_fd);
508 }
509
510 typedef int (*feature_probe_fn)(int /* token_fd */);
511
512 static struct kern_feature_cache feature_cache;
513
514 static struct kern_feature_desc {
515         const char *desc;
516         feature_probe_fn probe;
517 } feature_probes[__FEAT_CNT] = {
518         [FEAT_PROG_NAME] = {
519                 "BPF program name", probe_kern_prog_name,
520         },
521         [FEAT_GLOBAL_DATA] = {
522                 "global variables", probe_kern_global_data,
523         },
524         [FEAT_BTF] = {
525                 "minimal BTF", probe_kern_btf,
526         },
527         [FEAT_BTF_FUNC] = {
528                 "BTF functions", probe_kern_btf_func,
529         },
530         [FEAT_BTF_GLOBAL_FUNC] = {
531                 "BTF global function", probe_kern_btf_func_global,
532         },
533         [FEAT_BTF_DATASEC] = {
534                 "BTF data section and variable", probe_kern_btf_datasec,
535         },
536         [FEAT_ARRAY_MMAP] = {
537                 "ARRAY map mmap()", probe_kern_array_mmap,
538         },
539         [FEAT_EXP_ATTACH_TYPE] = {
540                 "BPF_PROG_LOAD expected_attach_type attribute",
541                 probe_kern_exp_attach_type,
542         },
543         [FEAT_PROBE_READ_KERN] = {
544                 "bpf_probe_read_kernel() helper", probe_kern_probe_read_kernel,
545         },
546         [FEAT_PROG_BIND_MAP] = {
547                 "BPF_PROG_BIND_MAP support", probe_prog_bind_map,
548         },
549         [FEAT_MODULE_BTF] = {
550                 "module BTF support", probe_module_btf,
551         },
552         [FEAT_BTF_FLOAT] = {
553                 "BTF_KIND_FLOAT support", probe_kern_btf_float,
554         },
555         [FEAT_PERF_LINK] = {
556                 "BPF perf link support", probe_perf_link,
557         },
558         [FEAT_BTF_DECL_TAG] = {
559                 "BTF_KIND_DECL_TAG support", probe_kern_btf_decl_tag,
560         },
561         [FEAT_BTF_TYPE_TAG] = {
562                 "BTF_KIND_TYPE_TAG support", probe_kern_btf_type_tag,
563         },
564         [FEAT_MEMCG_ACCOUNT] = {
565                 "memcg-based memory accounting", probe_memcg_account,
566         },
567         [FEAT_BPF_COOKIE] = {
568                 "BPF cookie support", probe_kern_bpf_cookie,
569         },
570         [FEAT_BTF_ENUM64] = {
571                 "BTF_KIND_ENUM64 support", probe_kern_btf_enum64,
572         },
573         [FEAT_SYSCALL_WRAPPER] = {
574                 "Kernel using syscall wrapper", probe_kern_syscall_wrapper,
575         },
576         [FEAT_UPROBE_MULTI_LINK] = {
577                 "BPF multi-uprobe link support", probe_uprobe_multi_link,
578         },
579         [FEAT_ARG_CTX_TAG] = {
580                 "kernel-side __arg_ctx tag", probe_kern_arg_ctx_tag,
581         },
582         [FEAT_BTF_QMARK_DATASEC] = {
583                 "BTF DATASEC names starting from '?'", probe_kern_btf_qmark_datasec,
584         },
585 };
586
587 bool feat_supported(struct kern_feature_cache *cache, enum kern_feature_id feat_id)
588 {
589         struct kern_feature_desc *feat = &feature_probes[feat_id];
590         int ret;
591
592         /* assume global feature cache, unless custom one is provided */
593         if (!cache)
594                 cache = &feature_cache;
595
596         if (READ_ONCE(cache->res[feat_id]) == FEAT_UNKNOWN) {
597                 ret = feat->probe(cache->token_fd);
598                 if (ret > 0) {
599                         WRITE_ONCE(cache->res[feat_id], FEAT_SUPPORTED);
600                 } else if (ret == 0) {
601                         WRITE_ONCE(cache->res[feat_id], FEAT_MISSING);
602                 } else {
603                         pr_warn("Detection of kernel %s support failed: %s\n",
604                                 feat->desc, errstr(ret));
605                         WRITE_ONCE(cache->res[feat_id], FEAT_MISSING);
606                 }
607         }
608
609         return READ_ONCE(cache->res[feat_id]) == FEAT_SUPPORTED;
610 }
This page took 0.060776 seconds and 4 git commands to generate.