1*49cdfc7eSAndroid Build Coastguard Worker // SPDX-License-Identifier: GPL-2.0-or-later
2*49cdfc7eSAndroid Build Coastguard Worker /*
3*49cdfc7eSAndroid Build Coastguard Worker * Copyright (c) International Business Machines Corp., 2001
4*49cdfc7eSAndroid Build Coastguard Worker *
5*49cdfc7eSAndroid Build Coastguard Worker * Description:
6*49cdfc7eSAndroid Build Coastguard Worker * Verify the basic function of sigsuspend():
7*49cdfc7eSAndroid Build Coastguard Worker * 1) sigsuspend() can replace process's current signal mask
8*49cdfc7eSAndroid Build Coastguard Worker * by the specified signal mask and suspend the process
9*49cdfc7eSAndroid Build Coastguard Worker * execution until the delivery of a signal.
10*49cdfc7eSAndroid Build Coastguard Worker * 2) sigsuspend() should return after the execution of signal
11*49cdfc7eSAndroid Build Coastguard Worker * handler and restore the previous signal mask.
12*49cdfc7eSAndroid Build Coastguard Worker */
13*49cdfc7eSAndroid Build Coastguard Worker
14*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
15*49cdfc7eSAndroid Build Coastguard Worker #include <unistd.h>
16*49cdfc7eSAndroid Build Coastguard Worker #include <string.h>
17*49cdfc7eSAndroid Build Coastguard Worker #include <signal.h>
18*49cdfc7eSAndroid Build Coastguard Worker
19*49cdfc7eSAndroid Build Coastguard Worker #include "tst_test.h"
20*49cdfc7eSAndroid Build Coastguard Worker
21*49cdfc7eSAndroid Build Coastguard Worker static sigset_t signalset, sigset1, sigset2;
22*49cdfc7eSAndroid Build Coastguard Worker static volatile sig_atomic_t alarm_num;
23*49cdfc7eSAndroid Build Coastguard Worker
sig_handler(int sig)24*49cdfc7eSAndroid Build Coastguard Worker static void sig_handler(int sig)
25*49cdfc7eSAndroid Build Coastguard Worker {
26*49cdfc7eSAndroid Build Coastguard Worker alarm_num = sig;
27*49cdfc7eSAndroid Build Coastguard Worker }
28*49cdfc7eSAndroid Build Coastguard Worker
verify_sigsuspend(void)29*49cdfc7eSAndroid Build Coastguard Worker static void verify_sigsuspend(void)
30*49cdfc7eSAndroid Build Coastguard Worker {
31*49cdfc7eSAndroid Build Coastguard Worker alarm_num = 0;
32*49cdfc7eSAndroid Build Coastguard Worker
33*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGFILLSET(&sigset2);
34*49cdfc7eSAndroid Build Coastguard Worker
35*49cdfc7eSAndroid Build Coastguard Worker alarm(1);
36*49cdfc7eSAndroid Build Coastguard Worker
37*49cdfc7eSAndroid Build Coastguard Worker /* Unblock SIGALRM */
38*49cdfc7eSAndroid Build Coastguard Worker TEST(sigsuspend(&signalset));
39*49cdfc7eSAndroid Build Coastguard Worker
40*49cdfc7eSAndroid Build Coastguard Worker alarm(0);
41*49cdfc7eSAndroid Build Coastguard Worker
42*49cdfc7eSAndroid Build Coastguard Worker if (TST_RET != -1 || TST_ERR != EINTR) {
43*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL | TTERRNO,
44*49cdfc7eSAndroid Build Coastguard Worker "sigsuspend() returned value %ld", TST_RET);
45*49cdfc7eSAndroid Build Coastguard Worker return;
46*49cdfc7eSAndroid Build Coastguard Worker }
47*49cdfc7eSAndroid Build Coastguard Worker
48*49cdfc7eSAndroid Build Coastguard Worker if (alarm_num != SIGALRM) {
49*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL, "sigsuspend() didn't unblock SIGALRM");
50*49cdfc7eSAndroid Build Coastguard Worker return;
51*49cdfc7eSAndroid Build Coastguard Worker }
52*49cdfc7eSAndroid Build Coastguard Worker
53*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGPROCMASK(0, NULL, &sigset2);
54*49cdfc7eSAndroid Build Coastguard Worker if (memcmp(&sigset1, &sigset2, sizeof(unsigned long))) {
55*49cdfc7eSAndroid Build Coastguard Worker tst_res(TFAIL, "sigsuspend() failed to "
56*49cdfc7eSAndroid Build Coastguard Worker "restore the previous signal mask");
57*49cdfc7eSAndroid Build Coastguard Worker return;
58*49cdfc7eSAndroid Build Coastguard Worker }
59*49cdfc7eSAndroid Build Coastguard Worker
60*49cdfc7eSAndroid Build Coastguard Worker tst_res(TPASS, "sigsuspend() succeeded");
61*49cdfc7eSAndroid Build Coastguard Worker }
62*49cdfc7eSAndroid Build Coastguard Worker
setup(void)63*49cdfc7eSAndroid Build Coastguard Worker static void setup(void)
64*49cdfc7eSAndroid Build Coastguard Worker {
65*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGEMPTYSET(&signalset);
66*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGEMPTYSET(&sigset1);
67*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGADDSET(&sigset1, SIGALRM);
68*49cdfc7eSAndroid Build Coastguard Worker
69*49cdfc7eSAndroid Build Coastguard Worker struct sigaction sa_new = {
70*49cdfc7eSAndroid Build Coastguard Worker .sa_handler = sig_handler,
71*49cdfc7eSAndroid Build Coastguard Worker };
72*49cdfc7eSAndroid Build Coastguard Worker
73*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGACTION(SIGALRM, &sa_new, 0);
74*49cdfc7eSAndroid Build Coastguard Worker
75*49cdfc7eSAndroid Build Coastguard Worker /* Block SIGALRM */
76*49cdfc7eSAndroid Build Coastguard Worker SAFE_SIGPROCMASK(SIG_SETMASK, &sigset1, NULL);
77*49cdfc7eSAndroid Build Coastguard Worker }
78*49cdfc7eSAndroid Build Coastguard Worker
79*49cdfc7eSAndroid Build Coastguard Worker static struct tst_test test = {
80*49cdfc7eSAndroid Build Coastguard Worker .setup = setup,
81*49cdfc7eSAndroid Build Coastguard Worker .test_all = verify_sigsuspend,
82*49cdfc7eSAndroid Build Coastguard Worker };
83