287 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			287 lines
		
	
	
		
			8.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2000 Silicon Graphics, Inc.  All Rights Reserved.
 | |
|  * 
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms of version 2 of the GNU General Public License as
 | |
|  * published by the Free Software Foundation.
 | |
|  * 
 | |
|  * This program is distributed in the hope that it would be useful, but
 | |
|  * WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 | |
|  * 
 | |
|  * Further, this software is distributed without any warranty that it is
 | |
|  * free of the rightful claim of any third person regarding infringement
 | |
|  * or the like.  Any license provided herein, whether implied or
 | |
|  * otherwise, applies only to this software file.  Patent licenses, if
 | |
|  * any, provided herein do not apply to combinations of this program with
 | |
|  * other software, or any other product whatsoever.
 | |
|  * 
 | |
|  * You should have received a copy of the GNU General Public License along
 | |
|  * with this program; if not, write the Free Software Foundation, Inc., 59
 | |
|  * Temple Place - Suite 330, Boston MA 02111-1307, USA.
 | |
|  * 
 | |
|  * Contact information: Silicon Graphics, Inc., 1600 Amphitheatre Pkwy,
 | |
|  * Mountain View, CA  94043, or:
 | |
|  * 
 | |
|  * http://www.sgi.com 
 | |
|  * 
 | |
|  * For further information regarding this notice, see: 
 | |
|  * 
 | |
|  * http://oss.sgi.com/projects/GenInfo/NoticeExplan/
 | |
|  */
 | |
| 
 | |
| /*****************************************************************************
 | |
| 	OS Testing  - Silicon Graphics, Inc.
 | |
| 
 | |
| 	FUNCTION IDENTIFIER : tst_sig  Set up for unexpected signals.
 | |
| 
 | |
| 	AUTHOR          : David D. Fenner
 | |
| 
 | |
| 	CO-PILOT        : Bill Roske
 | |
| 
 | |
| 	DATE STARTED    : 06/06/90
 | |
| 
 | |
| 	This module may be linked with c-modules requiring unexpected
 | |
| 	signal handling.  The parameters to tst_sig are as follows:
 | |
| 
 | |
| 		fork_flag - set to FORK or NOFORK depending upon whether the
 | |
| 			calling program executes a fork() system call.  It
 | |
| 			is normally the case that the calling program treats
 | |
| 			SIGCLD as an expected signal if fork() is being used.
 | |
| 
 | |
| 		handler - a pointer to the unexpected signal handler to
 | |
| 			be executed after an unexpected signal has been
 | |
| 			detected.  If handler is set to DEF_HANDLER, a 
 | |
| 			default handler is used.  This routine should be
 | |
| 			declared as function returning an int.
 | |
| 
 | |
| 		cleanup - a pointer to a cleanup routine to be executed
 | |
| 			by the unexpected signal handler before tst_exit is
 | |
| 			called.  This parameter is set to NULL if no cleanup
 | |
| 			routine is required.  An external variable, T_cleanup
 | |
| 			is set so that other user-defined handlers have 
 | |
| 			access to the cleanup routine.  This routine should be
 | |
| 			declared as returning type void.
 | |
| 
 | |
| ***************************************************************************/
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <string.h>
 | |
| #include <signal.h>
 | |
| #include <unistd.h>
 | |
| #include "test.h"
 | |
| 
 | |
| #define MAXMESG 150		/* size of mesg string sent to tst_res */
 | |
| 
 | |
| void (*T_cleanup)(void);		/* pointer to cleanup function */
 | |
| 
 | |
| /****************************************************************************
 | |
|  * STD_COPIES is defined in parse_opts.c but is externed here in order to
 | |
|  * test whether SIGCHILD should be ignored or not.
 | |
|  ***************************************************************************/
 | |
| extern int STD_COPIES;
 | |
| 
 | |
| static void def_handler(int);		/* default signal handler */
 | |
| static void (*tst_setup_signal( int, void (*)(int)))(int);
 | |
| 
 | |
