ack/lang/cem/libcc/mon/signal.c

101 lines
2.3 KiB
C
Raw Normal View History

1987-03-31 10:45:53 +00:00
/* $Header$ */
1985-01-01 22:27:02 +00:00
#include <signal.h>
1993-11-17 16:53:43 +00:00
typedef void (*callvec)() ;
1985-01-01 22:27:02 +00:00
static callvec vector[16] = {
SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL,
SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL, SIG_DFL
} ;
static char mapvec[] = {
0, /* EARRAY */
0, /* ERANGE */
0, /* ESET */
0, /* EIOVFL */
SIGFPE, /* EFOVFL */
SIGFPE, /* EFUNDFL */
0, /* EIDIVZ */
SIGFPE, /* EFDIVZ */
0, /* EIUND, already ignored */
SIGFPE, /* EFUND */
0, /* ECONV */
0, /* 11 */
0, /* 12 */
0, /* 13 */
0, /* 14 */
0, /* 15 */
SIGSEGV, /* ESTACK */
SIGSEGV, /* EHEAP */
0, /* EILLINS */
0, /* EODDZ */
0, /* ECASE */
SIGSEGV, /* EBADMEM */
SIGBUS, /* EBADPTR */
0, /* EBADPC */
0, /* EBADLAE */
SIGSYS, /* EBADMON */
0, /* EBADLIN */
0, /* EBADGTO */
} ;
#define VECBASE 128
static firsttime = 1 ;
static int catchtrp() ;
static int procesig() ;
callvec signal(sig,func) int sig ; callvec func ; {
register index, i ;
callvec prev ;
index= sig-1 ;
if ( index<0 || index>=(sizeof vector/sizeof vector[0]) ) {
return (callvec) -1 ;
}
if ( firsttime ) {
firsttime= 0 ;
_setsig(catchtrp) ;
}
prev= vector[index] ;
if ( prev!=func ) {
register int mapval ;
vector[index]= func ;
if ( func==SIG_IGN ) {
mapval= -3;
} else if ( func==SIG_DFL ) {
mapval= -2;
} else {
mapval=VECBASE+sig;
}
if ( sigtrp(mapval,sig)== -1 ) return (callvec) -1;
}
return prev ;
}
static int catchtrp(trapno) int trapno ; {
if ( trapno>VECBASE &&
trapno<=VECBASE + (sizeof vector/sizeof vector[0]) ) {
return procesig(trapno-VECBASE) ;
}
if ( trapno>=0 && trapno< (sizeof mapvec/sizeof mapvec[0]) &&
mapvec[trapno] ) {
return procesig(mapvec[trapno]) ;
}
return 0 ; /* Failed to handle the trap */
}
static int procesig(sig) int sig ; {
register index ;
callvec trf ;
index= sig-1 ;
trf= vector[index] ;
if ( trf==SIG_IGN ) return 1 ;
if ( sig!=SIGILL && sig!=SIGTRAP ) vector[index]= SIG_IGN ;
if ( trf==SIG_DFL ) return 0 ;
(*trf)(sig) ;
return 1 ;
}