xref: /freebsd/contrib/sendmail/libsm/assert.c (revision e2c0e292e8a7ca00ba99bcfccc9e637f45c3e8b1)
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