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 );
 | 
						|
}
 | 
						|
 |