Loading...
Note: File does not exist in v3.15.
1// SPDX-License-Identifier: GPL-2.0
2#define _GNU_SOURCE
3#include <pthread.h>
4#include <sched.h>
5#include <sys/socket.h>
6#include <test_progs.h>
7#include "test_perf_buffer.skel.h"
8#include "bpf/libbpf_internal.h"
9
10/* AddressSanitizer sometimes crashes due to data dereference below, due to
11 * this being mmap()'ed memory. Disable instrumentation with
12 * no_sanitize_address attribute
13 */
14__attribute__((no_sanitize_address))
15static void on_sample(void *ctx, int cpu, void *data, __u32 size)
16{
17 int cpu_data = *(int *)data, duration = 0;
18 cpu_set_t *cpu_seen = ctx;
19
20 if (cpu_data != cpu)
21 CHECK(cpu_data != cpu, "check_cpu_data",
22 "cpu_data %d != cpu %d\n", cpu_data, cpu);
23
24 CPU_SET(cpu, cpu_seen);
25}
26
27void test_perf_buffer(void)
28{
29 int err, on_len, nr_on_cpus = 0, nr_cpus, i, duration = 0;
30 struct perf_buffer_opts pb_opts = {};
31 struct test_perf_buffer *skel;
32 cpu_set_t cpu_set, cpu_seen;
33 struct perf_buffer *pb;
34 bool *online;
35
36 nr_cpus = libbpf_num_possible_cpus();
37 if (CHECK(nr_cpus < 0, "nr_cpus", "err %d\n", nr_cpus))
38 return;
39
40 err = parse_cpu_mask_file("/sys/devices/system/cpu/online",
41 &online, &on_len);
42 if (CHECK(err, "nr_on_cpus", "err %d\n", err))
43 return;
44
45 for (i = 0; i < on_len; i++)
46 if (online[i])
47 nr_on_cpus++;
48
49 /* load program */
50 skel = test_perf_buffer__open_and_load();
51 if (CHECK(!skel, "skel_load", "skeleton open/load failed\n"))
52 goto out_close;
53
54 /* attach probe */
55 err = test_perf_buffer__attach(skel);
56 if (CHECK(err, "attach_kprobe", "err %d\n", err))
57 goto out_close;
58
59 /* set up perf buffer */
60 pb_opts.sample_cb = on_sample;
61 pb_opts.ctx = &cpu_seen;
62 pb = perf_buffer__new(bpf_map__fd(skel->maps.perf_buf_map), 1, &pb_opts);
63 if (CHECK(IS_ERR(pb), "perf_buf__new", "err %ld\n", PTR_ERR(pb)))
64 goto out_close;
65
66 /* trigger kprobe on every CPU */
67 CPU_ZERO(&cpu_seen);
68 for (i = 0; i < nr_cpus; i++) {
69 if (i >= on_len || !online[i]) {
70 printf("skipping offline CPU #%d\n", i);
71 continue;
72 }
73
74 CPU_ZERO(&cpu_set);
75 CPU_SET(i, &cpu_set);
76
77 err = pthread_setaffinity_np(pthread_self(), sizeof(cpu_set),
78 &cpu_set);
79 if (err && CHECK(err, "set_affinity", "cpu #%d, err %d\n",
80 i, err))
81 goto out_close;
82
83 usleep(1);
84 }
85
86 /* read perf buffer */
87 err = perf_buffer__poll(pb, 100);
88 if (CHECK(err < 0, "perf_buffer__poll", "err %d\n", err))
89 goto out_free_pb;
90
91 if (CHECK(CPU_COUNT(&cpu_seen) != nr_on_cpus, "seen_cpu_cnt",
92 "expect %d, seen %d\n", nr_on_cpus, CPU_COUNT(&cpu_seen)))
93 goto out_free_pb;
94
95out_free_pb:
96 perf_buffer__free(pb);
97out_close:
98 test_perf_buffer__destroy(skel);
99 free(online);
100}