xref: /minix3/external/bsd/bind/dist/lib/isc/assertions.c (revision 00b67f09dd46474d133c95011a48590a8e8f94c7)
1*00b67f09SDavid van Moolenbroek /*	$NetBSD: assertions.c,v 1.6 2014/12/10 04:37:59 christos Exp $	*/
2*00b67f09SDavid van Moolenbroek 
3*00b67f09SDavid van Moolenbroek /*
4*00b67f09SDavid van Moolenbroek  * Copyright (C) 2004, 2005, 2007-2009  Internet Systems Consortium, Inc. ("ISC")
5*00b67f09SDavid van Moolenbroek  * Copyright (C) 1997-2001  Internet Software Consortium.
6*00b67f09SDavid van Moolenbroek  *
7*00b67f09SDavid van Moolenbroek  * Permission to use, copy, modify, and/or distribute this software for any
8*00b67f09SDavid van Moolenbroek  * purpose with or without fee is hereby granted, provided that the above
9*00b67f09SDavid van Moolenbroek  * copyright notice and this permission notice appear in all copies.
10*00b67f09SDavid van Moolenbroek  *
11*00b67f09SDavid van Moolenbroek  * THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
12*00b67f09SDavid van Moolenbroek  * REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
13*00b67f09SDavid van Moolenbroek  * AND FITNESS.  IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
14*00b67f09SDavid van Moolenbroek  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
15*00b67f09SDavid van Moolenbroek  * LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
16*00b67f09SDavid van Moolenbroek  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
17*00b67f09SDavid van Moolenbroek  * PERFORMANCE OF THIS SOFTWARE.
18*00b67f09SDavid van Moolenbroek  */
19*00b67f09SDavid van Moolenbroek 
20*00b67f09SDavid van Moolenbroek /* Id: assertions.c,v 1.26 2009/09/29 15:06:07 fdupont Exp  */
21*00b67f09SDavid van Moolenbroek 
22*00b67f09SDavid van Moolenbroek /*! \file */
23*00b67f09SDavid van Moolenbroek 
24*00b67f09SDavid van Moolenbroek #include <config.h>
25*00b67f09SDavid van Moolenbroek 
26*00b67f09SDavid van Moolenbroek #include <stdio.h>
27*00b67f09SDavid van Moolenbroek #include <stdlib.h>
28*00b67f09SDavid van Moolenbroek 
29*00b67f09SDavid van Moolenbroek #include <isc/assertions.h>
30*00b67f09SDavid van Moolenbroek #include <isc/backtrace.h>
31*00b67f09SDavid van Moolenbroek #include <isc/msgs.h>
32*00b67f09SDavid van Moolenbroek #include <isc/result.h>
33*00b67f09SDavid van Moolenbroek 
34*00b67f09SDavid van Moolenbroek /*
35*00b67f09SDavid van Moolenbroek  * The maximum number of stack frames to dump on assertion failure.
36*00b67f09SDavid van Moolenbroek  */
37*00b67f09SDavid van Moolenbroek #ifndef BACKTRACE_MAXFRAME
38*00b67f09SDavid van Moolenbroek #define BACKTRACE_MAXFRAME 128
39*00b67f09SDavid van Moolenbroek #endif
40*00b67f09SDavid van Moolenbroek 
41*00b67f09SDavid van Moolenbroek /*%
42*00b67f09SDavid van Moolenbroek  * Forward.
43*00b67f09SDavid van Moolenbroek  */
44*00b67f09SDavid van Moolenbroek static void
45*00b67f09SDavid van Moolenbroek default_callback(const char *, int, isc_assertiontype_t, const char *);
46*00b67f09SDavid van Moolenbroek 
47*00b67f09SDavid van Moolenbroek static isc_assertioncallback_t isc_assertion_failed_cb = default_callback;
48*00b67f09SDavid van Moolenbroek 
49*00b67f09SDavid van Moolenbroek /*%
50*00b67f09SDavid van Moolenbroek  * Public.
51*00b67f09SDavid van Moolenbroek  */
52*00b67f09SDavid van Moolenbroek 
53*00b67f09SDavid van Moolenbroek /*% assertion failed handler */
54*00b67f09SDavid van Moolenbroek /* coverity[+kill] */
55*00b67f09SDavid van Moolenbroek void
isc_assertion_failed(const char * file,int line,isc_assertiontype_t type,const char * cond)56*00b67f09SDavid van Moolenbroek isc_assertion_failed(const char *file, int line, isc_assertiontype_t type,
57*00b67f09SDavid van Moolenbroek 		     const char *cond)
58*00b67f09SDavid van Moolenbroek {
59*00b67f09SDavid van Moolenbroek 	isc_assertion_failed_cb(file, line, type, cond);
60*00b67f09SDavid van Moolenbroek 	abort();
61*00b67f09SDavid van Moolenbroek 	/* NOTREACHED */
62*00b67f09SDavid van Moolenbroek }
63*00b67f09SDavid van Moolenbroek 
64*00b67f09SDavid van Moolenbroek /*% Set callback. */
65*00b67f09SDavid van Moolenbroek void
isc_assertion_setcallback(isc_assertioncallback_t cb)66*00b67f09SDavid van Moolenbroek isc_assertion_setcallback(isc_assertioncallback_t cb) {
67*00b67f09SDavid van Moolenbroek 	if (cb == NULL)
68*00b67f09SDavid van Moolenbroek 		isc_assertion_failed_cb = default_callback;
69*00b67f09SDavid van Moolenbroek 	else
70*00b67f09SDavid van Moolenbroek 		isc_assertion_failed_cb = cb;
71*00b67f09SDavid van Moolenbroek }
72*00b67f09SDavid van Moolenbroek 
73*00b67f09SDavid van Moolenbroek /*% Type to Text */
74*00b67f09SDavid van Moolenbroek const char *
isc_assertion_typetotext(isc_assertiontype_t type)75*00b67f09SDavid van Moolenbroek isc_assertion_typetotext(isc_assertiontype_t type) {
76*00b67f09SDavid van Moolenbroek 	const char *result;
77*00b67f09SDavid van Moolenbroek 
78*00b67f09SDavid van Moolenbroek 	/*
79*00b67f09SDavid van Moolenbroek 	 * These strings have purposefully not been internationalized
80*00b67f09SDavid van Moolenbroek 	 * because they are considered to essentially be keywords of
81*00b67f09SDavid van Moolenbroek 	 * the ISC development environment.
82*00b67f09SDavid van Moolenbroek 	 */
83*00b67f09SDavid van Moolenbroek 	switch (type) {
84*00b67f09SDavid van Moolenbroek 	case isc_assertiontype_require:
85*00b67f09SDavid van Moolenbroek 		result = "REQUIRE";
86*00b67f09SDavid van Moolenbroek 		break;
87*00b67f09SDavid van Moolenbroek 	case isc_assertiontype_ensure:
88*00b67f09SDavid van Moolenbroek 		result = "ENSURE";
89*00b67f09SDavid van Moolenbroek 		break;
90*00b67f09SDavid van Moolenbroek 	case isc_assertiontype_insist:
91*00b67f09SDavid van Moolenbroek 		result = "INSIST";
92*00b67f09SDavid van Moolenbroek 		break;
93*00b67f09SDavid van Moolenbroek 	case isc_assertiontype_invariant:
94*00b67f09SDavid van Moolenbroek 		result = "INVARIANT";
95*00b67f09SDavid van Moolenbroek 		break;
96*00b67f09SDavid van Moolenbroek 	default:
97*00b67f09SDavid van Moolenbroek 		result = NULL;
98*00b67f09SDavid van Moolenbroek 	}
99*00b67f09SDavid van Moolenbroek 	return (result);
100*00b67f09SDavid van Moolenbroek }
101*00b67f09SDavid van Moolenbroek 
102*00b67f09SDavid van Moolenbroek /*
103*00b67f09SDavid van Moolenbroek  * Private.
104*00b67f09SDavid van Moolenbroek  */
105*00b67f09SDavid van Moolenbroek 
106*00b67f09SDavid van Moolenbroek /* coverity[+kill] */
107*00b67f09SDavid van Moolenbroek static void
default_callback(const char * file,int line,isc_assertiontype_t type,const char * cond)108*00b67f09SDavid van Moolenbroek default_callback(const char *file, int line, isc_assertiontype_t type,
109*00b67f09SDavid van Moolenbroek 		 const char *cond)
110*00b67f09SDavid van Moolenbroek {
111*00b67f09SDavid van Moolenbroek 	void *tracebuf[BACKTRACE_MAXFRAME];
112*00b67f09SDavid van Moolenbroek 	int i, nframes;
113*00b67f09SDavid van Moolenbroek 	const char *logsuffix = ".";
114*00b67f09SDavid van Moolenbroek 	const char *fname;
115*00b67f09SDavid van Moolenbroek 	isc_result_t result;
116*00b67f09SDavid van Moolenbroek 
117*00b67f09SDavid van Moolenbroek 	result = isc_backtrace_gettrace(tracebuf, BACKTRACE_MAXFRAME, &nframes);
118*00b67f09SDavid van Moolenbroek 		if (result == ISC_R_SUCCESS && nframes > 0)
119*00b67f09SDavid van Moolenbroek 			logsuffix = ", back trace";
120*00b67f09SDavid van Moolenbroek 
121*00b67f09SDavid van Moolenbroek 	fprintf(stderr, "%s:%d: %s(%s) %s%s\n",
122*00b67f09SDavid van Moolenbroek 		file, line, isc_assertion_typetotext(type), cond,
123*00b67f09SDavid van Moolenbroek 		isc_msgcat_get(isc_msgcat, ISC_MSGSET_GENERAL,
124*00b67f09SDavid van Moolenbroek 			       ISC_MSG_FAILED, "failed"), logsuffix);
125*00b67f09SDavid van Moolenbroek 	if (result == ISC_R_SUCCESS) {
126*00b67f09SDavid van Moolenbroek 		for (i = 0; i < nframes; i++) {
127*00b67f09SDavid van Moolenbroek 			unsigned long offset;
128*00b67f09SDavid van Moolenbroek 
129*00b67f09SDavid van Moolenbroek 			fname = NULL;
130*00b67f09SDavid van Moolenbroek 			result = isc_backtrace_getsymbol(tracebuf[i], &fname,
131*00b67f09SDavid van Moolenbroek 							 &offset);
132*00b67f09SDavid van Moolenbroek 			if (result == ISC_R_SUCCESS) {
133*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "#%d %p in %s()+0x%lx\n", i,
134*00b67f09SDavid van Moolenbroek 					tracebuf[i], fname, offset);
135*00b67f09SDavid van Moolenbroek 			} else {
136*00b67f09SDavid van Moolenbroek 				fprintf(stderr, "#%d %p in ??\n", i,
137*00b67f09SDavid van Moolenbroek 					tracebuf[i]);
138*00b67f09SDavid van Moolenbroek 			}
139*00b67f09SDavid van Moolenbroek 		}
140*00b67f09SDavid van Moolenbroek 	}
141*00b67f09SDavid van Moolenbroek 	fflush(stderr);
142*00b67f09SDavid van Moolenbroek }
143