140266059SGregory Neil Shapiro /*
25dd76dd0SGregory Neil Shapiro * Copyright (c) 2000-2001 Proofpoint, Inc. and its suppliers.
340266059SGregory Neil Shapiro * All rights reserved.
440266059SGregory Neil Shapiro *
540266059SGregory Neil Shapiro * By using this file, you agree to the terms and conditions set
640266059SGregory Neil Shapiro * forth in the LICENSE file which can be found at the top level of
740266059SGregory Neil Shapiro * the sendmail distribution.
840266059SGregory Neil Shapiro *
940266059SGregory Neil Shapiro */
1040266059SGregory Neil Shapiro
1140266059SGregory Neil Shapiro #include <sm/gen.h>
124313cc83SGregory Neil Shapiro SM_RCSID("@(#)$Id: assert.c,v 1.27 2013-11-22 20:51:42 ca Exp $")
1340266059SGregory Neil Shapiro
1440266059SGregory Neil Shapiro /*
1540266059SGregory Neil Shapiro ** Abnormal program termination and assertion checking.
1640266059SGregory Neil Shapiro ** For documentation, see assert.html.
1740266059SGregory Neil Shapiro */
1840266059SGregory Neil Shapiro
1940266059SGregory Neil Shapiro #include <signal.h>
2040266059SGregory Neil Shapiro #include <stdlib.h>
2140266059SGregory Neil Shapiro #include <unistd.h>
2240266059SGregory Neil Shapiro
2340266059SGregory Neil Shapiro #include <sm/assert.h>
2440266059SGregory Neil Shapiro #include <sm/exc.h>
2540266059SGregory Neil Shapiro #include <sm/io.h>
2640266059SGregory Neil Shapiro #include <sm/varargs.h>
2740266059SGregory Neil Shapiro
2840266059SGregory Neil Shapiro /*
2940266059SGregory Neil Shapiro ** Debug categories that are used to guard expensive assertion checks.
3040266059SGregory Neil Shapiro */
3140266059SGregory Neil Shapiro
3240266059SGregory Neil Shapiro SM_DEBUG_T SmExpensiveAssert = SM_DEBUG_INITIALIZER("sm_check_assert",
3340266059SGregory Neil Shapiro "@(#)$Debug: sm_check_assert - check assertions $");
3440266059SGregory Neil Shapiro
3540266059SGregory Neil Shapiro SM_DEBUG_T SmExpensiveRequire = SM_DEBUG_INITIALIZER("sm_check_require",
3640266059SGregory Neil Shapiro "@(#)$Debug: sm_check_require - check function preconditions $");
3740266059SGregory Neil Shapiro
3840266059SGregory Neil Shapiro SM_DEBUG_T SmExpensiveEnsure = SM_DEBUG_INITIALIZER("sm_check_ensure",
3940266059SGregory Neil Shapiro "@(#)$Debug: sm_check_ensure - check function postconditions $");
4040266059SGregory Neil Shapiro
4140266059SGregory Neil Shapiro /*
4240266059SGregory Neil Shapiro ** Debug category: send self SIGSTOP on fatal error,
4340266059SGregory Neil Shapiro ** so that you can run a debugger on the stopped process.
4440266059SGregory Neil Shapiro */
4540266059SGregory Neil Shapiro
4640266059SGregory Neil Shapiro SM_DEBUG_T SmAbortStop = SM_DEBUG_INITIALIZER("sm_abort_stop",
4740266059SGregory Neil Shapiro "@(#)$Debug: sm_abort_stop - stop process on fatal error $");
4840266059SGregory Neil Shapiro
4940266059SGregory Neil Shapiro /*
5040266059SGregory Neil Shapiro ** SM_ABORT_DEFAULTHANDLER -- Default procedure for abnormal program
5140266059SGregory Neil Shapiro ** termination.
5240266059SGregory Neil Shapiro **
5340266059SGregory Neil Shapiro ** The goal is to display an error message without disturbing the
5440266059SGregory Neil Shapiro ** process state too much, then dump core.
5540266059SGregory Neil Shapiro **
5640266059SGregory Neil Shapiro ** Parameters:
5740266059SGregory Neil Shapiro ** filename -- filename (can be NULL).
5840266059SGregory Neil Shapiro ** lineno -- line number.
5940266059SGregory Neil Shapiro ** msg -- message.
6040266059SGregory Neil Shapiro **
6140266059SGregory Neil Shapiro ** Returns:
6240266059SGregory Neil Shapiro ** doesn't return.
6340266059SGregory Neil Shapiro */
6440266059SGregory Neil Shapiro
6540266059SGregory Neil Shapiro static void
6640266059SGregory Neil Shapiro sm_abort_defaulthandler __P((
6740266059SGregory Neil Shapiro const char *filename,
6840266059SGregory Neil Shapiro int lineno,
6940266059SGregory Neil Shapiro const char *msg));
7040266059SGregory Neil Shapiro
7140266059SGregory Neil Shapiro static void
sm_abort_defaulthandler(filename,lineno,msg)7240266059SGregory Neil Shapiro sm_abort_defaulthandler(filename, lineno, msg)
7340266059SGregory Neil Shapiro const char *filename;
7440266059SGregory Neil Shapiro int lineno;
7540266059SGregory Neil Shapiro const char *msg;
7640266059SGregory Neil Shapiro {
7740266059SGregory Neil Shapiro if (filename != NULL)
7840266059SGregory Neil Shapiro sm_io_fprintf(smioerr, SM_TIME_DEFAULT, "%s:%d: %s\n", filename,
7940266059SGregory Neil Shapiro lineno, msg);
8040266059SGregory Neil Shapiro else
8140266059SGregory Neil Shapiro sm_io_fprintf(smioerr, SM_TIME_DEFAULT, "%s\n", msg);
8240266059SGregory Neil Shapiro sm_io_flush(smioerr, SM_TIME_DEFAULT);
8340266059SGregory Neil Shapiro #ifdef SIGSTOP
8440266059SGregory Neil Shapiro if (sm_debug_active(&SmAbortStop, 1))
8540266059SGregory Neil Shapiro kill(getpid(), SIGSTOP);
86*5b0945b5SGregory Neil Shapiro #endif
8740266059SGregory Neil Shapiro abort();
8840266059SGregory Neil Shapiro }
8940266059SGregory Neil Shapiro
9040266059SGregory Neil Shapiro /*
9140266059SGregory Neil Shapiro ** This is the action to be taken to cause abnormal program termination.
9240266059SGregory Neil Shapiro */
9340266059SGregory Neil Shapiro
9440266059SGregory Neil Shapiro static SM_ABORT_HANDLER_T SmAbortHandler = sm_abort_defaulthandler;
9540266059SGregory Neil Shapiro
9640266059SGregory Neil Shapiro /*
9740266059SGregory Neil Shapiro ** SM_ABORT_SETHANDLER -- Set handler for SM_ABORT()
9840266059SGregory Neil Shapiro **
9940266059SGregory Neil Shapiro ** This allows you to set a handler function for causing abnormal
10040266059SGregory Neil Shapiro ** program termination; it is called when a logic bug is detected.
10140266059SGregory Neil Shapiro **
10240266059SGregory Neil Shapiro ** Parameters:
10340266059SGregory Neil Shapiro ** f -- handler.
10440266059SGregory Neil Shapiro **
10540266059SGregory Neil Shapiro ** Returns:
10640266059SGregory Neil Shapiro ** none.
10740266059SGregory Neil Shapiro */
10840266059SGregory Neil Shapiro
10940266059SGregory Neil Shapiro void
sm_abort_sethandler(f)11040266059SGregory Neil Shapiro sm_abort_sethandler(f)
11140266059SGregory Neil Shapiro SM_ABORT_HANDLER_T f;
11240266059SGregory Neil Shapiro {
11340266059SGregory Neil Shapiro if (f == NULL)
11440266059SGregory Neil Shapiro SmAbortHandler = sm_abort_defaulthandler;
11540266059SGregory Neil Shapiro else
11640266059SGregory Neil Shapiro SmAbortHandler = f;
11740266059SGregory Neil Shapiro }
11840266059SGregory Neil Shapiro
11940266059SGregory Neil Shapiro /*
12040266059SGregory Neil Shapiro ** SM_ABORT -- Call it when you have detected a logic bug.
12140266059SGregory Neil Shapiro **
12240266059SGregory Neil Shapiro ** Parameters:
12340266059SGregory Neil Shapiro ** fmt -- format string.
12440266059SGregory Neil Shapiro ** ... -- arguments.
12540266059SGregory Neil Shapiro **
12640266059SGregory Neil Shapiro ** Returns:
12740266059SGregory Neil Shapiro ** doesn't.
12840266059SGregory Neil Shapiro */
12940266059SGregory Neil Shapiro
130323f6dcbSGregory Neil Shapiro void SM_DEAD_D
13140266059SGregory Neil Shapiro #if SM_VA_STD
sm_abort(char * fmt,...)13240266059SGregory Neil Shapiro sm_abort(char *fmt, ...)
13340266059SGregory Neil Shapiro #else /* SM_VA_STD */
13440266059SGregory Neil Shapiro sm_abort(fmt, va_alist)
13540266059SGregory Neil Shapiro char *fmt;
13640266059SGregory Neil Shapiro va_dcl
13740266059SGregory Neil Shapiro #endif /* SM_VA_STD */
13840266059SGregory Neil Shapiro {
13940266059SGregory Neil Shapiro char msg[128];
14040266059SGregory Neil Shapiro SM_VA_LOCAL_DECL
14140266059SGregory Neil Shapiro
14240266059SGregory Neil Shapiro SM_VA_START(ap, fmt);
14340266059SGregory Neil Shapiro sm_vsnprintf(msg, sizeof msg, fmt, ap);
14440266059SGregory Neil Shapiro SM_VA_END(ap);
14540266059SGregory Neil Shapiro sm_abort_at(NULL, 0, msg);
14640266059SGregory Neil Shapiro }
14740266059SGregory Neil Shapiro
14840266059SGregory Neil Shapiro /*
14940266059SGregory Neil Shapiro ** SM_ABORT_AT -- Initiate abnormal program termination.
15040266059SGregory Neil Shapiro **
15140266059SGregory Neil Shapiro ** This is the low level function that is called to initiate abnormal
15240266059SGregory Neil Shapiro ** program termination. It prints an error message and terminates the
15340266059SGregory Neil Shapiro ** program. It is called by sm_abort and by the assertion macros.
15440266059SGregory Neil Shapiro ** If filename != NULL then filename and lineno specify the line of source
15540266059SGregory Neil Shapiro ** code at which the bug was detected.
15640266059SGregory Neil Shapiro **
15740266059SGregory Neil Shapiro ** Parameters:
15840266059SGregory Neil Shapiro ** filename -- filename (can be NULL).
15940266059SGregory Neil Shapiro ** lineno -- line number.
16040266059SGregory Neil Shapiro ** msg -- message.
16140266059SGregory Neil Shapiro **
16240266059SGregory Neil Shapiro ** Returns:
16340266059SGregory Neil Shapiro ** doesn't.
16440266059SGregory Neil Shapiro */
16540266059SGregory Neil Shapiro
166323f6dcbSGregory Neil Shapiro void SM_DEAD_D
sm_abort_at(filename,lineno,msg)16740266059SGregory Neil Shapiro sm_abort_at(filename, lineno, msg)
16840266059SGregory Neil Shapiro const char *filename;
16940266059SGregory Neil Shapiro int lineno;
17040266059SGregory Neil Shapiro const char *msg;
17140266059SGregory Neil Shapiro {
17240266059SGregory Neil Shapiro SM_TRY
17340266059SGregory Neil Shapiro (*SmAbortHandler)(filename, lineno, msg);
17440266059SGregory Neil Shapiro SM_EXCEPT(exc, "*")
17540266059SGregory Neil Shapiro sm_io_fprintf(smioerr, SM_TIME_DEFAULT,
17640266059SGregory Neil Shapiro "exception raised by abort handler:\n");
17740266059SGregory Neil Shapiro sm_exc_print(exc, smioerr);
17840266059SGregory Neil Shapiro sm_io_flush(smioerr, SM_TIME_DEFAULT);
17940266059SGregory Neil Shapiro SM_END_TRY
18040266059SGregory Neil Shapiro
18140266059SGregory Neil Shapiro /*
18240266059SGregory Neil Shapiro ** SmAbortHandler isn't supposed to return.
18340266059SGregory Neil Shapiro ** Since it has, let's make sure that the program is terminated.
18440266059SGregory Neil Shapiro */
18540266059SGregory Neil Shapiro
18640266059SGregory Neil Shapiro abort();
18740266059SGregory Neil Shapiro }
188