| // SPDX-License-Identifier: GPL-2.0-only | 
 | #include <stdio.h> | 
 | #include <fcntl.h> | 
 | #include <poll.h> | 
 | #include <time.h> | 
 | #include <signal.h> | 
 | #include <bpf/libbpf.h> | 
 |  | 
 | static __u64 time_get_ns(void) | 
 | { | 
 | 	struct timespec ts; | 
 |  | 
 | 	clock_gettime(CLOCK_MONOTONIC, &ts); | 
 | 	return ts.tv_sec * 1000000000ull + ts.tv_nsec; | 
 | } | 
 |  | 
 | static __u64 start_time; | 
 | static __u64 cnt; | 
 |  | 
 | #define MAX_CNT 100000ll | 
 |  | 
 | static void print_bpf_output(void *ctx, int cpu, void *data, __u32 size) | 
 | { | 
 | 	struct { | 
 | 		__u64 pid; | 
 | 		__u64 cookie; | 
 | 	} *e = data; | 
 |  | 
 | 	if (e->cookie != 0x12345678) { | 
 | 		printf("BUG pid %llx cookie %llx sized %d\n", | 
 | 		       e->pid, e->cookie, size); | 
 | 		return; | 
 | 	} | 
 |  | 
 | 	cnt++; | 
 |  | 
 | 	if (cnt == MAX_CNT) { | 
 | 		printf("recv %lld events per sec\n", | 
 | 		       MAX_CNT * 1000000000ll / (time_get_ns() - start_time)); | 
 | 		return; | 
 | 	} | 
 | } | 
 |  | 
 | int main(int argc, char **argv) | 
 | { | 
 | 	struct perf_buffer_opts pb_opts = {}; | 
 | 	struct bpf_link *link = NULL; | 
 | 	struct bpf_program *prog; | 
 | 	struct perf_buffer *pb; | 
 | 	struct bpf_object *obj; | 
 | 	int map_fd, ret = 0; | 
 | 	char filename[256]; | 
 | 	FILE *f; | 
 |  | 
 | 	snprintf(filename, sizeof(filename), "%s_kern.o", argv[0]); | 
 | 	obj = bpf_object__open_file(filename, NULL); | 
 | 	if (libbpf_get_error(obj)) { | 
 | 		fprintf(stderr, "ERROR: opening BPF object file failed\n"); | 
 | 		return 0; | 
 | 	} | 
 |  | 
 | 	/* load BPF program */ | 
 | 	if (bpf_object__load(obj)) { | 
 | 		fprintf(stderr, "ERROR: loading BPF object file failed\n"); | 
 | 		goto cleanup; | 
 | 	} | 
 |  | 
 | 	map_fd = bpf_object__find_map_fd_by_name(obj, "my_map"); | 
 | 	if (map_fd < 0) { | 
 | 		fprintf(stderr, "ERROR: finding a map in obj file failed\n"); | 
 | 		goto cleanup; | 
 | 	} | 
 |  | 
 | 	prog = bpf_object__find_program_by_name(obj, "bpf_prog1"); | 
 | 	if (libbpf_get_error(prog)) { | 
 | 		fprintf(stderr, "ERROR: finding a prog in obj file failed\n"); | 
 | 		goto cleanup; | 
 | 	} | 
 |  | 
 | 	link = bpf_program__attach(prog); | 
 | 	if (libbpf_get_error(link)) { | 
 | 		fprintf(stderr, "ERROR: bpf_program__attach failed\n"); | 
 | 		link = NULL; | 
 | 		goto cleanup; | 
 | 	} | 
 |  | 
 | 	pb_opts.sample_cb = print_bpf_output; | 
 | 	pb = perf_buffer__new(map_fd, 8, &pb_opts); | 
 | 	ret = libbpf_get_error(pb); | 
 | 	if (ret) { | 
 | 		printf("failed to setup perf_buffer: %d\n", ret); | 
 | 		return 1; | 
 | 	} | 
 |  | 
 | 	f = popen("taskset 1 dd if=/dev/zero of=/dev/null", "r"); | 
 | 	(void) f; | 
 |  | 
 | 	start_time = time_get_ns(); | 
 | 	while ((ret = perf_buffer__poll(pb, 1000)) >= 0 && cnt < MAX_CNT) { | 
 | 	} | 
 | 	kill(0, SIGINT); | 
 |  | 
 | cleanup: | 
 | 	bpf_link__destroy(link); | 
 | 	bpf_object__close(obj); | 
 | 	return ret; | 
 | } |