1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or https://opensource.org/licenses/CDDL-1.0.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2007 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #include_next <assert.h>
28 
29 #ifndef _LIBSPL_ASSERT_H
30 #define	_LIBSPL_ASSERT_H
31 
32 #include <stdio.h>
33 #include <stdlib.h>
34 #include <stdarg.h>
35 #include <sys/types.h>
36 
37 /* Workaround for non-Clang compilers */
38 #ifndef __has_feature
39 #define	__has_feature(x) 0
40 #endif
41 
42 /* We need to workaround libspl_set_assert_ok() that we have for zdb */
43 #if __has_feature(attribute_analyzer_noreturn) || defined(__COVERITY__)
44 #define	NORETURN	__attribute__((__noreturn__))
45 #else
46 #define	NORETURN
47 #endif
48 
49 /* Set to non-zero to avoid abort()ing on an assertion failure */
50 extern void libspl_set_assert_ok(boolean_t val);
51 
52 /* printf version of libspl_assert */
53 extern void libspl_assertf(const char *file, const char *func, int line,
54     const char *format, ...) NORETURN __attribute__((format(printf, 4, 5)));
55 
56 static inline int
57 libspl_assert(const char *buf, const char *file, const char *func, int line)
58 {
59 	libspl_assertf(file, func, line, "%s", buf);
60 	return (0);
61 }
62 
63 #ifdef verify
64 #undef verify
65 #endif
66 
67 #define	VERIFY(cond)							\
68 	(void) ((!(cond)) &&						\
69 	    libspl_assert(#cond, __FILE__, __FUNCTION__, __LINE__))
70 #define	verify(cond)							\
71 	(void) ((!(cond)) &&						\
72 	    libspl_assert(#cond, __FILE__, __FUNCTION__, __LINE__))
73 
74 #define	VERIFY3B(LEFT, OP, RIGHT)					\
75 do {									\
76 	const boolean_t __left = (boolean_t)(LEFT);			\
77 	const boolean_t __right = (boolean_t)(RIGHT);			\
78 	if (!(__left OP __right))					\
79 		libspl_assertf(__FILE__, __FUNCTION__, __LINE__,	\
80 		    "%s %s %s (0x%llx %s 0x%llx)", #LEFT, #OP, #RIGHT,	\
81 		    (u_longlong_t)__left, #OP, (u_longlong_t)__right);	\
82 } while (0)
83 
84 #define	VERIFY3S(LEFT, OP, RIGHT)					\
85 do {									\
86 	const int64_t __left = (int64_t)(LEFT);				\
87 	const int64_t __right = (int64_t)(RIGHT);			\
88 	if (!(__left OP __right))					\
89 		libspl_assertf(__FILE__, __FUNCTION__, __LINE__,	\
90 		    "%s %s %s (0x%llx %s 0x%llx)", #LEFT, #OP, #RIGHT,	\
91 		    (u_longlong_t)__left, #OP, (u_longlong_t)__right);	\
92 } while (0)
93 
94 #define	VERIFY3U(LEFT, OP, RIGHT)					\
95 do {									\
96 	const uint64_t __left = (uint64_t)(LEFT);			\
97 	const uint64_t __right = (uint64_t)(RIGHT);			\
98 	if (!(__left OP __right))					\
99 		libspl_assertf(__FILE__, __FUNCTION__, __LINE__,	\
100 		    "%s %s %s (0x%llx %s 0x%llx)", #LEFT, #OP, #RIGHT,	\
101 		    (u_longlong_t)__left, #OP, (u_longlong_t)__right);	\
102 } while (0)
103 
104 #define	VERIFY3P(LEFT, OP, RIGHT)					\
105 do {									\
106 	const uintptr_t __left = (uintptr_t)(LEFT);			\
107 	const uintptr_t __right = (uintptr_t)(RIGHT);			\
108 	if (!(__left OP __right))					\
109 		libspl_assertf(__FILE__, __FUNCTION__, __LINE__,	\
110 		    "%s %s %s (0x%llx %s 0x%llx)", #LEFT, #OP, #RIGHT,	\
111 		    (u_longlong_t)__left, #OP, (u_longlong_t)__right);	\
112 } while (0)
113 
114 #define	VERIFY0(LEFT)							\
115 do {									\
116 	const uint64_t __left = (uint64_t)(LEFT);			\
117 	if (!(__left == 0))						\
118 		libspl_assertf(__FILE__, __FUNCTION__, __LINE__,	\
119 		    "%s == 0 (0x%llx == 0)", #LEFT,			\
120 		    (u_longlong_t)__left);				\
121 } while (0)
122 
123 #ifdef assert
124 #undef assert
125 #endif
126 
127 #ifdef NDEBUG
128 #define	ASSERT3B(x, y, z)						\
129 	((void) sizeof ((uintptr_t)(x)), (void) sizeof ((uintptr_t)(z)))
130 #define	ASSERT3S(x, y, z)						\
131 	((void) sizeof ((uintptr_t)(x)), (void) sizeof ((uintptr_t)(z)))
132 #define	ASSERT3U(x, y, z)						\
133 	((void) sizeof ((uintptr_t)(x)), (void) sizeof ((uintptr_t)(z)))
134 #define	ASSERT3P(x, y, z)						\
135 	((void) sizeof ((uintptr_t)(x)), (void) sizeof ((uintptr_t)(z)))
136 #define	ASSERT0(x)		((void) sizeof ((uintptr_t)(x)))
137 #define	ASSERT(x)		((void) sizeof ((uintptr_t)(x)))
138 #define	assert(x)		((void) sizeof ((uintptr_t)(x)))
139 #define	IMPLY(A, B)							\
140 	((void) sizeof ((uintptr_t)(A)), (void) sizeof ((uintptr_t)(B)))
141 #define	EQUIV(A, B)							\
142 	((void) sizeof ((uintptr_t)(A)), (void) sizeof ((uintptr_t)(B)))
143 #else
144 #define	ASSERT3B	VERIFY3B
145 #define	ASSERT3S	VERIFY3S
146 #define	ASSERT3U	VERIFY3U
147 #define	ASSERT3P	VERIFY3P
148 #define	ASSERT0		VERIFY0
149 #define	ASSERT		VERIFY
150 #define	assert		VERIFY
151 #define	IMPLY(A, B) \
152 	((void)(((!(A)) || (B)) || \
153 	    libspl_assert("(" #A ") implies (" #B ")", \
154 	    __FILE__, __FUNCTION__, __LINE__)))
155 #define	EQUIV(A, B) \
156 	((void)((!!(A) == !!(B)) || \
157 	    libspl_assert("(" #A ") is equivalent to (" #B ")", \
158 	    __FILE__, __FUNCTION__, __LINE__)))
159 
160 #endif  /* NDEBUG */
161 
162 #endif  /* _LIBSPL_ASSERT_H */
163