248 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			248 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  *
 | |
|  *   Copyright (c) International Business Machines  Corp., 2001
 | |
|  *
 | |
|  *   This program is free software;  you can redistribute it and/or modify
 | |
|  *   it under the terms of the GNU General Public License as published by
 | |
|  *   the Free Software Foundation; either version 2 of the License, or
 | |
|  *   (at your option) any later version.
 | |
|  *
 | |
|  *   This program is distributed in the hope that it will be useful,
 | |
|  *   but WITHOUT ANY WARRANTY;  without even the implied warranty of
 | |
|  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See
 | |
|  *   the GNU General Public License for more details.
 | |
|  *
 | |
|  *   You should have received a copy of the GNU General Public License
 | |
|  *   along with this program;  if not, write to the Free Software
 | |
|  *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 | |
|  */
 | |
| 
 | |
| /*
 | |
|  * NAME
 | |
|  *	waitpid05.c
 | |
|  *
 | |
|  * DESCRIPTION
 | |
|  *	Check that when a child kills itself with a kill statement after
 | |
|  *	determining its process id by using getpid, the parent receives a
 | |
|  *	correct report of the cause of its death. This also indirectly
 | |
|  *	checks that getpid returns the correct process id.
 | |
|  *
 | |
|  * ALGORITHM
 | |
|  *	For signals 1 - 15: fork a child that determines it's own process
 | |
|  *	id, then sends the signal to itself.  The parent waits to see if the
 | |
|  *	demise of the child results in the signal number being returned to
 | |
|  *	the parent.
 | |
|  *
 | |
|  * USAGE:  <for command-line>
 | |
|  *      waitpid05 [-c n] [-e] [-i n] [-I x] [-P x] [-t]
 | |
|  *      where,  -c n : Run n copies concurrently.
 | |
|  *              -e   : Turn on errno logging.
 | |
|  *              -i n : Execute test n times.
 | |
|  *              -I x : Execute test for x seconds.
 | |
|  *              -P x : Pause for x seconds between iterations.
 | |
|  *              -t   : Turn on syscall timing.
 | |
|  *
 | |
|  * History
 | |
|  *	07/2001 John George
 | |
|  *		-Ported
 | |
|  *	04/2002 wjhuie sigset cleanups
 | |
|  *
 | |
|  * Restrictions
 | |
|  *	None
 | |
|  */
 | |
| 
 | |
| #include <sys/file.h>
 | |
| #include <sys/signal.h>
 | |
| #include <sys/types.h>
 | |
| #include <sys/wait.h>
 | |
| #include <sys/time.h>
 | |
| #include <sys/resource.h>
 | |
| #include <unistd.h>
 | |
| #include <errno.h>
 | |
| #include "test.h"
 | |
| 
 | |
| static void do_child(int);
 | |
| static void setup(void);
 | |
| static void cleanup(void);
 | |
| 
 | |
| char *TCID = "waitpid05";
 | |
| int TST_TOTAL = 1;
 | |
| 
 | |
| #ifdef UCLINUX
 | |
| static void do_child_uclinux(void);
 | |
| static int sig_uclinux;
 | |
| #endif
 | |
| 
 | |
| int main(int ac, char **av)
 | |
