xref: /openbmc/linux/samples/bpf/test_current_task_under_cgroup_user.c (revision 9a87ffc99ec8eb8d35eed7c4f816d75f5cc9662e)
125763b3cSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
29e6e60ecSSargun Dhillon /* Copyright (c) 2016 Sargun Dhillon <sargun@sargun.me>
39e6e60ecSSargun Dhillon  */
49e6e60ecSSargun Dhillon 
59e6e60ecSSargun Dhillon #define _GNU_SOURCE
69e6e60ecSSargun Dhillon #include <stdio.h>
79e6e60ecSSargun Dhillon #include <unistd.h>
82bf3e2efSJakub Kicinski #include <bpf/bpf.h>
93677d0a1SDaniel T. Lee #include <bpf/libbpf.h>
101a922feeSSargun Dhillon #include "cgroup_helpers.h"
119e6e60ecSSargun Dhillon 
121a922feeSSargun Dhillon #define CGROUP_PATH		"/my-cgroup"
139e6e60ecSSargun Dhillon 
main(int argc,char ** argv)149e6e60ecSSargun Dhillon int main(int argc, char **argv)
159e6e60ecSSargun Dhillon {
169e6e60ecSSargun Dhillon 	pid_t remote_pid, local_pid = getpid();
1768be98e0SDaniel T. Lee 	int cg2 = -1, idx = 0, rc = 1;
183677d0a1SDaniel T. Lee 	struct bpf_link *link = NULL;
193677d0a1SDaniel T. Lee 	struct bpf_program *prog;
203677d0a1SDaniel T. Lee 	struct bpf_object *obj;
211a922feeSSargun Dhillon 	char filename[256];
223677d0a1SDaniel T. Lee 	int map_fd[2];
239e6e60ecSSargun Dhillon 
24*d4fffba4SDaniel T. Lee 	snprintf(filename, sizeof(filename), "%s.bpf.o", argv[0]);
253677d0a1SDaniel T. Lee 	obj = bpf_object__open_file(filename, NULL);
263677d0a1SDaniel T. Lee 	if (libbpf_get_error(obj)) {
273677d0a1SDaniel T. Lee 		fprintf(stderr, "ERROR: opening BPF object file failed\n");
283677d0a1SDaniel T. Lee 		return 0;
293677d0a1SDaniel T. Lee 	}
303677d0a1SDaniel T. Lee 
313677d0a1SDaniel T. Lee 	prog = bpf_object__find_program_by_name(obj, "bpf_prog1");
323677d0a1SDaniel T. Lee 	if (!prog) {
333677d0a1SDaniel T. Lee 		printf("finding a prog in obj file failed\n");
343677d0a1SDaniel T. Lee 		goto cleanup;
353677d0a1SDaniel T. Lee 	}
363677d0a1SDaniel T. Lee 
373677d0a1SDaniel T. Lee 	/* load BPF program */
383677d0a1SDaniel T. Lee 	if (bpf_object__load(obj)) {
393677d0a1SDaniel T. Lee 		fprintf(stderr, "ERROR: loading BPF object file failed\n");
403677d0a1SDaniel T. Lee 		goto cleanup;
413677d0a1SDaniel T. Lee 	}
423677d0a1SDaniel T. Lee 
433677d0a1SDaniel T. Lee 	map_fd[0] = bpf_object__find_map_fd_by_name(obj, "cgroup_map");
443677d0a1SDaniel T. Lee 	map_fd[1] = bpf_object__find_map_fd_by_name(obj, "perf_map");
453677d0a1SDaniel T. Lee 	if (map_fd[0] < 0 || map_fd[1] < 0) {
463677d0a1SDaniel T. Lee 		fprintf(stderr, "ERROR: finding a map in obj file failed\n");
473677d0a1SDaniel T. Lee 		goto cleanup;
483677d0a1SDaniel T. Lee 	}
493677d0a1SDaniel T. Lee 
503677d0a1SDaniel T. Lee 	link = bpf_program__attach(prog);
513677d0a1SDaniel T. Lee 	if (libbpf_get_error(link)) {
523677d0a1SDaniel T. Lee 		fprintf(stderr, "ERROR: bpf_program__attach failed\n");
533677d0a1SDaniel T. Lee 		link = NULL;
543677d0a1SDaniel T. Lee 		goto cleanup;
559e6e60ecSSargun Dhillon 	}
569e6e60ecSSargun Dhillon 
571a922feeSSargun Dhillon 	if (setup_cgroup_environment())
581a922feeSSargun Dhillon 		goto err;
599e6e60ecSSargun Dhillon 
601a922feeSSargun Dhillon 	cg2 = create_and_get_cgroup(CGROUP_PATH);
619e6e60ecSSargun Dhillon 
62a8911d6dSStanislav Fomichev 	if (cg2 < 0)
631a922feeSSargun Dhillon 		goto err;
649e6e60ecSSargun Dhillon 
65d40fc181SJoe Stringer 	if (bpf_map_update_elem(map_fd[0], &idx, &cg2, BPF_ANY)) {
669e6e60ecSSargun Dhillon 		log_err("Adding target cgroup to map");
671a922feeSSargun Dhillon 		goto err;
689e6e60ecSSargun Dhillon 	}
691a922feeSSargun Dhillon 
701a922feeSSargun Dhillon 	if (join_cgroup(CGROUP_PATH))
711a922feeSSargun Dhillon 		goto err;
729e6e60ecSSargun Dhillon 
739e6e60ecSSargun Dhillon 	/*
749e6e60ecSSargun Dhillon 	 * The installed helper program catched the sync call, and should
759e6e60ecSSargun Dhillon 	 * write it to the map.
769e6e60ecSSargun Dhillon 	 */
779e6e60ecSSargun Dhillon 
789e6e60ecSSargun Dhillon 	sync();
79d40fc181SJoe Stringer 	bpf_map_lookup_elem(map_fd[1], &idx, &remote_pid);
809e6e60ecSSargun Dhillon 
819e6e60ecSSargun Dhillon 	if (local_pid != remote_pid) {
829e6e60ecSSargun Dhillon 		fprintf(stderr,
839e6e60ecSSargun Dhillon 			"BPF Helper didn't write correct PID to map, but: %d\n",
849e6e60ecSSargun Dhillon 			remote_pid);
851a922feeSSargun Dhillon 		goto err;
869e6e60ecSSargun Dhillon 	}
879e6e60ecSSargun Dhillon 
889e6e60ecSSargun Dhillon 	/* Verify the negative scenario; leave the cgroup */
891a922feeSSargun Dhillon 	if (join_cgroup("/"))
901a922feeSSargun Dhillon 		goto err;
919e6e60ecSSargun Dhillon 
929e6e60ecSSargun Dhillon 	remote_pid = 0;
93d40fc181SJoe Stringer 	bpf_map_update_elem(map_fd[1], &idx, &remote_pid, BPF_ANY);
949e6e60ecSSargun Dhillon 
959e6e60ecSSargun Dhillon 	sync();
96d40fc181SJoe Stringer 	bpf_map_lookup_elem(map_fd[1], &idx, &remote_pid);
979e6e60ecSSargun Dhillon 
989e6e60ecSSargun Dhillon 	if (local_pid == remote_pid) {
999e6e60ecSSargun Dhillon 		fprintf(stderr, "BPF cgroup negative test did not work\n");
1001a922feeSSargun Dhillon 		goto err;
1019e6e60ecSSargun Dhillon 	}
1029e6e60ecSSargun Dhillon 
1033677d0a1SDaniel T. Lee 	rc = 0;
1049e6e60ecSSargun Dhillon 
1053677d0a1SDaniel T. Lee err:
10668be98e0SDaniel T. Lee 	if (cg2 != -1)
1071a922feeSSargun Dhillon 		close(cg2);
10868be98e0SDaniel T. Lee 
1091a922feeSSargun Dhillon 	cleanup_cgroup_environment();
1103677d0a1SDaniel T. Lee 
1113677d0a1SDaniel T. Lee cleanup:
1123677d0a1SDaniel T. Lee 	bpf_link__destroy(link);
1133677d0a1SDaniel T. Lee 	bpf_object__close(obj);
1141a922feeSSargun Dhillon 	return rc;
1159e6e60ecSSargun Dhillon }
116