xref: /aosp_15_r20/external/ltp/testcases/kernel/syscalls/kill/kill08.c (revision 49cdfc7efb34551c7342be41a7384b9c40d7cab7)
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