// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (c) International Business Machines Corp., 2007 * 13/11/08 Gowrishankar M * Copyright (C) 2022 SUSE LLC Andrea Cervesato */ /*\ * [Description] * * Clone a process with CLONE_NEWPID flag, block SIGUSR1 signal before sending * it from parent and check if it's received once SIGUSR1 signal is unblocked. */ #define _GNU_SOURCE 1 #include #include "tst_test.h" #include "lapi/sched.h" static volatile int signals; static volatile int last_signo; static void child_signal_handler(LTP_ATTRIBUTE_UNUSED int sig, siginfo_t *si, LTP_ATTRIBUTE_UNUSED void *unused) { last_signo = si->si_signo; signals++; } static void child_func(void) { struct sigaction sa; sigset_t newset; pid_t cpid, ppid; cpid = tst_getpid(); ppid = getppid(); if (cpid != 1 || ppid != 0) { tst_res(TFAIL, "Got unexpected result of cpid=%d ppid=%d", cpid, ppid); return; } SAFE_SIGEMPTYSET(&newset); SAFE_SIGADDSET(&newset, SIGUSR1); SAFE_SIGPROCMASK(SIG_BLOCK, &newset, 0); TST_CHECKPOINT_WAKE_AND_WAIT(0); sa.sa_flags = SA_SIGINFO; SAFE_SIGFILLSET(&sa.sa_mask); sa.sa_sigaction = child_signal_handler; SAFE_SIGACTION(SIGUSR1, &sa, NULL); SAFE_SIGPROCMASK(SIG_UNBLOCK, &newset, 0); if (signals != 1) { tst_res(TFAIL, "Received %d signals", signals); return; } if (last_signo != SIGUSR1) { tst_res(TFAIL, "Received %s signal", tst_strsig(last_signo)); return; } tst_res(TPASS, "Received SIGUSR1 signal after unblock"); } static void run(void) { const struct tst_clone_args args = { .flags = CLONE_NEWPID, .exit_signal = SIGCHLD, }; int pid; pid = SAFE_CLONE(&args); if (!pid) { child_func(); return; } TST_CHECKPOINT_WAIT(0); SAFE_KILL(pid, SIGUSR1); TST_CHECKPOINT_WAKE(0); } static struct tst_test test = { .test_all = run, .needs_root = 1, .forks_child = 1, .needs_checkpoints = 1, .needs_kconfigs = (const char *[]) { "CONFIG_PID_NS", NULL, }, };