| {
 | |
| 	int pid, npid, sig, nsig;
 | |
| 	int exno, nexno, status;
 | |
| 	int lc;
 | |
| 
 | |
| 	tst_parse_opts(ac, av, NULL, NULL);
 | |
| 
 | |
| #ifdef UCLINUX
 | |
| 	maybe_run_child(&do_child_uclinux, "d", &sig_uclinux);
 | |
| #endif
 | |
| 
 | |
| 	setup();
 | |
| 
 | |
| 	/* check for looping state if -i option is given */
 | |
| 	for (lc = 0; TEST_LOOPING(lc); lc++) {
 | |
| 		/* reset tst_count in case we are looping */
 | |
| 		tst_count = 0;
 | |
| 
 | |
| 		/*
 | |
| 		 * Set SIGTERM to SIG_DFL as test driver sets up to ignore
 | |
| 		 * SIGTERM
 | |
| 		 */
 | |
| 		if (signal(SIGTERM, SIG_DFL) == SIG_ERR) {
 | |
| 			tst_resm(TFAIL, "Sigset SIGTERM failed, errno = %d",
 | |
| 				 errno);
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		exno = 1;
 | |
| 		for (sig = 1; sig <= 15; sig++) {
 | |
| 			if (sig == SIGUSR1 || sig == SIGUSR2 || sig == SIGBUS)
 | |
| 				continue;
 | |
| 
 | |
| 			/*Initialize signal to its default action */
 | |
| 			signal(sig, SIG_DFL);
 | |
| 			pid = FORK_OR_VFORK();
 | |
| 
 | |
| 			if (pid == 0) {
 | |
| #ifdef UCLINUX
 | |
| 				self_exec(av[0], "d", sig);
 | |
| 				/* No fork() error check is done so don't */
 | |
| 				/* do an error check here */
 | |
| #else
 | |
| 				do_child(sig);
 | |
| #endif
 | |
| 			} else {
 | |
| 				errno = 0;
 | |
| 				while (((npid = waitpid(pid, &status, 0)) !=
 | |
| 					-1) || (errno == EINTR)) {
 | |
| 					if (errno == EINTR)
 | |
| 						continue;
 | |
| 
 | |
| 					if (npid != pid) {
 | |
| 						tst_resm(TFAIL, "waitpid "
 | |
| 							 "error: unexpected "
 | |
| 							 "pid returned");
 | |
| 					} else {
 | |
| 						tst_resm(TPASS, "received "
 | |
| 							 "expected pid.");
 | |
| 					}
 | |
| 
 | |
| 					nsig = status % 256;
 | |
| 
 | |
| 					/*
 | |
| 					 * to check if the core dump bit has
 | |
| 					 * been set, bit #7
 | |
| 					 */
 | |
| 					if (nsig >= 128) {
 | |
| 						nsig -= 128;
 | |
| 						if ((sig == 1) || (sig == 2) ||
 | |
| 						    (sig == 9) || (sig == 13) ||
 | |
| 						    (sig == 14) ||
 | |
| 						    (sig == 15)) {
 | |
| 							tst_resm(TFAIL,
 | |
| 								 "signal "
 | |
| 								 "error : "
 | |
| 								 "core dump "
 | |
| 								 "bit set for"
 | |
| 								 " exception "
 | |
| 								 "number %d",
 | |
| 								 sig);
 | |
| 						}
 | |
| 					} else if ((sig == 3) || (sig == 4) ||
 | |
| 						   (sig == 5) || (sig == 6) ||
 | |
| 						   (sig == 8) || (sig == 11)) {
 | |
| 						tst_resm(TFAIL,
 | |
| 							 "signal error: "
 | |
| 							 "core dump bit not "
 | |
| 							 "set for exception "
 | |
| 							 "number %d", sig);
 | |
| 					}
 | |
| 
 | |
| 					/*
 | |
| 					 * nsig is the signal number returned
 | |
| 					 * by waitpid
 | |
| 					 */
 | |
| 					if (nsig != sig) {
 | |
| 						tst_resm(TFAIL, "waitpid "
 | |
| 							 "error: unexpected "
 | |
| 							 "signal returned");
 | |
| 						tst_resm(TINFO, "got signal "
 | |
| 							 "%d, expected  "
 | |
| 							 "%d", nsig, sig);
 | |
| 					}
 | |
| 
 | |
| 					/*
 | |
| 					 * nexno is the exit number returned
 | |
| 					 * by waitpid
 | |
| 					 */
 | |
| 					nexno = status / 256;
 | |
| 					if (nexno != 0) {
 | |
| 						tst_resm(TFAIL, "signal "
 | |
| 							 "error: unexpected "
 | |
| 							 "exit number "
 | |
| 							 "returned");
 | |
| 					} else {
 | |
| 						tst_resm(TPASS, "received "
 | |
| 							 "expected exit number.");
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (access("core", F_OK) == 0)
 | |
| 			unlink("core");
 | |
| 	}
 | |
| 
 | |
| 	cleanup();
 | |
| 	tst_exit();
 | |
| }
 | |
| 
 | |
| static void do_child(int sig)
 | |
| {
 | |
| 	int exno = 1;
 | |
| 	int pid = getpid();
 | |
| 
 | |
| 	if (kill(pid, sig) == -1) {
 | |
| 		tst_resm(TFAIL, "kill error: kill unsuccessful");
 | |
| 		exit(exno);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| #ifdef UCLINUX
 | |
| /*
 | |
|  * do_child_uclinux()
 | |
|  *	run do_child with the appropriate sig variable
 | |
|  */
 | |
| static void do_child_uclinux(void)
 | |
| {
 | |
| 	do_child(sig_uclinux);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void setup(void)
 | |
| {
 | |
| 	struct rlimit newlimit;
 | |
| 
 | |
| 	TEST_PAUSE;
 | |
| 
 | |
| 	tst_tmpdir();
 | |
| 
 | |
| 	newlimit.rlim_max = newlimit.rlim_cur = RLIM_INFINITY;
 | |
| 	if (setrlimit(RLIMIT_CORE, &newlimit) != 0)
 | |
| 		tst_resm(TWARN,
 | |
| 			 "setrlimit(RLIMIT_CORE,RLIM_INFINITY) failed; this may cause some false core-dump test failures");
 | |
| }
 | |
| 
 | |
| static void cleanup(void)
 | |
| {
 | |
| 	tst_rmdir();
 | |
| }
 |