| /****************************************************************************
 | |
|  * tst_sig() : set-up to catch unexpected signals.  fork_flag is set to NOFORK
 | |
|  *    if SIGCLD is to be an "unexpected signal", otherwise it is set to
 | |
|  *    FORK.  cleanup points to a cleanup routine to be executed before
 | |
|  *    tst_exit is called (cleanup is set to NULL if no cleanup is desired).
 | |
|  *    handler is a pointer to the signal handling routine (if handler is
 | |
|  *    set to NULL, a default handler is used).
 | |
|  ***************************************************************************/
 | |
| 
 | |
| void
 | |
| tst_sig(int fork_flag, void (*handler)(int), void (*cleanup)(void))
 | |
| {
 | |
| 	char mesg[MAXMESG];		/* message buffer for tst_res */
 | |
| 	int sig;
 | |
| #ifdef _SC_SIGRT_MIN
 | |
|         long sigrtmin, sigrtmax;
 | |
| #endif
 | |
| 
 | |
| 	/*
 | |
| 	 * save T_cleanup and handler function pointers
 | |
| 	 */
 | |
| 	T_cleanup = cleanup;		/* used by default handler */
 | |
| 
 | |
| 	if (handler == DEF_HANDLER) {
 | |
| 		/* use default handler */
 | |
| 		handler = def_handler;
 | |
| 	}
 | |
|   
 | |
| #ifdef _SC_SIGRT_MIN
 | |
|         sigrtmin = sysconf(_SC_SIGRT_MIN);
 | |
|         sigrtmax = sysconf(_SC_SIGRT_MAX);
 | |
| #endif
 | |
| 
 | |
| 	/*
 | |
| 	 * now loop through all signals and set the handlers
 | |
| 	 */
 | |
| 
 | |
| 	for (sig = 1; sig < NSIG; sig++) {
 | |
| 	    /*
 | |
| 	     * SIGKILL is never unexpected.
 | |
| 	     * SIGCLD is only unexpected when
 | |
| 	     *    no forking is being done.
 | |
| 	     * SIGINFO is used for file quotas and should be expected
 | |
| 	     */
 | |
| 
 | |
| #ifdef _SC_SIGRT_MIN
 | |
|             if (sig >= sigrtmin && sig <= sigrtmax)
 | |
|                 continue;
 | |
| #endif
 | |
| 
 | |
| 	    switch (sig) {
 | |
| 	        case SIGKILL:
 | |
| 	        case SIGSTOP:
 | |
| 	        case SIGCONT:
 | |
| #if !defined(_SC_SIGRT_MIN) && defined(__SIGRTMIN) && defined(__SIGRTMAX)
 | |
| 	     /* Ignore all real-time signals */
 | |
| 		case __SIGRTMIN:
 | |
| 		case __SIGRTMIN+1:
 | |
| 		case __SIGRTMIN+2:
 | |
| 		case __SIGRTMIN+3:
 | |
| 		case __SIGRTMIN+4:
 | |
| 		case __SIGRTMIN+5:
 | |
| 		case __SIGRTMIN+6:
 | |
| 		case __SIGRTMIN+7:
 | |
| 		case __SIGRTMIN+8:
 | |
| 		case __SIGRTMIN+9:
 | |
| 		case __SIGRTMIN+10:
 | |
| 		case __SIGRTMIN+11:
 | |
| 		case __SIGRTMIN+12:
 | |
| 		case __SIGRTMIN+13:
 | |
| 		case __SIGRTMIN+14:
 | |
| 		case __SIGRTMIN+15:
 | |
| /* __SIGRTMIN is 37 on HPPA rather than 32 *
 | |
|  * as on i386, etc.                        */
 | |
| #if !defined(__hppa__)
 | |
| 		case __SIGRTMAX-15:
 | |
| 		case __SIGRTMAX-14:
 | |
| 		case __SIGRTMAX-13:
 | |
| 		case __SIGRTMAX-12:
 | |
| 		case __SIGRTMAX-11:
 | |
| #endif
 | |
| 		case __SIGRTMAX-10:
 | |
| 		case __SIGRTMAX-9:
 | |
| 		case __SIGRTMAX-8:
 | |
| 		case __SIGRTMAX-7:
 | |
| 		case __SIGRTMAX-6:
 | |
| 		case __SIGRTMAX-5:
 | |
| 		case __SIGRTMAX-4:
 | |
| 		case __SIGRTMAX-3:
 | |
| 		case __SIGRTMAX-2:
 | |
| 		case __SIGRTMAX-1:
 | |
| 		case __SIGRTMAX:
 | |
| #endif
 | |
| #ifdef CRAY
 | |
| 	        case SIGINFO:
 | |
| 	        case SIGRECOVERY:	/* allow chkpnt/restart */
 | |
| #endif  /* CRAY */
 | |
| 
 | |
| #ifdef SIGSWAP
 | |
| 		case SIGSWAP:
 | |
| #endif /* SIGSWAP */
 | |
| 
 | |
| #ifdef SIGCKPT
 | |
| 	        case SIGCKPT:
 | |
| #endif
 | |
| #ifdef SIGRESTART
 | |
| 	        case SIGRESTART:
 | |
| #endif
 | |
|                 /*
 | |
|                  * pthread-private signals SIGPTINTR and SIGPTRESCHED.
 | |
|                  * Setting a handler for these signals is disallowed when
 | |
|                  * the binary is linked against libpthread.
 | |
|                  */
 | |
| #ifdef SIGPTINTR
 | |
|                 case SIGPTINTR:
 | |
| #endif /* SIGPTINTR */
 | |
| #ifdef SIGPTRESCHED
 | |
|                 case SIGPTRESCHED:
 | |
| #endif /* SIGPTRESCHED */
 | |
| #ifdef _SIGRESERVE
 | |
|               case _SIGRESERVE:
 | |
| #endif
 | |
| #ifdef _SIGDIL
 | |
|               case _SIGDIL:
 | |
| #endif
 | |
| #ifdef _SIGCANCEL
 | |
|               case _SIGCANCEL:
 | |
| #endif
 | |
| #ifdef _SIGGFAULT
 | |
|               case _SIGGFAULT:
 | |
| #endif
 | |
| 	            break;
 | |
| 
 | |
| 	        case SIGCLD:
 | |
| 	            if ( fork_flag == FORK || STD_COPIES > 1)
 | |
| 		        continue;
 | |
| 
 | |
| 	        default:
 | |
| 		    if (tst_setup_signal(sig, handler) == SIG_ERR) {
 | |
| 		        (void) sprintf(mesg,
 | |
| 			    "signal() failed for signal %d. error:%d %s.",
 | |
| 			    sig, errno, strerror(errno));
 | |
| 		        tst_resm(TWARN, mesg);
 | |
| 		    }
 | |
| 		break;
 | |
|             }
 | |
| #ifdef __sgi
 | |
| 	    /* On irix  (07/96), signal() fails when signo is 33 or higher */
 | |
| 	    if ( sig+1 >= 33 )
 | |
| 		break;
 | |
| #endif  /*  __sgi */
 | |
| 
 | |
| 	} /* endfor */
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| /****************************************************************************
 | |
|  * def_handler() : default signal handler that is invoked when
 | |
|  *      an unexpected signal is caught.
 | |
|  ***************************************************************************/
 | |
| 
 | |
| static void
 | |
| def_handler(int sig)
 | |
| {
 | |
| 
 | |
| 	/*
 | |
|          * Break remaining test cases, do any cleanup, then exit
 | |
| 	 */
 | |
| 	tst_brkm(TBROK, 0, "Unexpected signal %d received.", sig);
 | |
| 
 | |
| 	/* now cleanup and exit */
 | |
| 	if (T_cleanup) {
 | |
| 		(*T_cleanup)();
 | |
| 	}
 | |
| 
 | |
| 	tst_exit();
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * tst_setup_signal - A function like signal(), but we have
 | |
|  *                    control over its personality.
 | |
|  */
 | |
| static void (*tst_setup_signal( int sig, void (*handler)(int)))(int)
 | |
| { 
 | |
|   struct sigaction my_act,old_act;
 | |
|   int ret;
 | |
| 
 | |
|   my_act.sa_handler = handler;
 | |
|   my_act.sa_flags = SA_RESTART;
 | |
|   sigemptyset(&my_act.sa_mask);
 | |
| 
 | |
|   ret = sigaction(sig, &my_act, &old_act);
 | |
| 
 | |
|   if ( ret == 0 )
 | |
|     return( old_act.sa_handler );
 | |
|   else
 | |
|     return( SIG_ERR );
 | |
| }
 | |
| 
 | 
