Training courses

Kernel and Embedded Linux

Bootlin training courses

Embedded Linux, kernel,
Yocto Project, Buildroot, real-time,
graphics, boot time, debugging...

Bootlin logo

Elixir Cross Referencer

/* SPDX-License-Identifier: GPL-2.0 */

#define _GNU_SOURCE
#include <errno.h>
#include <linux/sched.h>
#include <linux/types.h>
#include <signal.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <sched.h>
#include <string.h>
#include <sys/resource.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>

#include "pidfd.h"
#include "../kselftest.h"

#define ptr_to_u64(ptr) ((__u64)((uintptr_t)(ptr)))

static pid_t sys_clone3(struct clone_args *args)
{
	return syscall(__NR_clone3, args, sizeof(struct clone_args));
}

static int sys_waitid(int which, pid_t pid, siginfo_t *info, int options,
		      struct rusage *ru)
{
	return syscall(__NR_waitid, which, pid, info, options, ru);
}

static int test_pidfd_wait_simple(void)
{
	const char *test_name = "pidfd wait simple";
	int pidfd = -1, status = 0;
	pid_t parent_tid = -1;
	struct clone_args args = {
		.parent_tid = ptr_to_u64(&parent_tid),
		.pidfd = ptr_to_u64(&pidfd),
		.flags = CLONE_PIDFD | CLONE_PARENT_SETTID,
		.exit_signal = SIGCHLD,
	};
	int ret;
	pid_t pid;
	siginfo_t info = {
		.si_signo = 0,
	};

	pidfd = open("/proc/self", O_DIRECTORY | O_RDONLY | O_CLOEXEC);
	if (pidfd < 0)
		ksft_exit_fail_msg("%s test: failed to open /proc/self %s\n",
				   test_name, strerror(errno));

	pid = sys_waitid(P_PIDFD, pidfd, &info, WEXITED, NULL);
	if (pid == 0)
		ksft_exit_fail_msg(
			"%s test: succeeded to wait on invalid pidfd %s\n",
			test_name, strerror(errno));
	close(pidfd);
	pidfd = -1;

	pidfd = open("/dev/null", O_RDONLY | O_CLOEXEC);
	if (pidfd == 0)
		ksft_exit_fail_msg("%s test: failed to open /dev/null %s\n",
				   test_name, strerror(errno));

	pid = sys_waitid(P_PIDFD, pidfd, &info, WEXITED, NULL);
	if (pid == 0)
		ksft_exit_fail_msg(
			"%s test: succeeded to wait on invalid pidfd %s\n",
			test_name, strerror(errno));
	close(pidfd);
	pidfd = -1;

	pid = sys_clone3(&args);
	if (pid < 0)
		ksft_exit_fail_msg("%s test: failed to create new process %s\n",
				   test_name, strerror(errno));

	if (pid == 0)
		exit(EXIT_SUCCESS);

	pid = sys_waitid(P_PIDFD, pidfd, &info, WEXITED, NULL);
	if (pid < 0)
		ksft_exit_fail_msg(
			"%s test: failed to wait on process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	if (!WIFEXITED(info.si_status) || WEXITSTATUS(info.si_status))
		ksft_exit_fail_msg(
			"%s test: unexpected status received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));
	close(pidfd);

	if (info.si_signo != SIGCHLD)
		ksft_exit_fail_msg(
			"%s test: unexpected si_signo value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_signo, parent_tid, pidfd,
			strerror(errno));

	if (info.si_code != CLD_EXITED)
		ksft_exit_fail_msg(
			"%s test: unexpected si_code value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_code, parent_tid, pidfd,
			strerror(errno));

	if (info.si_pid != parent_tid)
		ksft_exit_fail_msg(
			"%s test: unexpected si_pid value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_pid, parent_tid, pidfd,
			strerror(errno));

	ksft_test_result_pass("%s test: Passed\n", test_name);
	return 0;
}

static int test_pidfd_wait_states(void)
{
	const char *test_name = "pidfd wait states";
	int pidfd = -1, status = 0;
	pid_t parent_tid = -1;
	struct clone_args args = {
		.parent_tid = ptr_to_u64(&parent_tid),
		.pidfd = ptr_to_u64(&pidfd),
		.flags = CLONE_PIDFD | CLONE_PARENT_SETTID,
		.exit_signal = SIGCHLD,
	};
	int ret;
	pid_t pid;
	siginfo_t info = {
		.si_signo = 0,
	};

	pid = sys_clone3(&args);
	if (pid < 0)
		ksft_exit_fail_msg("%s test: failed to create new process %s\n",
				   test_name, strerror(errno));

	if (pid == 0) {
		kill(getpid(), SIGSTOP);
		kill(getpid(), SIGSTOP);
		exit(EXIT_SUCCESS);
	}

	ret = sys_waitid(P_PIDFD, pidfd, &info, WSTOPPED, NULL);
	if (ret < 0)
		ksft_exit_fail_msg(
			"%s test: failed to wait on WSTOPPED process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	if (info.si_signo != SIGCHLD)
		ksft_exit_fail_msg(
			"%s test: unexpected si_signo value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_signo, parent_tid, pidfd,
			strerror(errno));

	if (info.si_code != CLD_STOPPED)
		ksft_exit_fail_msg(
			"%s test: unexpected si_code value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_code, parent_tid, pidfd,
			strerror(errno));

	if (info.si_pid != parent_tid)
		ksft_exit_fail_msg(
			"%s test: unexpected si_pid value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_pid, parent_tid, pidfd,
			strerror(errno));

	ret = sys_pidfd_send_signal(pidfd, SIGCONT, NULL, 0);
	if (ret < 0)
		ksft_exit_fail_msg(
			"%s test: failed to send signal to process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	ret = sys_waitid(P_PIDFD, pidfd, &info, WCONTINUED, NULL);
	if (ret < 0)
		ksft_exit_fail_msg(
			"%s test: failed to wait WCONTINUED on process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	if (info.si_signo != SIGCHLD)
		ksft_exit_fail_msg(
			"%s test: unexpected si_signo value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_signo, parent_tid, pidfd,
			strerror(errno));

	if (info.si_code != CLD_CONTINUED)
		ksft_exit_fail_msg(
			"%s test: unexpected si_code value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_code, parent_tid, pidfd,
			strerror(errno));

	if (info.si_pid != parent_tid)
		ksft_exit_fail_msg(
			"%s test: unexpected si_pid value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_pid, parent_tid, pidfd,
			strerror(errno));

	ret = sys_waitid(P_PIDFD, pidfd, &info, WUNTRACED, NULL);
	if (ret < 0)
		ksft_exit_fail_msg(
			"%s test: failed to wait on WUNTRACED process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	if (info.si_signo != SIGCHLD)
		ksft_exit_fail_msg(
			"%s test: unexpected si_signo value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_signo, parent_tid, pidfd,
			strerror(errno));

	if (info.si_code != CLD_STOPPED)
		ksft_exit_fail_msg(
			"%s test: unexpected si_code value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_code, parent_tid, pidfd,
			strerror(errno));

	if (info.si_pid != parent_tid)
		ksft_exit_fail_msg(
			"%s test: unexpected si_pid value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_pid, parent_tid, pidfd,
			strerror(errno));

	ret = sys_pidfd_send_signal(pidfd, SIGKILL, NULL, 0);
	if (ret < 0)
		ksft_exit_fail_msg(
			"%s test: failed to send SIGKILL to process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	ret = sys_waitid(P_PIDFD, pidfd, &info, WEXITED, NULL);
	if (ret < 0)
		ksft_exit_fail_msg(
			"%s test: failed to wait on WEXITED process with pid %d and pidfd %d: %s\n",
			test_name, parent_tid, pidfd, strerror(errno));

	if (info.si_signo != SIGCHLD)
		ksft_exit_fail_msg(
			"%s test: unexpected si_signo value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_signo, parent_tid, pidfd,
			strerror(errno));

	if (info.si_code != CLD_KILLED)
		ksft_exit_fail_msg(
			"%s test: unexpected si_code value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_code, parent_tid, pidfd,
			strerror(errno));

	if (info.si_pid != parent_tid)
		ksft_exit_fail_msg(
			"%s test: unexpected si_pid value %d received after waiting on process with pid %d and pidfd %d: %s\n",
			test_name, info.si_pid, parent_tid, pidfd,
			strerror(errno));

	close(pidfd);

	ksft_test_result_pass("%s test: Passed\n", test_name);
	return 0;
}

int main(int argc, char **argv)
{
	ksft_print_header();
	ksft_set_plan(2);

	test_pidfd_wait_simple();
	test_pidfd_wait_states();

	return ksft_exit_pass();
}