Home | History | Annotate | Line # | Download | only in gen
assert.c revision 1.18
      1  1.18  christos /*	$NetBSD: assert.c,v 1.18 2017/05/15 16:09:09 christos Exp $	*/
      2   1.6       cgd 
      3   1.5       cgd /*-
      4   1.5       cgd  * Copyright (c) 1992, 1993
      5   1.5       cgd  *	The Regents of the University of California.  All rights reserved.
      6   1.1       jtc  *
      7   1.1       jtc  * Redistribution and use in source and binary forms, with or without
      8   1.1       jtc  * modification, are permitted provided that the following conditions
      9   1.1       jtc  * are met:
     10   1.1       jtc  * 1. Redistributions of source code must retain the above copyright
     11   1.1       jtc  *    notice, this list of conditions and the following disclaimer.
     12   1.1       jtc  * 2. Redistributions in binary form must reproduce the above copyright
     13   1.1       jtc  *    notice, this list of conditions and the following disclaimer in the
     14   1.1       jtc  *    documentation and/or other materials provided with the distribution.
     15  1.15       agc  * 3. Neither the name of the University nor the names of its contributors
     16   1.5       cgd  *    may be used to endorse or promote products derived from this software
     17   1.5       cgd  *    without specific prior written permission.
     18   1.1       jtc  *
     19   1.5       cgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
     20   1.5       cgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     21   1.5       cgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     22   1.5       cgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
     23   1.5       cgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     24   1.5       cgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     25   1.5       cgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     26   1.5       cgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     27   1.5       cgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     28   1.5       cgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     29   1.5       cgd  * SUCH DAMAGE.
     30   1.1       jtc  */
     31   1.1       jtc 
     32   1.7  christos #include <sys/cdefs.h>
     33   1.2   mycroft #if defined(LIBC_SCCS) && !defined(lint)
     34   1.6       cgd #if 0
     35   1.6       cgd static char sccsid[] = "@(#)assert.c	8.1 (Berkeley) 6/4/93";
     36   1.6       cgd #else
     37  1.18  christos __RCSID("$NetBSD: assert.c,v 1.18 2017/05/15 16:09:09 christos Exp $");
     38   1.6       cgd #endif
     39   1.2   mycroft #endif /* LIBC_SCCS and not lint */
     40   1.2   mycroft 
     41  1.16    kleink #include "namespace.h"
     42   1.5       cgd #include <sys/types.h>
     43   1.8     lukem 
     44   1.5       cgd #include <assert.h>
     45   1.1       jtc #include <stdio.h>
     46   1.1       jtc #include <stdlib.h>
     47  1.18  christos #include <unistd.h>
     48   1.8     lukem #include <syslog.h>
     49  1.18  christos #include "extern.h"
     50   1.9  christos 
     51  1.18  christos static int
     52  1.18  christos fmtassert(char *buf, size_t len, const char *file, int line,
     53  1.18  christos     const char *function, const char *failedexpr)
     54  1.10    kleink {
     55  1.18  christos 	return snprintf_ss(buf, len,
     56  1.10    kleink 	    "assertion \"%s\" failed: file \"%s\", line %d%s%s%s\n",
     57  1.10    kleink 	    failedexpr, file, line,
     58  1.10    kleink 	    function ? ", function \"" : "",
     59  1.10    kleink 	    function ? function : "",
     60  1.10    kleink 	    function ? "\"" : "");
     61  1.18  christos }
     62  1.18  christos 
     63  1.18  christos void
     64  1.18  christos __assert13(const char *file, int line, const char *function,
     65  1.18  christos     const char *failedexpr)
     66  1.18  christos {
     67  1.18  christos 	char buf[1024];
     68  1.18  christos 	int l = fmtassert(buf, sizeof(buf), file, line, function, failedexpr);
     69  1.18  christos 	if (l < 0)
     70  1.18  christos 		abort();
     71  1.18  christos 	(void)write(STDERR_FILENO, buf, (size_t)l);
     72  1.10    kleink 	abort();
     73  1.10    kleink 	/* NOTREACHED */
     74  1.10    kleink }
     75  1.10    kleink 
     76  1.10    kleink void
     77  1.17       abs __assert(const char *file, int line, const char *failedexpr)
     78   1.1       jtc {
     79  1.12     lukem 
     80  1.12     lukem 	__assert13(file, line, NULL, failedexpr);
     81   1.5       cgd 	/* NOTREACHED */
     82  1.10    kleink }
     83  1.10    kleink 
     84  1.12     lukem 
     85  1.12     lukem enum {
     86  1.12     lukem 	DIAGASSERT_ABORT =	1<<0,
     87  1.12     lukem 	DIAGASSERT_STDERR =	1<<1,
     88  1.12     lukem 	DIAGASSERT_SYSLOG =	1<<2
     89  1.12     lukem };
     90  1.12     lukem 
     91  1.12     lukem static int	diagassert_flags = -1;
     92  1.12     lukem 
     93  1.10    kleink void
     94  1.17       abs __diagassert13(const char *file, int line, const char *function,
     95  1.17       abs     const char *failedexpr)
     96  1.10    kleink {
     97  1.12     lukem 	char buf[1024];
     98  1.12     lukem 
     99  1.12     lukem 	if (diagassert_flags == -1) {
    100  1.12     lukem 		char *p;
    101  1.12     lukem 
    102  1.12     lukem 		diagassert_flags = DIAGASSERT_SYSLOG;
    103  1.12     lukem 
    104  1.12     lukem 		for (p = getenv("LIBC_DIAGASSERT"); p && *p; p++) {
    105  1.12     lukem 			switch (*p) {
    106  1.12     lukem 			case 'a':
    107  1.12     lukem 				diagassert_flags |= DIAGASSERT_ABORT;
    108  1.12     lukem 				break;
    109  1.12     lukem 			case 'A':
    110  1.12     lukem 				diagassert_flags &= ~DIAGASSERT_ABORT;
    111  1.12     lukem 				break;
    112  1.12     lukem 			case 'e':
    113  1.12     lukem 				diagassert_flags |= DIAGASSERT_STDERR;
    114  1.12     lukem 				break;
    115  1.12     lukem 			case 'E':
    116  1.12     lukem 				diagassert_flags &= ~DIAGASSERT_STDERR;
    117  1.12     lukem 				break;
    118  1.12     lukem 			case 'l':
    119  1.12     lukem 				diagassert_flags |= DIAGASSERT_SYSLOG;
    120  1.12     lukem 				break;
    121  1.12     lukem 			case 'L':
    122  1.12     lukem 				diagassert_flags &= ~DIAGASSERT_SYSLOG;
    123  1.12     lukem 				break;
    124  1.12     lukem 			}
    125  1.12     lukem 		}
    126  1.12     lukem 	}
    127  1.12     lukem 
    128  1.18  christos 	fmtassert(buf, sizeof(buf), file, line, function, failedexpr);
    129  1.18  christos 	if (diagassert_flags & DIAGASSERT_STDERR) {
    130  1.18  christos 		char ebuf[1024];
    131  1.18  christos 		int l = snprintf_ss(ebuf, sizeof(ebuf), "%s: %s\n",
    132  1.18  christos 		    getprogname(), buf);
    133  1.18  christos 		if (l == -1)
    134  1.18  christos 			abort();
    135  1.18  christos 		(void)write(STDERR_FILENO, ebuf, (size_t)l);
    136  1.18  christos 	}
    137  1.18  christos 	if (diagassert_flags & DIAGASSERT_SYSLOG) {
    138  1.18  christos 		struct syslog_data sdata = SYSLOG_DATA_INIT;
    139  1.18  christos 		syslog_ss(LOG_DEBUG | LOG_USER, &sdata, "%s", buf);
    140  1.18  christos 	}
    141  1.12     lukem 	if (diagassert_flags & DIAGASSERT_ABORT)
    142  1.12     lukem 		abort();
    143   1.8     lukem }
    144   1.8     lukem 
    145   1.8     lukem void
    146  1.17       abs __diagassert(const char *file, int line, const char *failedexpr)
    147   1.8     lukem {
    148  1.12     lukem 
    149  1.12     lukem 	__diagassert13(file, line, NULL, failedexpr);
    150   1.1       jtc }
    151