587 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			587 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*-
 | 
						|
 * Copyright (c) 1991, 1993
 | 
						|
 *	The Regents of the University of California.  All rights reserved.
 | 
						|
 *
 | 
						|
 * This code is derived from software contributed to Berkeley by
 | 
						|
 * Kenneth Almquist.
 | 
						|
 *
 | 
						|
 * Redistribution and use in source and binary forms, with or without
 | 
						|
 * modification, are permitted provided that the following conditions
 | 
						|
 * are met:
 | 
						|
 * 1. Redistributions of source code must retain the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer.
 | 
						|
 * 2. Redistributions in binary form must reproduce the above copyright
 | 
						|
 *    notice, this list of conditions and the following disclaimer in the
 | 
						|
 *    documentation and/or other materials provided with the distribution.
 | 
						|
 * 4. Neither the name of the University nor the names of its contributors
 | 
						|
 *    may be used to endorse or promote products derived from this software
 | 
						|
 *    without specific prior written permission.
 | 
						|
 *
 | 
						|
 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
 | 
						|
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | 
						|
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 | 
						|
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
 | 
						|
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | 
						|
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 | 
						|
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 | 
						|
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 | 
						|
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 | 
						|
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 | 
						|
 * SUCH DAMAGE.
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef lint
 | 
						|
#if 0
 | 
						|
static char sccsid[] = "@(#)trap.c	8.5 (Berkeley) 6/5/95";
 | 
						|
#endif
 | 
						|
#endif /* not lint */
 | 
						|
/*
 | 
						|
#include <sys/cdefs.h>
 | 
						|
__FBSDID("$FreeBSD: src/bin/sh/trap.c,v 1.29 2004/04/06 20:06:51 markm Exp $");
 | 
						|
*/
 | 
						|
 | 
						|
#include <signal.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <stdlib.h>
 | 
						|
 | 
						|
#include "shell.h"
 | 
						|
#include "main.h"
 | 
						|
#include "nodes.h"	/* for other headers */
 | 
						|
#include "eval.h"
 | 
						|
#include "jobs.h"
 | 
						|
#include "show.h"
 | 
						|
#include "options.h"
 | 
						|
#include "syntax.h"
 | 
						|
#include "output.h"
 | 
						|
#include "memalloc.h"
 | 
						|
#include "error.h"
 | 
						|
#include "trap.h"
 | 
						|
#include "mystring.h"
 | 
						|
#if !defined(NO_HISTORY) && !defined(EDITLINE)
 | 
						|
#include "myhistedit.h"
 | 
						|
#endif
 | 
						|
#include "builtins.h"
 | 
						|
 | 
						|
#ifdef __minix
 | 
						|
#define NO_SIGINTERRUPT
 | 
						|
#define NO_SYS_SIGNAME
 | 
						|
#define NO_SYS_SIGLIST
 | 
						|
 | 
						|
#endif
 | 
						|
 | 
						|
typedef void (*sig_T)(int);
 | 
						|
 | 
						|
/*
 | 
						|
 * Sigmode records the current value of the signal handlers for the various
 | 
						|
 * modes.  A value of zero means that the current handler is not known.
 | 
						|
 * S_HARD_IGN indicates that the signal was ignored on entry to the shell,
 | 
						|
 */
 | 
						|
 | 
						|
#define S_DFL 1			/* default signal handling (SIG_DFL) */
 | 
						|
#define S_CATCH 2		/* signal is caught */
 | 
						|
#define S_IGN 3			/* signal is ignored (SIG_IGN) */
 | 
						|
#define S_HARD_IGN 4		/* signal is ignored permanently */
 | 
						|
#define S_RESET 5		/* temporary - to reset a hard ignored sig */
 | 
						|
 | 
						|
 | 
						|
MKINIT char sigmode[_NSIG];	/* current value of signal */
 | 
						|
int pendingsigs;		/* indicates some signal received */
 | 
						|
int is_interactive= -1;		/* Shell is interactive */
 | 
						|
int in_dotrap;			/* do we execute in a trap handler? */
 | 
						|
static char *volatile trap[_NSIG];	/* trap handler commands */
 | 
						|
static volatile sig_atomic_t gotsig[_NSIG]; 
 | 
						|
				/* indicates specified signal received */
 | 
						|
static int ignore_sigchld;	/* Used while handling SIGCHLD traps. */
 | 
						|
volatile sig_atomic_t gotwinch;
 | 
						|
 | 
						|
static int sigstring_to_signum (char *);
 | 
						|
static void printsignals (void);
 | 
						|
static int getsigaction(int, sig_T *);
 | 
						|
