struct/buf.c: Add functions for serializing and deserializing `kludge64'.
[mLib] / utils / macros.h
1 /* -*-c-*-
2 *
3 * Handy macros
4 *
5 * (c) 2003 Straylight/Edgeware
6 */
7
8 /*----- Licensing notice --------------------------------------------------*
9 *
10 * This file is part of the mLib utilities library.
11 *
12 * mLib is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU Library General Public License as
14 * published by the Free Software Foundation; either version 2 of the
15 * License, or (at your option) any later version.
16 *
17 * mLib is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU Library General Public License for more details.
21 *
22 * You should have received a copy of the GNU Library General Public
23 * License along with mLib; if not, write to the Free
24 * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston,
25 * MA 02111-1307, USA.
26 */
27
28 #ifndef MLIB_MACROS_H
29 #define MLIB_MACROS_H
30
31 #ifdef __cplusplus
32 extern "C" {
33 #endif
34
35 /*----- Header files ------------------------------------------------------*/
36
37 #include <assert.h>
38
39 #ifndef MLIB_COMPILER_H
40 # include "compiler.h"
41 #endif
42
43 /*----- Miscellaneous utility macros --------------------------------------*/
44
45 #define N(v) (sizeof(v)/sizeof(*(v)))
46
47 #define MLIB__STR(x) #x
48 #define STR(x) MLIB__STR(x)
49
50 #define MLIB__GLUE(x, y) x##y
51 #define GLUE(x, y) MLIB__GLUE(x, y)
52
53 #ifdef static_assert
54 # define STATIC_ASSERT(cond, msg) static_assert(!!(cond), msg)
55 #else
56 # define STATIC_ASSERT(cond, msg) \
57 IGNORABLE extern char static_assert_failed[2*!!(cond) - 1]
58 #endif
59
60 /*----- String and character hacks ----------------------------------------*/
61
62 #define CTYPE_HACK(func, ch) (func((unsigned char)(ch)))
63
64 #define ISALNUM(ch) CTYPE_HACK(isalnum, ch)
65 #define ISALPHA(ch) CTYPE_HACK(isalpha, ch)
66 #define ISASCII(ch) CTYPE_HACK(isascii, ch)
67 #define ISBLANK(ch) CTYPE_HACK(isblank, ch)
68 #define ISCNTRL(ch) CTYPE_HACK(iscntrl, ch)
69 #define ISDIGIT(ch) CTYPE_HACK(isdigit, ch)
70 #define ISGRAPH(ch) CTYPE_HACK(isgraph, ch)
71 #define ISLOWER(ch) CTYPE_HACK(islower, ch)
72 #define ISPRINT(ch) CTYPE_HACK(isprint, ch)
73 #define ISPUNCT(ch) CTYPE_HACK(ispunct, ch)
74 #define ISSPACE(ch) CTYPE_HACK(isspace, ch)
75 #define ISUPPER(ch) CTYPE_HACK(isupper, ch)
76 #define ISXDIGIT(ch) CTYPE_HACK(isxdigit, ch)
77
78 #define TOASCII(ch) CTYPE_HACK(toascii, ch)
79 #define TOLOWER(ch) CTYPE_HACK(tolower, ch)
80 #define TOUPPER(ch) CTYPE_HACK(toupper, ch)
81
82 #define MEMCMP(x, op, y, n) (memcmp((x), (y), (n)) op 0)
83 #define STRCMP(x, op, y) (strcmp((x), (y)) op 0)
84 #define STRNCMP(x, op, y, n) (strncmp((x), (y), (n)) op 0)
85
86 /*----- Compiler diagnostics ----------------------------------------------*/
87
88 /* --- Compiler-specific definitions --- */
89
90 #if GCC_VERSION_P(2, 5) || CLANG_VERSION_P(3, 3)
91 # define NORETURN __attribute__((__noreturn__))
92 # define PRINTF_LIKE(fix, aix) __attribute__((__format__(printf, fix, aix)))
93 # define SCANF_LIKE(fix, aix) __attribute__((__format__(scanf, fix, aix)))
94 # define IGNORABLE __attribute__((__unused__))
95 #endif
96
97 #if GCC_VERSION_P(3, 4) || CLANG_VERSION_P(3, 3)
98 # define MUST_CHECK __attribute__((__warn_unused_result__))
99 #endif
100
101 #if GCC_VERSION_P(4, 5) || CLANG_VERSION_P(3, 3)
102 # define DEPRECATED(msg) __attribute__((__deprecated__(msg)))
103 #elif GCC_VERSION_P(3, 1)
104 # define DEPRECATED(msg) __attribute__((__deprecated__))
105 #endif
106
107 #if GCC_VERSION_P(4, 0) || CLANG_VERSION_P(3, 3)
108 # define EXECL_LIKE(ntrail) __attribute__((__sentinel__(ntrail)))
109 #endif
110
111 #if CLANG_VERSION_P(3, 3)
112
113 # define MLIB__PRAGMA_HACK(x) _Pragma(#x)
114 # define MLIB__MUFFLE_WARNINGS(warns, body) \
115 _Pragma("clang diagnostic push") \
116 warns \
117 body \
118 _Pragma("clang diagnostic pop")
119 # define CLANG_WARNING(warn) \
120 MLIB__PRAGMA_HACK(clang diagnostic ignored warn)
121 # define MUFFLE_WARNINGS_DECL(warns, body) \
122 MLIB__MUFFLE_WARNINGS(warns, body)
123 # define MUFFLE_WARNINGS_EXPR(warns, body) \
124 __extension__ ({ MLIB__MUFFLE_WARNINGS(warns, (body);) })
125 # define MUFFLE_WARNINGS_STMT(warns, body) \
126 do { MLIB__MUFFLE_WARNINGS(warns, body) } while (0)
127
128 #elif GCC_VERSION_P(4, 6)
129
130 /* --- Diagnostic suppression in GCC: a tale of woe --- *
131 *
132 * This is extremely unpleasant, largely as a result of bugs in the GCC
133 * preprocessor's handling of @_Pragma@. The fundamental problem is
134 * that it's the preprocessor, and not the compiler proper, which
135 * detects @_Pragma@, emitting @#pragma@ lines into its output; and it
136 * does it during macro expansion, even if the macro is being expanded
137 * during argument collection. Since arguments are expanded before
138 * replacing the macro's invocation with its body, a pragma in an
139 * argument will be emitted %%\emph{before}%% any pragmata in the body,
140 * even if they appear before the argument in the body -- and even if
141 * the argument doesn't actually appear anywhere at all in the body.
142 *
143 * Another, rather less significant, problem is that @_Pragma@'s
144 * argument is a single string literal, recognized in translation phase
145 * 4, before string-literal concatenation in phase 6, so we must build
146 * pragma bodies as token lists and then stringify them.
147 *
148 * As a result, we need some subterfuge here. The @MLIB__PRAGMA_HACK@
149 * macro issues a @_Pragma@ on its argument token list, which it
150 * stringifies; this deals with the second problem. The first is
151 * trickier: we must delay expansion of @MLIB__PRAGMA_HACK@ from the
152 * argument collection phase to the body rescanning phase, and we do
153 * this by splitting the invocations between @GCC_WARNING@ macro calls:
154 * the name is left hanging from the previous call (or from
155 * @MLIB__MUFFLE_WARNINGS@, in the first case) and the body is supplied
156 * by @GCC_WARNING@, which also supplies the next @MLIB__PRAGMA_HACK@.
157 * The remaining problem is to make sure we can dispose of the final
158 * trailing @MLIB__PRAGMA_HACK@ harmlessly, which we do by introducing
159 * an extra argument @emitp@, which may be either @t@ or @nil@; this
160 * dispatches to an appropriate helper macro by means of token-pasting.
161 *
162 * I'm so sorry.
163 */
164
165 # define MLIB__PRAGMA_HACK_t(x) _Pragma(#x)
166 # define MLIB__PRAGMA_HACK_nil(x)
167 # define MLIB__PRAGMA_HACK(emitp, x) MLIB__PRAGMA_HACK_##emitp(x)
168 # define MLIB__MUFFLE_WARNINGS(warns, body) \
169 _Pragma("GCC diagnostic push") MLIB__PRAGMA_HACK \
170 warns \
171 (nil, nil) \
172 body \
173 _Pragma("GCC diagnostic pop")
174 # define GCC_WARNING(warn) \
175 (t, GCC diagnostic ignored warn) MLIB__PRAGMA_HACK
176 # define MUFFLE_WARNINGS_DECL(warns, body) \
177 MLIB__MUFFLE_WARNINGS(warns, body)
178 # define MUFFLE_WARNINGS_EXPR(warns, body) \
179 __extension__ ({ MLIB__MUFFLE_WARNINGS(warns, (body);) })
180 # define MUFFLE_WARNINGS_STMT(warns, body) \
181 do { MLIB__MUFFLE_WARNINGS(warns, body) } while (0)
182 #endif
183
184 /* --- Fallback definitions, mostly trivial --- */
185
186 #ifndef DEPRECATED
187 # define DEPRECATED(msg)
188 #endif
189
190 #ifndef EXECL_LIKE
191 # define EXECL_LIKE(ntrail)
192 #endif
193
194 #ifndef DISCARD
195 # define DISCARD(x) do if (x); while (0)
196 #endif
197
198 #ifndef IGNORE
199 # define IGNORE(x) ((void)(x))
200 #endif
201
202 #ifndef MUFFLE_WARNINGS_DECL
203 # define MUFFLE_WARNINGS_DECL(warns, body) body
204 #endif
205
206 #ifndef MUFFLE_WARNINGS_EXPR
207 # define MUFFLE_WARNINGS_EXPR(warns, body) (body)
208 #endif
209
210 #ifndef MUFFLE_WARNINGS_STMT
211 # define MUFFLE_WARNINGS_STMT(warns, body) do { body } while (0)
212 #endif
213
214 #ifndef PRINTF_LIKE
215 # define PRINF_LIKE(fmtix, argix)
216 #endif
217
218 #ifndef SCANF_LIKE
219 # define SCANF_LIKE(fmtix, argix)
220 #endif
221
222 #ifndef NORETURN
223 # define NORETURN
224 #endif
225
226 #ifndef IGNORABLE
227 # define IGNORABLE
228 #endif
229
230 #ifndef MUST_CHECK
231 # define MUST_CHECK
232 #endif
233
234 #ifndef GCC_WARNING
235 # define GCC_WARNING(warn)
236 #endif
237
238 #ifndef CLANG_WARNING
239 # define CLANG_WARNING(warn)
240 #endif
241
242 /*----- That's all, folks -------------------------------------------------*/
243
244 #ifdef __cplusplus
245 }
246 #endif
247
248 #endif