1 /*
2 *
3 * Copyright (c) International Business Machines Corp., 2001
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See
13 * the GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18 */
19
20 /*
21 * NAME
22 * kill08.c
23 *
24 * DESCRIPTION
25 * Test case to check the basic functionality of kill() when kill an
26 * entire process group.
27 *
28 * ALGORITHM
29 * call setup
30 * loop if the -i option was given
31 * fork 5 childeren
32 * execute the kill system call
33 * check the return value
34 * if return value is -1
35 * issue a FAIL message, break remaining tests and cleanup
36 * if we are doing functional testing
37 * if the processes were terminated with the expected signal.
38 * issue a PASS message
39 * otherwise
40 * issue a FAIL message
41 * call cleanup
42 *
43 * USAGE
44 * kill08 [-c n] [-f] [-i n] [-I x] [-P x] [-t]
45 * where, -c n : Run n copies concurrently.
46 * -f : Turn off functionality Testing.
47 * -i n : Execute test n times.
48 * -I x : Execute test for x seconds.
49 * -P x : Pause for x seconds between iterations.
50 * -t : Turn on syscall timing.
51 *
52 * HISTORY
53 * 07/2001 Ported by Wayne Boyer
54 *
55 * RESTRICTIONS
56 * This test should be run as a non-root user.
57 */
58
59 #include "test.h"
60
61 #include <signal.h>
62 #include <errno.h>
63 #include <sys/wait.h>
64
65 void cleanup(void);
66 void setup(void);
67 void do_child(void);
68
69 char *TCID = "kill08";
70 int TST_TOTAL = 1;
71
72 #define TEST_SIG SIGKILL
73
main(int ac,char ** av)74 int main(int ac, char **av)
75 {
76 int lc;
77 pid_t pid1, pid2;
78 int exno, status, nsig, i;
79
80 tst_parse_opts(ac, av, NULL, NULL);
81
82 setup(); /* global setup */
83
84 /* The following loop checks looping state if -i option given */
85 for (lc = 0; TEST_LOOPING(lc); lc++) {
86
87 /* reset tst_count in case we are looping */
88 tst_count = 0;
89 status = 1;
90 exno = 1;
91
92 /* Fork a process and set the process group so that */
93 /* it is different from this one. Fork 5 more children. */
94
95 pid1 = tst_fork();
96 if (pid1 < 0) {
97 tst_brkm(TBROK, cleanup, "Fork of first child failed");
98 } else if (pid1 == 0) {
99 setpgrp();
100 for (i = 0; i < 5; i++) {
101 pid2 = tst_fork();
102 if (pid2 < 0) {
103 tst_brkm(TBROK, cleanup, "Fork failed");
104 } else if (pid2 == 0) {
105 do_child();
106 }
107 }
108 /* Kill all processes in this process group */
109 TEST(kill(0, TEST_SIG));
110 pause();
111 exit(exno);
112 } else {
113 waitpid(pid1, &status, 0);
114 if (TEST_RETURN != 0) {
115 tst_brkm(TFAIL, cleanup, "%s failed - errno = "
116 "%d : %s", TCID, TEST_ERRNO,
117 strerror(TEST_ERRNO));
118 }
119 }
120
121 /*
122 * Check to see if the process was terminated with the
123 * expected signal.
124 */
125 nsig = WTERMSIG(status);
126 if (nsig == TEST_SIG) {
127 tst_resm(TPASS, "received expected signal %d",
128 nsig);
129 } else {
130 tst_resm(TFAIL,
131 "expected signal %d received %d",
132 TEST_SIG, nsig);
133 }
134 }
135
136 cleanup();
137 tst_exit();
138 }
139
140 /*
141 * do_child()
142 */
do_child(void)143 void do_child(void)
144 {
145 int exno = 1;
146
147 pause();
148 exit(exno);
149 }
150
151 /*
152 * setup() - performs all ONE TIME setup for this test
153 */
setup(void)154 void setup(void)
155 {
156
157 TEST_PAUSE;
158 }
159
160 /*
161 * cleanup() - performs all the ONE TIME cleanup for this test at completion
162 * or premature exit.
163 */
cleanup(void)164 void cleanup(void)
165 {
166
167 }
168