Linux Audio

Check our new training course

Real-Time Linux with PREEMPT_RT training

Feb 18-20, 2025
Register
Loading...
v6.8
  1// SPDX-License-Identifier: GPL-2.0
  2#include <test_progs.h>
  3#include <sys/time.h>
  4#include <sys/resource.h>
  5#include "test_send_signal_kern.skel.h"
  6
  7static int sigusr1_received;
  8
  9static void sigusr1_handler(int signum)
 10{
 11	sigusr1_received = 1;
 12}
 13
 14static void test_send_signal_common(struct perf_event_attr *attr,
 15				    bool signal_thread)
 
 16{
 17	struct test_send_signal_kern *skel;
 18	int pipe_c2p[2], pipe_p2c[2];
 19	int err = -1, pmu_fd = -1;
 
 20	char buf[256];
 21	pid_t pid;
 22
 23	if (!ASSERT_OK(pipe(pipe_c2p), "pipe_c2p"))
 
 24		return;
 25
 26	if (!ASSERT_OK(pipe(pipe_p2c), "pipe_p2c")) {
 
 27		close(pipe_c2p[0]);
 28		close(pipe_c2p[1]);
 29		return;
 30	}
 31
 32	pid = fork();
 33	if (!ASSERT_GE(pid, 0, "fork")) {
 34		close(pipe_c2p[0]);
 35		close(pipe_c2p[1]);
 36		close(pipe_p2c[0]);
 37		close(pipe_p2c[1]);
 38		return;
 39	}
 40
 41	if (pid == 0) {
 42		int old_prio;
 43		volatile int j = 0;
 44
 45		/* install signal handler and notify parent */
 46		ASSERT_NEQ(signal(SIGUSR1, sigusr1_handler), SIG_ERR, "signal");
 47
 48		close(pipe_c2p[0]); /* close read */
 49		close(pipe_p2c[1]); /* close write */
 50
 51		/* boost with a high priority so we got a higher chance
 52		 * that if an interrupt happens, the underlying task
 53		 * is this process.
 54		 */
 55		errno = 0;
 56		old_prio = getpriority(PRIO_PROCESS, 0);
 57		ASSERT_OK(errno, "getpriority");
 58		ASSERT_OK(setpriority(PRIO_PROCESS, 0, -20), "setpriority");
 59
 60		/* notify parent signal handler is installed */
 61		ASSERT_EQ(write(pipe_c2p[1], buf, 1), 1, "pipe_write");
 62
 63		/* make sure parent enabled bpf program to send_signal */
 64		ASSERT_EQ(read(pipe_p2c[0], buf, 1), 1, "pipe_read");
 65
 66		/* wait a little for signal handler */
 67		for (int i = 0; i < 1000000000 && !sigusr1_received; i++) {
 68			j /= i + j + 1;
 69			if (!attr)
 70				/* trigger the nanosleep tracepoint program. */
 71				usleep(1);
 72		}
 73
 74		buf[0] = sigusr1_received ? '2' : '0';
 75		ASSERT_EQ(sigusr1_received, 1, "sigusr1_received");
 76		ASSERT_EQ(write(pipe_c2p[1], buf, 1), 1, "pipe_write");
 77
 78		/* wait for parent notification and exit */
 79		ASSERT_EQ(read(pipe_p2c[0], buf, 1), 1, "pipe_read");
 80
 81		/* restore the old priority */
 82		ASSERT_OK(setpriority(PRIO_PROCESS, 0, old_prio), "setpriority");
 83
 84		close(pipe_c2p[1]);
 85		close(pipe_p2c[0]);
 86		exit(0);
 87	}
 88
 89	close(pipe_c2p[1]); /* close write */
 90	close(pipe_p2c[0]); /* close read */
 91
 92	skel = test_send_signal_kern__open_and_load();
 93	if (!ASSERT_OK_PTR(skel, "skel_open_and_load"))
 94		goto skel_open_load_failure;
 95
 96	if (!attr) {
 97		err = test_send_signal_kern__attach(skel);
 98		if (!ASSERT_OK(err, "skel_attach")) {
 99			err = -1;
100			goto destroy_skel;
101		}
102	} else {
103		pmu_fd = syscall(__NR_perf_event_open, attr, pid, -1 /* cpu */,
104				 -1 /* group id */, 0 /* flags */);
105		if (!ASSERT_GE(pmu_fd, 0, "perf_event_open")) {
 
106			err = -1;
107			goto destroy_skel;
108		}
109
110		skel->links.send_signal_perf =
111			bpf_program__attach_perf_event(skel->progs.send_signal_perf, pmu_fd);
112		if (!ASSERT_OK_PTR(skel->links.send_signal_perf, "attach_perf_event"))
 
113			goto disable_pmu;
114	}
115
116	/* wait until child signal handler installed */
117	ASSERT_EQ(read(pipe_c2p[0], buf, 1), 1, "pipe_read");
118
119	/* trigger the bpf send_signal */
120	skel->bss->signal_thread = signal_thread;
121	skel->bss->sig = SIGUSR1;
122	skel->bss->pid = pid;
 
 
123
124	/* notify child that bpf program can send_signal now */
125	ASSERT_EQ(write(pipe_p2c[1], buf, 1), 1, "pipe_write");
126
127	/* wait for result */
128	err = read(pipe_c2p[0], buf, 1);
129	if (!ASSERT_GE(err, 0, "reading pipe"))
130		goto disable_pmu;
131	if (!ASSERT_GT(err, 0, "reading pipe error: size 0")) {
132		err = -1;
133		goto disable_pmu;
134	}
135
136	ASSERT_EQ(buf[0], '2', "incorrect result");
137
138	/* notify child safe to exit */
139	ASSERT_EQ(write(pipe_p2c[1], buf, 1), 1, "pipe_write");
140
141disable_pmu:
142	close(pmu_fd);
143destroy_skel:
144	test_send_signal_kern__destroy(skel);
145skel_open_load_failure:
146	close(pipe_c2p[0]);
147	close(pipe_p2c[1]);
148	wait(NULL);
149}
150
151static void test_send_signal_tracepoint(bool signal_thread)
152{
153	test_send_signal_common(NULL, signal_thread);
154}
155
156static void test_send_signal_perf(bool signal_thread)
157{
158	struct perf_event_attr attr = {
159		.sample_period = 1,
160		.type = PERF_TYPE_SOFTWARE,
161		.config = PERF_COUNT_SW_CPU_CLOCK,
162	};
163
164	test_send_signal_common(&attr, signal_thread);
165}
166
167static void test_send_signal_nmi(bool signal_thread)
168{
169	struct perf_event_attr attr = {
170		.sample_period = 1,
171		.type = PERF_TYPE_HARDWARE,
172		.config = PERF_COUNT_HW_CPU_CYCLES,
173	};
174	int pmu_fd;
175
176	/* Some setups (e.g. virtual machines) might run with hardware
177	 * perf events disabled. If this is the case, skip this test.
178	 */
179	pmu_fd = syscall(__NR_perf_event_open, &attr, 0 /* pid */,
180			 -1 /* cpu */, -1 /* group_fd */, 0 /* flags */);
181	if (pmu_fd == -1) {
182		if (errno == ENOENT) {
183			printf("%s:SKIP:no PERF_COUNT_HW_CPU_CYCLES\n",
184			       __func__);
185			test__skip();
186			return;
187		}
188		/* Let the test fail with a more informative message */
189	} else {
190		close(pmu_fd);
191	}
192
193	test_send_signal_common(&attr, signal_thread);
194}
195
196void test_send_signal(void)
197{
198	if (test__start_subtest("send_signal_tracepoint"))
199		test_send_signal_tracepoint(false);
200	if (test__start_subtest("send_signal_perf"))
201		test_send_signal_perf(false);
202	if (test__start_subtest("send_signal_nmi"))
203		test_send_signal_nmi(false);
204	if (test__start_subtest("send_signal_tracepoint_thread"))
205		test_send_signal_tracepoint(true);
206	if (test__start_subtest("send_signal_perf_thread"))
207		test_send_signal_perf(true);
208	if (test__start_subtest("send_signal_nmi_thread"))
209		test_send_signal_nmi(true);
210}
v5.9
  1// SPDX-License-Identifier: GPL-2.0
  2#include <test_progs.h>
 
 
  3#include "test_send_signal_kern.skel.h"
  4
  5static volatile int sigusr1_received = 0;
  6
  7static void sigusr1_handler(int signum)
  8{
  9	sigusr1_received++;
 10}
 11
 12static void test_send_signal_common(struct perf_event_attr *attr,
 13				    bool signal_thread,
 14				    const char *test_name)
 15{
 16	struct test_send_signal_kern *skel;
 17	int pipe_c2p[2], pipe_p2c[2];
 18	int err = -1, pmu_fd = -1;
 19	__u32 duration = 0;
 20	char buf[256];
 21	pid_t pid;
 22
 23	if (CHECK(pipe(pipe_c2p), test_name,
 24		  "pipe pipe_c2p error: %s\n", strerror(errno)))
 25		return;
 26
 27	if (CHECK(pipe(pipe_p2c), test_name,
 28		  "pipe pipe_p2c error: %s\n", strerror(errno))) {
 29		close(pipe_c2p[0]);
 30		close(pipe_c2p[1]);
 31		return;
 32	}
 33
 34	pid = fork();
 35	if (CHECK(pid < 0, test_name, "fork error: %s\n", strerror(errno))) {
 36		close(pipe_c2p[0]);
 37		close(pipe_c2p[1]);
 38		close(pipe_p2c[0]);
 39		close(pipe_p2c[1]);
 40		return;
 41	}
 42
 43	if (pid == 0) {
 
 
 
 44		/* install signal handler and notify parent */
 45		signal(SIGUSR1, sigusr1_handler);
 46
 47		close(pipe_c2p[0]); /* close read */
 48		close(pipe_p2c[1]); /* close write */
 49
 
 
 
 
 
 
 
 
 
 50		/* notify parent signal handler is installed */
 51		CHECK(write(pipe_c2p[1], buf, 1) != 1, "pipe_write", "err %d\n", -errno);
 52
 53		/* make sure parent enabled bpf program to send_signal */
 54		CHECK(read(pipe_p2c[0], buf, 1) != 1, "pipe_read", "err %d\n", -errno);
 55
 56		/* wait a little for signal handler */
 57		sleep(1);
 
 
 
 
 
 58
 59		buf[0] = sigusr1_received ? '2' : '0';
 60		CHECK(write(pipe_c2p[1], buf, 1) != 1, "pipe_write", "err %d\n", -errno);
 
 61
 62		/* wait for parent notification and exit */
 63		CHECK(read(pipe_p2c[0], buf, 1) != 1, "pipe_read", "err %d\n", -errno);
 
 
 
 64
 65		close(pipe_c2p[1]);
 66		close(pipe_p2c[0]);
 67		exit(0);
 68	}
 69
 70	close(pipe_c2p[1]); /* close write */
 71	close(pipe_p2c[0]); /* close read */
 72
 73	skel = test_send_signal_kern__open_and_load();
 74	if (CHECK(!skel, "skel_open_and_load", "skeleton open_and_load failed\n"))
 75		goto skel_open_load_failure;
 76
 77	if (!attr) {
 78		err = test_send_signal_kern__attach(skel);
 79		if (CHECK(err, "skel_attach", "skeleton attach failed\n")) {
 80			err = -1;
 81			goto destroy_skel;
 82		}
 83	} else {
 84		pmu_fd = syscall(__NR_perf_event_open, attr, pid, -1,
 85				 -1 /* group id */, 0 /* flags */);
 86		if (CHECK(pmu_fd < 0, test_name, "perf_event_open error: %s\n",
 87			strerror(errno))) {
 88			err = -1;
 89			goto destroy_skel;
 90		}
 91
 92		skel->links.send_signal_perf =
 93			bpf_program__attach_perf_event(skel->progs.send_signal_perf, pmu_fd);
 94		if (CHECK(IS_ERR(skel->links.send_signal_perf), "attach_perf_event",
 95			  "err %ld\n", PTR_ERR(skel->links.send_signal_perf)))
 96			goto disable_pmu;
 97	}
 98
 99	/* wait until child signal handler installed */
100	CHECK(read(pipe_c2p[0], buf, 1) != 1, "pipe_read", "err %d\n", -errno);
101
102	/* trigger the bpf send_signal */
 
 
103	skel->bss->pid = pid;
104	skel->bss->sig = SIGUSR1;
105	skel->bss->signal_thread = signal_thread;
106
107	/* notify child that bpf program can send_signal now */
108	CHECK(write(pipe_p2c[1], buf, 1) != 1, "pipe_write", "err %d\n", -errno);
109
110	/* wait for result */
111	err = read(pipe_c2p[0], buf, 1);
112	if (CHECK(err < 0, test_name, "reading pipe error: %s\n", strerror(errno)))
113		goto disable_pmu;
114	if (CHECK(err == 0, test_name, "reading pipe error: size 0\n")) {
115		err = -1;
116		goto disable_pmu;
117	}
118
119	CHECK(buf[0] != '2', test_name, "incorrect result\n");
120
121	/* notify child safe to exit */
122	CHECK(write(pipe_p2c[1], buf, 1) != 1, "pipe_write", "err %d\n", -errno);
123
124disable_pmu:
125	close(pmu_fd);
126destroy_skel:
127	test_send_signal_kern__destroy(skel);
128skel_open_load_failure:
129	close(pipe_c2p[0]);
130	close(pipe_p2c[1]);
131	wait(NULL);
132}
133
134static void test_send_signal_tracepoint(bool signal_thread)
135{
136	test_send_signal_common(NULL, signal_thread, "tracepoint");
137}
138
139static void test_send_signal_perf(bool signal_thread)
140{
141	struct perf_event_attr attr = {
142		.sample_period = 1,
143		.type = PERF_TYPE_SOFTWARE,
144		.config = PERF_COUNT_SW_CPU_CLOCK,
145	};
146
147	test_send_signal_common(&attr, signal_thread, "perf_sw_event");
148}
149
150static void test_send_signal_nmi(bool signal_thread)
151{
152	struct perf_event_attr attr = {
153		.sample_period = 1,
154		.type = PERF_TYPE_HARDWARE,
155		.config = PERF_COUNT_HW_CPU_CYCLES,
156	};
157	int pmu_fd;
158
159	/* Some setups (e.g. virtual machines) might run with hardware
160	 * perf events disabled. If this is the case, skip this test.
161	 */
162	pmu_fd = syscall(__NR_perf_event_open, &attr, 0 /* pid */,
163			 -1 /* cpu */, -1 /* group_fd */, 0 /* flags */);
164	if (pmu_fd == -1) {
165		if (errno == ENOENT) {
166			printf("%s:SKIP:no PERF_COUNT_HW_CPU_CYCLES\n",
167			       __func__);
168			test__skip();
169			return;
170		}
171		/* Let the test fail with a more informative message */
172	} else {
173		close(pmu_fd);
174	}
175
176	test_send_signal_common(&attr, signal_thread, "perf_hw_event");
177}
178
179void test_send_signal(void)
180{
181	if (test__start_subtest("send_signal_tracepoint"))
182		test_send_signal_tracepoint(false);
183	if (test__start_subtest("send_signal_perf"))
184		test_send_signal_perf(false);
185	if (test__start_subtest("send_signal_nmi"))
186		test_send_signal_nmi(false);
187	if (test__start_subtest("send_signal_tracepoint_thread"))
188		test_send_signal_tracepoint(true);
189	if (test__start_subtest("send_signal_perf_thread"))
190		test_send_signal_perf(true);
191	if (test__start_subtest("send_signal_nmi_thread"))
192		test_send_signal_nmi(true);
193}