1*6216798bSMarco Elver // SPDX-License-Identifier: GPL-2.0
2*6216798bSMarco Elver /*
3*6216798bSMarco Elver * Test for remove_on_exec.
4*6216798bSMarco Elver *
5*6216798bSMarco Elver * Copyright (C) 2021, Google LLC.
6*6216798bSMarco Elver */
7*6216798bSMarco Elver
8*6216798bSMarco Elver #define _GNU_SOURCE
9*6216798bSMarco Elver
10*6216798bSMarco Elver /* We need the latest siginfo from the kernel repo. */
11*6216798bSMarco Elver #include <sys/types.h>
12*6216798bSMarco Elver #include <asm/siginfo.h>
13*6216798bSMarco Elver #define __have_siginfo_t 1
14*6216798bSMarco Elver #define __have_sigval_t 1
15*6216798bSMarco Elver #define __have_sigevent_t 1
16*6216798bSMarco Elver #define __siginfo_t_defined
17*6216798bSMarco Elver #define __sigval_t_defined
18*6216798bSMarco Elver #define __sigevent_t_defined
19*6216798bSMarco Elver #define _BITS_SIGINFO_CONSTS_H 1
20*6216798bSMarco Elver #define _BITS_SIGEVENT_CONSTS_H 1
21*6216798bSMarco Elver
22*6216798bSMarco Elver #include <stdbool.h>
23*6216798bSMarco Elver #include <stddef.h>
24*6216798bSMarco Elver #include <stdint.h>
25*6216798bSMarco Elver #include <stdio.h>
26*6216798bSMarco Elver #include <linux/perf_event.h>
27*6216798bSMarco Elver #include <pthread.h>
28*6216798bSMarco Elver #include <signal.h>
29*6216798bSMarco Elver #include <sys/ioctl.h>
30*6216798bSMarco Elver #include <sys/syscall.h>
31*6216798bSMarco Elver #include <unistd.h>
32*6216798bSMarco Elver
33*6216798bSMarco Elver #include "../kselftest_harness.h"
34*6216798bSMarco Elver
35*6216798bSMarco Elver static volatile int signal_count;
36*6216798bSMarco Elver
make_event_attr(void)37*6216798bSMarco Elver static struct perf_event_attr make_event_attr(void)
38*6216798bSMarco Elver {
39*6216798bSMarco Elver struct perf_event_attr attr = {
40*6216798bSMarco Elver .type = PERF_TYPE_HARDWARE,
41*6216798bSMarco Elver .size = sizeof(attr),
42*6216798bSMarco Elver .config = PERF_COUNT_HW_INSTRUCTIONS,
43*6216798bSMarco Elver .sample_period = 1000,
44*6216798bSMarco Elver .exclude_kernel = 1,
45*6216798bSMarco Elver .exclude_hv = 1,
46*6216798bSMarco Elver .disabled = 1,
47*6216798bSMarco Elver .inherit = 1,
48*6216798bSMarco Elver /*
49*6216798bSMarco Elver * Children normally retain their inherited event on exec; with
50*6216798bSMarco Elver * remove_on_exec, we'll remove their event, but the parent and
51*6216798bSMarco Elver * any other non-exec'd children will keep their events.
52*6216798bSMarco Elver */
53*6216798bSMarco Elver .remove_on_exec = 1,
54*6216798bSMarco Elver .sigtrap = 1,
55*6216798bSMarco Elver };
56*6216798bSMarco Elver return attr;
57*6216798bSMarco Elver }
58*6216798bSMarco Elver
sigtrap_handler(int signum,siginfo_t * info,void * ucontext)59*6216798bSMarco Elver static void sigtrap_handler(int signum, siginfo_t *info, void *ucontext)
60*6216798bSMarco Elver {
61*6216798bSMarco Elver if (info->si_code != TRAP_PERF) {
62*6216798bSMarco Elver fprintf(stderr, "%s: unexpected si_code %d\n", __func__, info->si_code);
63*6216798bSMarco Elver return;
64*6216798bSMarco Elver }
65*6216798bSMarco Elver
66*6216798bSMarco Elver signal_count++;
67*6216798bSMarco Elver }
68*6216798bSMarco Elver
FIXTURE(remove_on_exec)69*6216798bSMarco Elver FIXTURE(remove_on_exec)
70*6216798bSMarco Elver {
71*6216798bSMarco Elver struct sigaction oldact;
72*6216798bSMarco Elver int fd;
73*6216798bSMarco Elver };
74*6216798bSMarco Elver
FIXTURE_SETUP(remove_on_exec)75*6216798bSMarco Elver FIXTURE_SETUP(remove_on_exec)
76*6216798bSMarco Elver {
77*6216798bSMarco Elver struct perf_event_attr attr = make_event_attr();
78*6216798bSMarco Elver struct sigaction action = {};
79*6216798bSMarco Elver
80*6216798bSMarco Elver signal_count = 0;
81*6216798bSMarco Elver
82*6216798bSMarco Elver /* Initialize sigtrap handler. */
83*6216798bSMarco Elver action.sa_flags = SA_SIGINFO | SA_NODEFER;
84*6216798bSMarco Elver action.sa_sigaction = sigtrap_handler;
85*6216798bSMarco Elver sigemptyset(&action.sa_mask);
86*6216798bSMarco Elver ASSERT_EQ(sigaction(SIGTRAP, &action, &self->oldact), 0);
87*6216798bSMarco Elver
88*6216798bSMarco Elver /* Initialize perf event. */
89*6216798bSMarco Elver self->fd = syscall(__NR_perf_event_open, &attr, 0, -1, -1, PERF_FLAG_FD_CLOEXEC);
90*6216798bSMarco Elver ASSERT_NE(self->fd, -1);
91*6216798bSMarco Elver }
92*6216798bSMarco Elver
FIXTURE_TEARDOWN(remove_on_exec)93*6216798bSMarco Elver FIXTURE_TEARDOWN(remove_on_exec)
94*6216798bSMarco Elver {
95*6216798bSMarco Elver close(self->fd);
96*6216798bSMarco Elver sigaction(SIGTRAP, &self->oldact, NULL);
97*6216798bSMarco Elver }
98*6216798bSMarco Elver
99*6216798bSMarco Elver /* Verify event propagates to fork'd child. */
TEST_F(remove_on_exec,fork_only)100*6216798bSMarco Elver TEST_F(remove_on_exec, fork_only)
101*6216798bSMarco Elver {
102*6216798bSMarco Elver int status;
103*6216798bSMarco Elver pid_t pid = fork();
104*6216798bSMarco Elver
105*6216798bSMarco Elver if (pid == 0) {
106*6216798bSMarco Elver ASSERT_EQ(signal_count, 0);
107*6216798bSMarco Elver ASSERT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0);
108*6216798bSMarco Elver while (!signal_count);
109*6216798bSMarco Elver _exit(42);
110*6216798bSMarco Elver }
111*6216798bSMarco Elver
112*6216798bSMarco Elver while (!signal_count); /* Child enables event. */
113*6216798bSMarco Elver EXPECT_EQ(waitpid(pid, &status, 0), pid);
114*6216798bSMarco Elver EXPECT_EQ(WEXITSTATUS(status), 42);
115*6216798bSMarco Elver }
116*6216798bSMarco Elver
117*6216798bSMarco Elver /*
118*6216798bSMarco Elver * Verify that event does _not_ propagate to fork+exec'd child; event enabled
119*6216798bSMarco Elver * after fork+exec.
120*6216798bSMarco Elver */
TEST_F(remove_on_exec,fork_exec_then_enable)121*6216798bSMarco Elver TEST_F(remove_on_exec, fork_exec_then_enable)
122*6216798bSMarco Elver {
123*6216798bSMarco Elver pid_t pid_exec, pid_only_fork;
124*6216798bSMarco Elver int pipefd[2];
125*6216798bSMarco Elver int tmp;
126*6216798bSMarco Elver
127*6216798bSMarco Elver /*
128*6216798bSMarco Elver * Non-exec child, to ensure exec does not affect inherited events of
129*6216798bSMarco Elver * other children.
130*6216798bSMarco Elver */
131*6216798bSMarco Elver pid_only_fork = fork();
132*6216798bSMarco Elver if (pid_only_fork == 0) {
133*6216798bSMarco Elver /* Block until parent enables event. */
134*6216798bSMarco Elver while (!signal_count);
135*6216798bSMarco Elver _exit(42);
136*6216798bSMarco Elver }
137*6216798bSMarco Elver
138*6216798bSMarco Elver ASSERT_NE(pipe(pipefd), -1);
139*6216798bSMarco Elver pid_exec = fork();
140*6216798bSMarco Elver if (pid_exec == 0) {
141*6216798bSMarco Elver ASSERT_NE(dup2(pipefd[1], STDOUT_FILENO), -1);
142*6216798bSMarco Elver close(pipefd[0]);
143*6216798bSMarco Elver execl("/proc/self/exe", "exec_child", NULL);
144*6216798bSMarco Elver _exit((perror("exec failed"), 1));
145*6216798bSMarco Elver }
146*6216798bSMarco Elver close(pipefd[1]);
147*6216798bSMarco Elver
148*6216798bSMarco Elver ASSERT_EQ(waitpid(pid_exec, &tmp, WNOHANG), 0); /* Child is running. */
149*6216798bSMarco Elver /* Wait for exec'd child to start spinning. */
150*6216798bSMarco Elver EXPECT_EQ(read(pipefd[0], &tmp, sizeof(int)), sizeof(int));
151*6216798bSMarco Elver EXPECT_EQ(tmp, 42);
152*6216798bSMarco Elver close(pipefd[0]);
153*6216798bSMarco Elver /* Now we can enable the event, knowing the child is doing work. */
154*6216798bSMarco Elver EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0);
155*6216798bSMarco Elver /* If the event propagated to the exec'd child, it will exit normally... */
156*6216798bSMarco Elver usleep(100000); /* ... give time for event to trigger (in case of bug). */
157*6216798bSMarco Elver EXPECT_EQ(waitpid(pid_exec, &tmp, WNOHANG), 0); /* Should still be running. */
158*6216798bSMarco Elver EXPECT_EQ(kill(pid_exec, SIGKILL), 0);
159*6216798bSMarco Elver
160*6216798bSMarco Elver /* Verify removal from child did not affect this task's event. */
161*6216798bSMarco Elver tmp = signal_count;
162*6216798bSMarco Elver while (signal_count == tmp); /* Should not hang! */
163*6216798bSMarco Elver /* Nor should it have affected the first child. */
164*6216798bSMarco Elver EXPECT_EQ(waitpid(pid_only_fork, &tmp, 0), pid_only_fork);
165*6216798bSMarco Elver EXPECT_EQ(WEXITSTATUS(tmp), 42);
166*6216798bSMarco Elver }
167*6216798bSMarco Elver
168*6216798bSMarco Elver /*
169*6216798bSMarco Elver * Verify that event does _not_ propagate to fork+exec'd child; event enabled
170*6216798bSMarco Elver * before fork+exec.
171*6216798bSMarco Elver */
TEST_F(remove_on_exec,enable_then_fork_exec)172*6216798bSMarco Elver TEST_F(remove_on_exec, enable_then_fork_exec)
173*6216798bSMarco Elver {
174*6216798bSMarco Elver pid_t pid_exec;
175*6216798bSMarco Elver int tmp;
176*6216798bSMarco Elver
177*6216798bSMarco Elver EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0);
178*6216798bSMarco Elver
179*6216798bSMarco Elver pid_exec = fork();
180*6216798bSMarco Elver if (pid_exec == 0) {
181*6216798bSMarco Elver execl("/proc/self/exe", "exec_child", NULL);
182*6216798bSMarco Elver _exit((perror("exec failed"), 1));
183*6216798bSMarco Elver }
184*6216798bSMarco Elver
185*6216798bSMarco Elver /*
186*6216798bSMarco Elver * The child may exit abnormally at any time if the event propagated and
187*6216798bSMarco Elver * a SIGTRAP is sent before the handler was set up.
188*6216798bSMarco Elver */
189*6216798bSMarco Elver usleep(100000); /* ... give time for event to trigger (in case of bug). */
190*6216798bSMarco Elver EXPECT_EQ(waitpid(pid_exec, &tmp, WNOHANG), 0); /* Should still be running. */
191*6216798bSMarco Elver EXPECT_EQ(kill(pid_exec, SIGKILL), 0);
192*6216798bSMarco Elver
193*6216798bSMarco Elver /* Verify removal from child did not affect this task's event. */
194*6216798bSMarco Elver tmp = signal_count;
195*6216798bSMarco Elver while (signal_count == tmp); /* Should not hang! */
196*6216798bSMarco Elver }
197*6216798bSMarco Elver
TEST_F(remove_on_exec,exec_stress)198*6216798bSMarco Elver TEST_F(remove_on_exec, exec_stress)
199*6216798bSMarco Elver {
200*6216798bSMarco Elver pid_t pids[30];
201*6216798bSMarco Elver int i, tmp;
202*6216798bSMarco Elver
203*6216798bSMarco Elver for (i = 0; i < sizeof(pids) / sizeof(pids[0]); i++) {
204*6216798bSMarco Elver pids[i] = fork();
205*6216798bSMarco Elver if (pids[i] == 0) {
206*6216798bSMarco Elver execl("/proc/self/exe", "exec_child", NULL);
207*6216798bSMarco Elver _exit((perror("exec failed"), 1));
208*6216798bSMarco Elver }
209*6216798bSMarco Elver
210*6216798bSMarco Elver /* Some forked with event disabled, rest with enabled. */
211*6216798bSMarco Elver if (i > 10)
212*6216798bSMarco Elver EXPECT_EQ(ioctl(self->fd, PERF_EVENT_IOC_ENABLE, 0), 0);
213*6216798bSMarco Elver }
214*6216798bSMarco Elver
215*6216798bSMarco Elver usleep(100000); /* ... give time for event to trigger (in case of bug). */
216*6216798bSMarco Elver
217*6216798bSMarco Elver for (i = 0; i < sizeof(pids) / sizeof(pids[0]); i++) {
218*6216798bSMarco Elver /* All children should still be running. */
219*6216798bSMarco Elver EXPECT_EQ(waitpid(pids[i], &tmp, WNOHANG), 0);
220*6216798bSMarco Elver EXPECT_EQ(kill(pids[i], SIGKILL), 0);
221*6216798bSMarco Elver }
222*6216798bSMarco Elver
223*6216798bSMarco Elver /* Verify event is still alive. */
224*6216798bSMarco Elver tmp = signal_count;
225*6216798bSMarco Elver while (signal_count == tmp);
226*6216798bSMarco Elver }
227*6216798bSMarco Elver
228*6216798bSMarco Elver /* For exec'd child. */
exec_child(void)229*6216798bSMarco Elver static void exec_child(void)
230*6216798bSMarco Elver {
231*6216798bSMarco Elver struct sigaction action = {};
232*6216798bSMarco Elver const int val = 42;
233*6216798bSMarco Elver
234*6216798bSMarco Elver /* Set up sigtrap handler in case we erroneously receive a trap. */
235*6216798bSMarco Elver action.sa_flags = SA_SIGINFO | SA_NODEFER;
236*6216798bSMarco Elver action.sa_sigaction = sigtrap_handler;
237*6216798bSMarco Elver sigemptyset(&action.sa_mask);
238*6216798bSMarco Elver if (sigaction(SIGTRAP, &action, NULL))
239*6216798bSMarco Elver _exit((perror("sigaction failed"), 1));
240*6216798bSMarco Elver
241*6216798bSMarco Elver /* Signal parent that we're starting to spin. */
242*6216798bSMarco Elver if (write(STDOUT_FILENO, &val, sizeof(int)) == -1)
243*6216798bSMarco Elver _exit((perror("write failed"), 1));
244*6216798bSMarco Elver
245*6216798bSMarco Elver /* Should hang here until killed. */
246*6216798bSMarco Elver while (!signal_count);
247*6216798bSMarco Elver }
248*6216798bSMarco Elver
249*6216798bSMarco Elver #define main test_main
250*6216798bSMarco Elver TEST_HARNESS_MAIN
251*6216798bSMarco Elver #undef main
main(int argc,char * argv[])252*6216798bSMarco Elver int main(int argc, char *argv[])
253*6216798bSMarco Elver {
254*6216798bSMarco Elver if (!strcmp(argv[0], "exec_child")) {
255*6216798bSMarco Elver exec_child();
256*6216798bSMarco Elver return 1;
257*6216798bSMarco Elver }
258*6216798bSMarco Elver
259*6216798bSMarco Elver return test_main(argc, argv);
260*6216798bSMarco Elver }
261