static void onsig (int);
 | 
						|
#ifdef NO_SIGINTERRUPT
 | 
						|
static int siginterrupt (int,int);
 | 
						|
#endif
 | 
						|
static char *strsigname (int);
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Map a string to a signal number.
 | 
						|
 */
 | 
						|
static int
 | 
						|
sigstring_to_signum(char *sig)
 | 
						|
{
 | 
						|
 | 
						|
	if (is_number(sig)) {
 | 
						|
		int signo;
 | 
						|
 | 
						|
		signo = atoi(sig);
 | 
						|
		return ((signo >= 0 && signo < _NSIG) ? signo : (-1));
 | 
						|
	} else if (strcasecmp(sig, "exit") == 0) {
 | 
						|
		return (0);
 | 
						|
	} else {
 | 
						|
		int n;
 | 
						|
 | 
						|
		if (strncasecmp(sig, "sig", 3) == 0)
 | 
						|
			sig += 3;
 | 
						|
		for (n = 1; n < _NSIG; n++)
 | 
						|
			if (strcasecmp(strsigname(n), sig) == 0)
 | 
						|
				return (n);
 | 
						|
	}
 | 
						|
	return (-1);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Print a list of valid signal names.
 | 
						|
 */
 | 
						|
static void
 | 
						|
printsignals(void)
 | 
						|
{
 | 
						|
	int n, outlen;
 | 
						|
 | 
						|
	outlen = 0;
 | 
						|
	for (n = 1; n < _NSIG; n++) {
 | 
						|
		if (strsigname(n)) {
 | 
						|
			out1fmt("%s", strsigname(n));
 | 
						|
			outlen += strlen(strsigname(n));
 | 
						|
		} else {
 | 
						|
			out1fmt("%d", n);
 | 
						|
			outlen += 3;	/* good enough */
 | 
						|
		}
 | 
						|
		++outlen;
 | 
						|
		if (outlen > 70 || n == _NSIG - 1) {
 | 
						|
			out1str("\n");
 | 
						|
			outlen = 0;
 | 
						|
		} else {
 | 
						|
			out1c(' ');
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * The trap builtin.
 | 
						|
 */
 | 
						|
int
 | 
						|
trapcmd(int argc, char **argv)
 | 
						|
{
 | 
						|
	char *action;
 | 
						|
	int signo;
 | 
						|
 | 
						|
	if (argc <= 1) {
 | 
						|
		for (signo = 0 ; signo < _NSIG ; signo++) {
 | 
						|
			if (trap[signo] != NULL) {
 | 
						|
				if (signo == 0) {
 | 
						|
					out1fmt("trap -- '%s' %s\n",
 | 
						|
					    trap[signo], "exit");
 | 
						|
				} else if (strsigname(signo)) {
 | 
						|
					out1fmt("trap -- '%s' %s\n",
 | 
						|
					    trap[signo], strsigname(signo));
 | 
						|
				} else {
 | 
						|
					out1fmt("trap -- '%s' %d\n",
 | 
						|
					    trap[signo], signo);
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return 0;
 | 
						|
	}
 | 
						|
	action = NULL;
 | 
						|
	if (*++argv && strcmp(*argv, "--") == 0)
 | 
						|
		argv++;
 | 
						|
	if (*argv && sigstring_to_signum(*argv) == -1) {
 | 
						|
		if ((*argv)[0] != '-') {
 | 
						|
			action = *argv;
 | 
						|
			argv++;
 | 
						|
		} else if ((*argv)[1] == '\0') {
 | 
						|
			argv++;
 | 
						|
		} else if ((*argv)[1] == 'l' && (*argv)[2] == '\0') {
 | 
						|
			printsignals();
 | 
						|
			return 0;
 | 
						|
		} else {
 | 
						|
			error("bad option %s", *argv);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	while (*argv) {
 | 
						|
		if ((signo = sigstring_to_signum(*argv)) == -1)
 | 
						|
			error("bad signal %s", *argv);
 | 
						|
		INTOFF;
 | 
						|
		if (action)
 | 
						|
			action = savestr(action);
 | 
						|
		if (trap[signo])
 | 
						|
			ckfree(trap[signo]);
 | 
						|
		trap[signo] = action;
 | 
						|
		if (signo != 0)
 | 
						|
			setsignal(signo);
 | 
						|
		INTON;
 | 
						|
		argv++;
 | 
						|
	}
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Clear traps on a fork.
 | 
						|
 */
 | 
						|
void
 | 
						|
clear_traps(void)
 | 
						|
{
 | 
						|
	char *volatile *tp;
 | 
						|
 | 
						|
	for (tp = trap ; tp <= &trap[_NSIG - 1] ; tp++) {
 | 
						|
		if (*tp && **tp) {	/* trap not NULL or SIG_IGN */
 | 
						|
			INTOFF;
 | 
						|
			ckfree(*tp);
 | 
						|
			*tp = NULL;
 | 
						|
			if (tp != &trap[0])
 | 
						|
				setsignal(tp - trap);
 | 
						|
			INTON;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Set the signal handler for the specified signal.  The routine figures
 | 
						|
 * out what it should be set to.
 | 
						|
 */
 | 
						|
void
 | 
						|
setsignal(int signo)
 | 
						|
{
 | 
						|
	int action;
 | 
						|
	sig_T sig, sigact = SIG_DFL;
 | 
						|
	char *t;
 | 
						|
 | 
						|
	if ((t = trap[signo]) == NULL)
 | 
						|
		action = S_DFL;
 | 
						|
	else if (*t != '\0')
 | 
						|
		action = S_CATCH;
 | 
						|
	else
 | 
						|
		action = S_IGN;
 | 
						|
	if (action == S_DFL) {
 | 
						|
		switch (signo) {
 | 
						|
		case SIGINT:
 | 
						|
			action = S_CATCH;
 | 
						|
			break;
 | 
						|
		case SIGQUIT:
 | 
						|
#if DEBUG
 | 
						|
			{
 | 
						|
			extern int debug;
 | 
						|
 | 
						|
			if (debug)
 | 
						|
				break;
 | 
						|
			}
 | 
						|
#endif
 | 
						|
			action = S_CATCH;
 | 
						|
			break;
 | 
						|
		case SIGTERM:
 | 
						|
			if (rootshell && iflag)
 | 
						|
				action = S_IGN;
 | 
						|
			break;
 | 
						|
#if JOBS
 | 
						|
		case SIGTSTP:
 | 
						|
		case SIGTTOU:
 | 
						|
			if (rootshell && mflag)
 | 
						|
				action = S_IGN;
 | 
						|
			break;
 | 
						|
#endif
 | 
						|
#ifndef NO_HISTORY
 | 
						|
		case SIGWINCH:
 | 
						|
			if (rootshell && iflag)
 | 
						|
				action = S_CATCH;
 | 
						|
			break;
 | 
						|
#endif
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	t = &sigmode[signo];
 | 
						|
	if (*t == 0) {
 | 
						|
		/*
 | 
						|
		 * current setting unknown
 | 
						|
		 */
 | 
						|
		if (!getsigaction(signo, &sigact)) {
 | 
						|
			/*
 | 
						|
			 * Pretend it worked; maybe we should give a warning
 | 
						|
			 * here, but other shells don't. We don't alter
 | 
						|
			 * sigmode, so that we retry every time.
 | 
						|
			 */
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		if (sigact == SIG_IGN) {
 | 
						|
			if (mflag && (signo == SIGTSTP ||
 | 
						|
			     signo == SIGTTIN || signo == SIGTTOU)) {
 | 
						|
				*t = S_IGN;	/* don't hard ignore these */
 | 
						|
			} else
 | 
						|
				*t = S_HARD_IGN;
 | 
						|
		} else {
 | 
						|
			*t = S_RESET;	/* force to be set */
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (*t == S_HARD_IGN || *t == action)
 | 
						|
		return;
 | 
						|
	switch (action) {
 | 
						|
		case S_DFL:	sigact = SIG_DFL;	break;
 | 
						|
		case S_CATCH:  	sigact = onsig;		break;
 | 
						|
		case S_IGN:	sigact = SIG_IGN;	break;
 | 
						|
	}
 | 
						|
	*t = action;
 | 
						|
	sig = signal(signo, sigact);
 | 
						|
	if (sig != SIG_ERR && action == S_CATCH)
 | 
						|
		siginterrupt(signo, 1);
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Return the current setting for sig w/o changing it.
 | 
						|
 */
 | 
						|
static int
 | 
						|
getsigaction(int signo, sig_T *sigact)
 | 
						|
{
 | 
						|
	struct sigaction sa;
 | 
						|
 | 
						|
	if (sigaction(signo, (struct sigaction *)0, &sa) == -1)
 | 
						|
		return 0;
 | 
						|
	*sigact = (sig_T) sa.sa_handler;
 | 
						|
	return 1;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Ignore a signal.
 | 
						|
 */
 | 
						|
void
 | 
						|
ignoresig(int signo)
 | 
						|
{
 | 
						|
 | 
						|
	if (sigmode[signo] != S_IGN && sigmode[signo] != S_HARD_IGN) {
 | 
						|
		signal(signo, SIG_IGN);
 | 
						|
	}
 | 
						|
	sigmode[signo] = S_HARD_IGN;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
#ifdef mkinit
 | 
						|
INCLUDE <signal.h>
 | 
						|
INCLUDE "trap.h"
 | 
						|
 | 
						|
SHELLPROC {
 | 
						|
	char *sm;
 | 
						|
 | 
						|
	clear_traps();
 | 
						|
	for (sm = sigmode ; sm < sigmode + _NSIG ; sm++) {
 | 
						|
		if (*sm == S_IGN)
 | 
						|
			*sm = S_HARD_IGN;
 | 
						|
	}
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Signal handler.
 | 
						|
 */
 | 
						|
static void
 | 
						|
onsig(int signo)
 | 
						|
{
 | 
						|
 | 
						|
#ifndef BSD
 | 
						|
	signal(signo, onsig);
 | 
						|
#endif
 | 
						|
	if (signo == SIGINT && trap[SIGINT] == NULL) {
 | 
						|
		onint();
 | 
						|
		return;
 | 
						|
	}
 | 
						|
 | 
						|
	if (signo != SIGCHLD || !ignore_sigchld)
 | 
						|
		gotsig[signo] = 1;
 | 
						|
	pendingsigs++;
 | 
						|
 | 
						|
	/* If we are currently in a wait builtin, prepare to break it */
 | 
						|
	if ((signo == SIGINT || signo == SIGQUIT) && in_waitcmd != 0)
 | 
						|
		breakwaitcmd = 1;
 | 
						|
	/* 
 | 
						|
	 * If a trap is set, not ignored and not the null command, we need 
 | 
						|
	 * to make sure traps are executed even when a child blocks signals.
 | 
						|
	 */
 | 
						|
	if (Tflag &&
 | 
						|
	    trap[signo] != NULL && 
 | 
						|
	    ! trap[signo][0] == '\0' &&
 | 
						|
	    ! (trap[signo][0] == ':' && trap[signo][1] == '\0'))
 | 
						|
		breakwaitcmd = 1;
 | 
						|
 | 
						|
#ifndef NO_HISTORY
 | 
						|
	if (signo == SIGWINCH)
 | 
						|
		gotwinch = 1;
 | 
						|
#endif
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Called to execute a trap.  Perhaps we should avoid entering new trap
 | 
						|
 * handlers while we are executing a trap handler.
 | 
						|
 */
 | 
						|
void
 | 
						|
dotrap(void)
 | 
						|
{
 | 
						|
	int i;
 | 
						|
	int savestatus;
 | 
						|
 | 
						|
	in_dotrap++;
 | 
						|
	for (;;) {
 | 
						|
		for (i = 1; i < _NSIG; i++) {
 | 
						|
			if (gotsig[i]) {
 | 
						|
				gotsig[i] = 0;
 | 
						|
				if (trap[i]) {
 | 
						|
					/*
 | 
						|
					 * Ignore SIGCHLD to avoid infinite
 | 
						|
					 * recursion if the trap action does
 | 
						|
					 * a fork.
 | 
						|
					 */
 | 
						|
					if (i == SIGCHLD)
 | 
						|
						ignore_sigchld++;
 | 
						|
					savestatus = exitstatus;
 | 
						|
					evalstring(trap[i]);
 | 
						|
					exitstatus = savestatus;
 | 
						|
					if (i == SIGCHLD)
 | 
						|
						ignore_sigchld--;
 | 
						|
				}
 | 
						|
				break;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if (i >= _NSIG)
 | 
						|
			break;
 | 
						|
	}
 | 
						|
	in_dotrap--;
 | 
						|
	pendingsigs = 0;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Controls whether the shell is interactive or not.
 | 
						|
 */
 | 
						|
void
 | 
						|
setinteractive(int on)
 | 
						|
{
 | 
						|
	if (on == is_interactive)
 | 
						|
		return;
 | 
						|
	setsignal(SIGINT);
 | 
						|
	setsignal(SIGQUIT);
 | 
						|
	setsignal(SIGTERM);
 | 
						|
#ifndef NO_HISTORY
 | 
						|
	setsignal(SIGWINCH);
 | 
						|
#endif
 | 
						|
	is_interactive = on;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * Called to exit the shell.
 | 
						|
 */
 | 
						|
void
 | 
						|
exitshell(int status)
 | 
						|
{
 | 
						|
	struct jmploc loc1, loc2;
 | 
						|
	char *p;
 | 
						|
 | 
						|
	TRACE(("exitshell(%d) pid=%d\n", status, getpid()));
 | 
						|
	if (setjmp(loc1.loc)) {
 | 
						|
		goto l1;
 | 
						|
	}
 | 
						|
	if (setjmp(loc2.loc)) {
 | 
						|
		goto l2;
 | 
						|
	}
 | 
						|
	handler = &loc1;
 | 
						|
	if ((p = trap[0]) != NULL && *p != '\0') {
 | 
						|
		trap[0] = NULL;
 | 
						|
		evalstring(p);
 | 
						|
	}
 | 
						|
l1:   handler = &loc2;			/* probably unnecessary */
 | 
						|
	flushall();
 | 
						|
#if JOBS
 | 
						|
	setjobctl(0);
 | 
						|
#endif
 | 
						|
l2:   _exit(status);
 | 
						|
}
 | 
						|
 | 
						|
#ifdef NO_SIGINTERRUPT
 | 
						|
static int siginterrupt(sig, flag)
 | 
						|
int sig;
 | 
						|
int flag;
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef NO_SYS_SIGNAME
 | 
						|
static char *strsigname(sig)
 | 
						|
int sig;
 | 
						|
{
 | 
						|
	switch(sig)
 | 
						|
	{
 | 
						|
	case 0:		return "Signal 0";	/*  0 */
 | 
						|
	case SIGHUP:	return "hup";		/*  1 */
 | 
						|
	case SIGINT:	return "int";		/*  2 */
 | 
						|
	case SIGQUIT:	return "quit";		/*  3 */
 | 
						|
	case SIGILL:	return "ill";		/*  4 */
 | 
						|
	case SIGTRAP:	return "trap";		/*  5 */
 | 
						|
	case SIGABRT:	return "abrt";		/*  6 */
 | 
						|
#ifdef __minix_vmd
 | 
						|
	case SIGEMT:	return "emt";		/*  7 */
 | 
						|
#else
 | 
						|
	case SIGBUS:	return "bus";		/*  7 */
 | 
						|
#endif
 | 
						|
	case SIGFPE:	return "fpe";		/*  8 */
 | 
						|
	case SIGKILL:	return "kill";		/*  9 */
 | 
						|
	case SIGUSR1:	return "usr1";		/* 10 */
 | 
						|
	case SIGSEGV:	return "segv";		/* 11 */
 | 
						|
	case SIGUSR2:	return "usr2";		/* 12 */
 | 
						|
	case SIGPIPE:	return "pipe";		/* 13 */
 | 
						|
	case SIGALRM:	return "alrm";		/* 14 */
 | 
						|
	case SIGTERM:	return "term";		/* 15 */
 | 
						|
#ifdef __minix_vmd
 | 
						|
	case 16:	return "Signal 16";	/* 16 */
 | 
						|
#else
 | 
						|
	case SIGEMT:	return "emt";		/* 16 */
 | 
						|
#endif
 | 
						|
	case SIGCHLD:	return "chld";		/* 17 */
 | 
						|
	case SIGCONT:	return "cont";		/* 18 */
 | 
						|
	case SIGSTOP:	return "stop";		/* 19 */
 | 
						|
	case SIGTSTP:	return "tstp";		/* 20 */
 | 
						|
	case SIGTTIN:	return "ttin";		/* 21 */
 | 
						|
	case SIGTTOU:	return "ttou";		/* 22 */
 | 
						|
	case SIGWINCH:	return "winch";		/* 23 */
 | 
						|
	case SIGVTALRM:	return "vtalrm";	/* 24 */
 | 
						|
	case SIGPROF:	return "prof";		/* 25 */
 | 
						|
#ifdef __minix_vmd
 | 
						|
	case SIGFPEMU:	return "fpemu";		/* 30 */
 | 
						|
#endif
 | 
						|
	default:	return "Signal n";
 | 
						|
	}
 | 
						|
}
 | 
						|
#else
 | 
						|
static char *strsigname(sig)
 | 
						|
int sig;
 | 
						|
{
 | 
						|
	return sys_signame[sig];
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
#ifdef NO_SYS_SIGLIST
 | 
						|
#include "signames.h"
 | 
						|
char *strsiglist(sig)
 | 
						|
int sig;
 | 
						|
{
 | 
						|
	if (sig > MAXSIG)
 | 
						|
		return NULL;
 | 
						|
	return sigmesg[sig];
 | 
						|
}
 | 
						|
#else
 | 
						|
char *strsiglist(sig)
 | 
						|
int sig;
 | 
						|
{
 | 
						|
	return sys_siglist[sig];
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
/*
 | 
						|
 * $PchId: trap.c,v 1.7 2006/05/23 11:56:21 philip Exp $
 | 
						|
 */
 |