mirror of https://github.com/lianthony/NT4.0
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
568 lines
14 KiB
568 lines
14 KiB
/***
|
|
*winxfltr.c - startup exception filter
|
|
*
|
|
* Copyright (c) 1990-1992, Microsoft Corporation. All rights reserved
|
|
*
|
|
*Purpose:
|
|
* Defines _XcptFilter(), the function called by the exception filter
|
|
* expression in the startup code.
|
|
*
|
|
*Revision History:
|
|
* 10-31-91 GJF Module created. Copied from the original xcptfltr.c
|
|
* then extensively revised.
|
|
* 11-08-91 GJF Cleaned up header files usage.
|
|
* 12-13-91 GJF Fixed multi-thread build.
|
|
*
|
|
*******************************************************************************/
|
|
|
|
#include <cruntime.h>
|
|
#include <excpt.h>
|
|
#include <signal.h>
|
|
#include <stddef.h>
|
|
|
|
|
|
/*
|
|
* special code denoting no signal.
|
|
*/
|
|
#define NOSIG -1
|
|
|
|
|
|
struct _XCPT_ACTION _VARTYPE1 _XcptActTab[] = {
|
|
|
|
/*
|
|
* Exceptions corresponding to the same signal (e.g., SIGFPE) must be grouped
|
|
* together.
|
|
*
|
|
* XcptNum SigNum XcptAction
|
|
* -------------------------------------------------------------------
|
|
*/
|
|
{ (unsigned long)0, SIGABRT, SIG_DFL },
|
|
|
|
{ (unsigned long)0, SIGTERM, SIG_DFL },
|
|
|
|
{ (unsigned long)0, SIGINT, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_DATATYPE_MISALIGNMENT*/, NOSIG, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_ACCESS_VIOLATION*/, SIGSEGV, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_ILLEGAL_INSTRUCTION*/, SIGILL, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_PRIVILEGED_INSTRUCTION*/, SIGILL, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_NONCONTINUABLE_EXCEPTION*/, NOSIG, SIG_DIE },
|
|
|
|
{ (unsigned long)0/*STATUS_INVALID_DISPOSITION*/, NOSIG, SIG_DIE },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_DENORMAL_OPERAND*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_DIVIDE_BY_ZERO*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_INEXACT_RESULT*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_INVALID_OPERATION*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_OVERFLOW*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_STACK_CHECK*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_FLOAT_UNDERFLOW*/, SIGFPE, SIG_DFL },
|
|
|
|
{ (unsigned long)0/*STATUS_INTEGER_DIVIDE_BY_ZERO*/, NOSIG, SIG_DIE },
|
|
|
|
{ (unsigned long)0/*STATUS_STACK_OVERFLOW*/, NOSIG, SIG_DIE }
|
|
|
|
};
|
|
|
|
/*
|
|
* size of the exception-action table (in bytes)
|
|
*/
|
|
int _VARTYPE1 _XcptActTabSize = sizeof _XcptActTab;
|
|
|
|
/*
|
|
* number of entries in the exception-action table
|
|
*/
|
|
int _VARTYPE1 _XcptActTabCount = (sizeof _XcptActTab)/sizeof(_XcptActTab[0]);
|
|
|
|
|
|
/*
|
|
* function to look up the exception action table (_XcptActTab[]) corresponding
|
|
* to the given exception
|
|
*/
|
|
|
|
|
|
static struct _XCPT_ACTION * _CALLTYPE4 xcptlookup(
|
|
unsigned long
|
|
);
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
/*
|
|
* prototypes for debugging routines
|
|
*/
|
|
void prXcptActTabEntry(struct _XCPT_ACTION *);
|
|
void prXcptActTab(void);
|
|
|
|
#endif /* DEBUG */
|
|
|
|
#ifdef _WIN32
|
|
/***
|
|
*int _XcptFilter(xcptnum, pxcptptrs) - Identify exception and the action to
|
|
* be taken with it
|
|
*
|
|
*Purpose:
|
|
* _XcptFilter() is called by the exception filter expression of the
|
|
* _try - _except statement, in the startup code, which guards the call
|
|
* to the user's main(). _XcptFilter() consults the _XcptActTab[] table
|
|
* to identify the exception and determine its disposition. The
|
|
* is disposition of an exception corresponding to a C signal may be
|
|
* modified by a call to signal(). There are three broad cases:
|
|
*
|
|
* (1) Unrecognized exceptions and exceptions for which the XcptAction
|
|
* value is SIG_DFL.
|
|
*
|
|
* In both of these cases, EXCEPTION_CONTINUE_SEARCH is returned to
|
|
* cause the OS exception dispatcher to pass the exception onto the
|
|
* next exception handler in the chain (usually a system default
|
|
* handler).
|
|
*
|
|
* (2) Exceptions corresponding to C signals with an XcptAction value
|
|
* NOT equal to SIG_DFL.
|
|
*
|
|
* These are the C signals whose disposition has been affected by a
|
|
* call to signal() or whose default semantics differ slightly from
|
|
* from the corresponding OS exception. In all cases, the appropriate
|
|
* disposition of the C signal is made by the function (e.g., calling
|
|
* a user-specified signal handler). Then, EXCEPTION_CONTINUE_EXECU-
|
|
* TION is returned to cause the OS exception dispatcher to dismiss
|
|
* the exception and resume execution at the point where the
|
|
* exception occurred.
|
|
*
|
|
* (3) Exceptions for which the XcptAction value is SIG_DIE.
|
|
*
|
|
* These are the exceptions corresponding to fatal C runtime errors.
|
|
* _XCPT_HANDLE is returned to cause control to pass into the
|
|
* _except-block of the _try - _except statement. There, the runtime
|
|
* error is identified, an appropriate error message is printed out
|
|
* and the program is terminated.
|
|
*
|
|
*Entry:
|
|
*
|
|
*Exit:
|
|
*
|
|
*Exceptions:
|
|
* That's what it's all about!
|
|
*
|
|
*******************************************************************************/
|
|
|
|
int _CALLTYPE1 _XcptFilter (
|
|
unsigned long xcptnum,
|
|
PEXCEPTION_POINTERS pxcptinfoptrs
|
|
)
|
|
{
|
|
struct _XCPT_ACTION * pxcptact;
|
|
_PHNDLR phandler;
|
|
void *oldpxcptinfoptrs;
|
|
int oldfpecode;
|
|
|
|
/*
|
|
* first, determine if this exception needs to be passed on to the next
|
|
* higher exception handler. note that this includes any exception that
|
|
* is not recognized (i.e., is not in the table)
|
|
*/
|
|
if ( ((pxcptact = xcptlookup(xcptnum)) == NULL) ||
|
|
(pxcptact->XcptAction == SIG_DFL) )
|
|
/*
|
|
* pass the buck to the next level exception handler
|
|
*/
|
|
return(EXCEPTION_CONTINUE_SEARCH);
|
|
|
|
#ifdef DEBUG
|
|
prXcptActTabEntry(pxcptact);
|
|
#endif /* DEBUG */
|
|
|
|
/*
|
|
* next, weed out all of the exceptions that need to be handled by
|
|
* dying, perhaps with a runtime error message
|
|
*/
|
|
if ( pxcptact->XcptAction == SIG_DIE ) {
|
|
/*
|
|
* reset XcptAction (in case of recursion) and drop into the
|
|
* except-clause.
|
|
*/
|
|
pxcptact->XcptAction = SIG_DFL;
|
|
return(EXCEPTION_EXECUTE_HANDLER);
|
|
}
|
|
|
|
/*
|
|
* next, weed out all of the exceptions that are simply ignored
|
|
*/
|
|
if ( pxcptact->XcptAction == SIG_IGN )
|
|
/*
|
|
* resume execution
|
|
*/
|
|
return(EXCEPTION_CONTINUE_EXECUTION);
|
|
|
|
/*
|
|
* the remaining exceptions all correspond to C signals which have
|
|
* signal handlers associated with them. for some, special setup
|
|
* is required before the signal handler is called. in all cases,
|
|
* if the signal handler returns, -1 is returned by this function
|
|
* to resume execution at the point where the exception occurred.
|
|
*
|
|
* before calling the signal handler, the XcptAction field of the
|
|
* exception-action table entry must be reset.
|
|
*/
|
|
phandler = pxcptact->XcptAction;
|
|
|
|
/*
|
|
* reset the action to be the default
|
|
*/
|
|
pxcptact->XcptAction = SIG_DFL;
|
|
|
|
/*
|
|
* save the old value of _pxcptinfoptrs (in case this is a nested
|
|
* exception/signal) and store the current one.
|
|
*/
|
|
oldpxcptinfoptrs = _pxcptinfoptrs;
|
|
_pxcptinfoptrs = pxcptinfoptrs;
|
|
|
|
/*
|
|
* call the user-supplied signal handler
|
|
*
|
|
* floating point exceptions must be handled specially since, from
|
|
* the C point-of-view, there is only one signal. the exact identity
|
|
* of the exception is passed in the global variable _fpecode.
|
|
*/
|
|
if ( pxcptact->SigNum == SIGFPE ) {
|
|
/*
|
|
* Save the current _fpecode in case it is a nested floating
|
|
* point exception (not clear that we need to support this,
|
|
* but it's easy).
|
|
*/
|
|
oldfpecode = _fpecode;
|
|
|
|
/*
|
|
* there are no exceptions corresponding to
|
|
* following _FPE_xxx codes:
|
|
*
|
|
* _FPE_UNEMULATED
|
|
* _FPE_SQRTNEG
|
|
*
|
|
* futhermore, STATUS_FLOATING_STACK_CHECK is
|
|
* raised for both floating point stack under-
|
|
* flow and overflow. thus, the exception does
|
|
* not distinguish between _FPE_STACKOVERLOW
|
|
* and _FPE_STACKUNDERFLOW. arbitrarily, _fpecode
|
|
* is set to the former value.
|
|
*
|
|
* the following should be a switch statement but, alas, the
|
|
* compiler doesn't like switching on unsigned longs...
|
|
*/
|
|
if ( pxcptact->XcptNum == STATUS_FLOAT_DIVIDE_BY_ZERO )
|
|
_fpecode = _FPE_ZERODIVIDE;
|
|
else if ( pxcptact->XcptNum == STATUS_FLOAT_INVALID_OPERATION )
|
|
_fpecode = _FPE_INVALID;
|
|
else if ( pxcptact->XcptNum == STATUS_FLOAT_OVERFLOW )
|
|
_fpecode = _FPE_OVERFLOW;
|
|
else if ( pxcptact->XcptNum == STATUS_FLOAT_UNDERFLOW )
|
|
_fpecode = _FPE_UNDERFLOW;
|
|
else if ( pxcptact->XcptNum == STATUS_FLOAT_DENORMAL_OPERAND )
|
|
|
|
#ifdef _MT
|
|
(*pptd)->_tfpecode = _FPE_DENORMAL;
|
|
#else /* not _MT */
|
|
_fpecode = _FPE_DENORMAL;
|
|
#endif /* _MT */
|
|
|
|
else if ( pxcptact->XcptNum == STATUS_FLOAT_INEXACT_RESULT )
|
|
|
|
#ifdef _MT
|
|
(*pptd)->_tfpecode = _FPE_INEXACT;
|
|
#else /* not _MT */
|
|
_fpecode = _FPE_INEXACT;
|
|
#endif /* _MT */
|
|
|
|
else if ( pxcptact->XcptNum == STATUS_FLOAT_STACK_CHECK )
|
|
|
|
#ifdef _MT
|
|
(*pptd)->_tfpecode = _FPE_STACKOVERFLOW;
|
|
#else /* not _MT */
|
|
_fpecode = _FPE_STACKOVERFLOW;
|
|
#endif /* _MT */
|
|
|
|
/*
|
|
* call the SIGFPE handler. note the special code to support
|
|
* old MS-C programs whose SIGFPE handlers expect two args.
|
|
*
|
|
* NOTE: THE CAST AND CALL BELOW DEPEND ON _CALLTYPE1 BEING
|
|
* CALLER CLEANUP!
|
|
*/
|
|
#ifdef _MT
|
|
(*(void (* _CALLTYPE1)(int, int))phandler)(SIGFPE,
|
|
(*pptd)->_tfpecode);
|
|
#else /* _MT */
|
|
(*(void (* _CALLTYPE1)(int, int))phandler)(SIGFPE, _fpecode);
|
|
#endif
|
|
/*
|
|
* restore the old value of _fpecode
|
|
*/
|
|
#ifdef _MT
|
|
(*pptd)->_tfpecode = oldfpecode;
|
|
#else /* not _MT */
|
|
_fpecode = oldfpecode;
|
|
#endif /* _MT */
|
|
}
|
|
else
|
|
(*phandler)(pxcptact->SigNum);
|
|
|
|
/*
|
|
* restore the old value of _pxcptinfoptrs
|
|
*/
|
|
#ifdef _MT
|
|
(*pptd)->_tpxcptinfoptrs = oldpxcptinfoptrs;
|
|
#else /* not _MT */
|
|
_pxcptinfoptrs = oldpxcptinfoptrs;
|
|
#endif /* _MT */
|
|
|
|
return(EXCEPTION_CONTINUE_EXECUTION);
|
|
|
|
}
|
|
|
|
|
|
/***
|
|
*struct _XCPT_ACTION * xcptlookup(xcptnum, pxcptrec) - look up exception-action
|
|
* table entry for xcptnum
|
|
*
|
|
*Purpose:
|
|
* Find the in _XcptActTab[] whose Xcptnum field is xcptnum.
|
|
*
|
|
*Entry:
|
|
* unsigned long xcptnum - exception type
|
|
*
|
|
* _PEXCEPTIONREPORTRECORD pxcptrec - pointer to exception report record
|
|
* (used only to distinguish different types of XCPT_SIGNAL)
|
|
*
|
|
*Exit:
|
|
* If successful, pointer to the table entry. If no such entry, NULL is
|
|
* returned.
|
|
*
|
|
*Exceptions:
|
|
*
|
|
*******************************************************************************/
|
|
|
|
#ifdef _MT
|
|
|
|
static struct _XCPT_ACTION * _CALLTYPE4 xcptlookup (
|
|
unsigned long xcptnum,
|
|
struct _XCPT_ACTION * pxcptacttab
|
|
)
|
|
|
|
#else /* not _MT */
|
|
|
|
static struct _XCPT_ACTION * _CALLTYPE4 xcptlookup (
|
|
unsigned long xcptnum
|
|
)
|
|
|
|
#endif /* _MT */
|
|
|
|
{
|
|
#ifdef _MT
|
|
struct _XCPT_ACTION *pxcptact = pxcptacttab;
|
|
#else /* ndef _MT */
|
|
struct _XCPT_ACTION *pxcptact = _XcptActTab;
|
|
#endif /* _MT */
|
|
|
|
/*
|
|
* walk thru the _xcptactab table looking for the proper entry
|
|
*/
|
|
#ifdef _MT
|
|
|
|
while ( (pxcptact->XcptNum != xcptnum) && (++pxcptact <
|
|
pxcptacttab + _XcptActTabCount) ) ;
|
|
|
|
#else /* not _MT */
|
|
|
|
while ( (pxcptact->XcptNum != xcptnum) && (++pxcptact <
|
|
_XcptActTab + _XcptActTabCount) ) ;
|
|
|
|
#endif /* _MT */
|
|
|
|
/*
|
|
* if no table entry was found corresponding to xcptnum, return NULL
|
|
*/
|
|
if ( pxcptact->XcptNum != xcptnum )
|
|
return(NULL);
|
|
|
|
return(pxcptact);
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
|
|
/*
|
|
* DEBUGGING TOOLS!
|
|
*/
|
|
struct xcptnumstr {
|
|
unsigned long num;
|
|
char *str;
|
|
};
|
|
|
|
struct xcptnumstr XcptNumStr[] = {
|
|
|
|
{ (unsigned long)STATUS_DATATYPE_MISALIGNMENT,
|
|
"STATUS_DATATYPE_MISALIGNMENT" },
|
|
|
|
{ (unsigned long)STATUS_ACCESS_VIOLATION,
|
|
"STATUS_ACCESS_VIOLATION" },
|
|
|
|
{ (unsigned long)STATUS_ILLEGAL_INSTRUCTION,
|
|
"STATUS_ILLEGAL_INSTRUCTION" },
|
|
|
|
{ (unsigned long)STATUS_NONCONTINUABLE_EXCEPTION,
|
|
"STATUS_NONCONTINUABLE_EXCEPTION" },
|
|
|
|
{ (unsigned long)STATUS_INVALID_DISPOSITION,
|
|
"STATUS_INVALID_DISPOSITION" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_DENORMAL_OPERAND,
|
|
"STATUS_FLOAT_DENORMAL_OPERAND" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_DIVIDE_BY_ZERO,
|
|
"STATUS_FLOAT_DIVIDE_BY_ZERO" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_INEXACT_RESULT,
|
|
"STATUS_FLOAT_INEXACT_RESULT" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_INVALID_OPERATION,
|
|
"STATUS_FLOAT_INVALID_OPERATION" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_OVERFLOW,
|
|
"STATUS_FLOAT_OVERFLOW" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_STACK_CHECK,
|
|
"STATUS_FLOAT_STACK_CHECK" },
|
|
|
|
{ (unsigned long)STATUS_FLOAT_UNDERFLOW,
|
|
"STATUS_FLOAT_UNDERFLOW" },
|
|
|
|
{ (unsigned long)STATUS_INTEGER_DIVIDE_BY_ZERO,
|
|
"STATUS_INTEGER_DIVIDE_BY_ZERO" },
|
|
|
|
{ (unsigned long)STATUS_PRIVILEGED_INSTRUCTION,
|
|
"STATUS_PRIVILEGED_INSTRUCTION" },
|
|
|
|
{ (unsigned long)_STATUS_STACK_OVERFLOW,
|
|
"_STATUS_STACK_OVERFLOW" }
|
|
};
|
|
|
|
#define XCPTNUMSTR_SZ ( sizeof XcptNumStr / sizeof XcptNumStr[0] )
|
|
|
|
/*
|
|
* return string mnemonic for exception
|
|
*/
|
|
char * XcptNumToStr (
|
|
unsigned long xcptnum
|
|
)
|
|
{
|
|
int indx;
|
|
|
|
for ( indx = 0 ; indx < XCPTNUMSTR_SZ ; indx++ )
|
|
if ( XcptNumStr[indx].num == xcptnum )
|
|
return(XcptNumStr[indx].str);
|
|
|
|
return(NULL);
|
|
}
|
|
|
|
struct signumstr {
|
|
int num;
|
|
char *str;
|
|
};
|
|
|
|
struct signumstr SigNumStr[] = {
|
|
{ SIGINT, "SIGINT" },
|
|
{ SIGILL, "SIGILL" },
|
|
{ SIGFPE, "SIGFPE" },
|
|
{ SIGSEGV, "SIGSEGV" },
|
|
{ SIGTERM, "SIGTERM" },
|
|
{ SIGBREAK, "SIGBREAK" },
|
|
{ SIGABRT, "SIGABRT" }
|
|
};
|
|
|
|
#define SIGNUMSTR_SZ ( sizeof SigNumStr / sizeof SigNumStr[0] )
|
|
|
|
/*
|
|
* return string mnemonic for signal
|
|
*/
|
|
char * SigNumToStr (
|
|
int signum
|
|
)
|
|
{
|
|
int indx;
|
|
|
|
for ( indx = 0 ; indx < SIGNUMSTR_SZ ; indx++ )
|
|
if ( SigNumStr[indx].num == signum )
|
|
return(SigNumStr[indx].str);
|
|
|
|
return(NULL);
|
|
}
|
|
|
|
struct actcodestr {
|
|
_PHNDLR code;
|
|
char *str;
|
|
};
|
|
|
|
struct actcodestr ActCodeStr[] = {
|
|
{ SIG_DFL, "SIG_DFL" },
|
|
{ SIG_IGN, "SIG_IGN" },
|
|
{ SIG_DIE, "SIG_DIE" }
|
|
};
|
|
|
|
#define ACTCODESTR_SZ ( sizeof ActCodeStr / sizeof ActCodeStr[0] )
|
|
|
|
/*
|
|
* return string mnemonic for action code
|
|
*/
|
|
char * ActCodeToStr (
|
|
_PHNDLR action
|
|
)
|
|
{
|
|
int indx;
|
|
|
|
for ( indx = 0 ; indx < ACTCODESTR_SZ ; indx++ )
|
|
if ( ActCodeStr[indx].code == action)
|
|
return(ActCodeStr[indx].str);
|
|
|
|
return("FUNCTION ADDRESS");
|
|
}
|
|
|
|
/*
|
|
* print out exception-action table entry
|
|
*/
|
|
void prXcptActTabEntry (
|
|
struct _XCPT_ACTION *pxcptact
|
|
)
|
|
{
|
|
printf("XcptNum = %s\n", XcptNumToStr(pxcptact->XcptNum));
|
|
printf("SigNum = %s\n", SigNumToStr(pxcptact->SigNum));
|
|
printf("XcptAction = %s\n", ActCodeToStr(pxcptact->XcptAction));
|
|
}
|
|
|
|
/*
|
|
* print out all entries in the exception-action table
|
|
*/
|
|
void prXcptActTab (
|
|
void
|
|
)
|
|
{
|
|
int indx;
|
|
|
|
for ( indx = 0 ; indx < _XcptActTabCount ; indx++ ) {
|
|
printf("\n_XcptActTab[%d] = \n", indx);
|
|
prXcptActTabEntry(&_XcptActTab[indx]);
|
|
}
|
|
}
|
|
|
|
#endif /* DEBUG */
|
|
|
|
#endif /* MAC */
|