117 lines
2.4 KiB
C
117 lines
2.4 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
|
||
|
#define _GNU_SOURCE
|
||
|
#include <errno.h>
|
||
|
#include <linux/types.h>
|
||
|
#include <poll.h>
|
||
|
#include <signal.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <syscall.h>
|
||
|
#include <sys/wait.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
#include "pidfd.h"
|
||
|
#include "../kselftest.h"
|
||
|
|
||
|
static bool timeout;
|
||
|
|
||
|
static void handle_alarm(int sig)
|
||
|
{
|
||
|
timeout = true;
|
||
|
}
|
||
|
|
||
|
int main(int argc, char **argv)
|
||
|
{
|
||
|
struct pollfd fds;
|
||
|
int iter, nevents;
|
||
|
int nr_iterations = 10000;
|
||
|
|
||
|
fds.events = POLLIN;
|
||
|
|
||
|
if (argc > 2)
|
||
|
ksft_exit_fail_msg("Unexpected command line argument\n");
|
||
|
|
||
|
if (argc == 2) {
|
||
|
nr_iterations = atoi(argv[1]);
|
||
|
if (nr_iterations <= 0)
|
||
|
ksft_exit_fail_msg("invalid input parameter %s\n",
|
||
|
argv[1]);
|
||
|
}
|
||
|
|
||
|
ksft_print_msg("running pidfd poll test for %d iterations\n",
|
||
|
nr_iterations);
|
||
|
|
||
|
for (iter = 0; iter < nr_iterations; iter++) {
|
||
|
int pidfd;
|
||
|
int child_pid = fork();
|
||
|
|
||
|
if (child_pid < 0) {
|
||
|
if (errno == EAGAIN) {
|
||
|
iter--;
|
||
|
continue;
|
||
|
}
|
||
|
ksft_exit_fail_msg(
|
||
|
"%s - failed to fork a child process\n",
|
||
|
strerror(errno));
|
||
|
}
|
||
|
|
||
|
if (child_pid == 0) {
|
||
|
/* Child process just sleeps for a min and exits */
|
||
|
sleep(60);
|
||
|
exit(EXIT_SUCCESS);
|
||
|
}
|
||
|
|
||
|
/* Parent kills the child and waits for its death */
|
||
|
pidfd = sys_pidfd_open(child_pid, 0);
|
||
|
if (pidfd < 0)
|
||
|
ksft_exit_fail_msg("%s - pidfd_open failed\n",
|
||
|
strerror(errno));
|
||
|
|
||
|
/* Setup 3 sec alarm - plenty of time */
|
||
|
if (signal(SIGALRM, handle_alarm) == SIG_ERR)
|
||
|
ksft_exit_fail_msg("%s - signal failed\n",
|
||
|
strerror(errno));
|
||
|
alarm(3);
|
||
|
|
||
|
/* Send SIGKILL to the child */
|
||
|
if (sys_pidfd_send_signal(pidfd, SIGKILL, NULL, 0))
|
||
|
ksft_exit_fail_msg("%s - pidfd_send_signal failed\n",
|
||
|
strerror(errno));
|
||
|
|
||
|
/* Wait for the death notification */
|
||
|
fds.fd = pidfd;
|
||
|
nevents = poll(&fds, 1, -1);
|
||
|
|
||
|
/* Check for error conditions */
|
||
|
if (nevents < 0)
|
||
|
ksft_exit_fail_msg("%s - poll failed\n",
|
||
|
strerror(errno));
|
||
|
|
||
|
if (nevents != 1)
|
||
|
ksft_exit_fail_msg("unexpected poll result: %d\n",
|
||
|
nevents);
|
||
|
|
||
|
if (!(fds.revents & POLLIN))
|
||
|
ksft_exit_fail_msg(
|
||
|
"unexpected event type received: 0x%x\n",
|
||
|
fds.revents);
|
||
|
|
||
|
if (timeout)
|
||
|
ksft_exit_fail_msg(
|
||
|
"death notification wait timeout\n");
|
||
|
|
||
|
close(pidfd);
|
||
|
/* Wait for child to prevent zombies */
|
||
|
if (waitpid(child_pid, NULL, 0) < 0)
|
||
|
ksft_exit_fail_msg("%s - waitpid failed\n",
|
||
|
strerror(errno));
|
||
|
|
||
|
}
|
||
|
|
||
|
ksft_test_result_pass("pidfd poll test: pass\n");
|
||
|
return ksft_exit_pass();
|
||
|
}
|