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