-#if defined(__GNUC__)
-
-# define GCC_VERSION_P(maj, min) \
- (__GNUC__ > (maj) || (__GNUC__ == (maj) && __GNUC_MINOR__ >= (min)))
-
-# if GCC_VERSION_P(2, 5)
-# define NORETURN __attribute__((noreturn))
-# define PRINTF_LIKE(fix, aix) __attribute__((format(printf, fix, aix)))
-# define SCANF_LIKE(fix, aix) __attribute__((format(scanf, fix, aix)))
-# define IGNORABLE __attribute__((unused))
-# endif
-
-# if GCC_VERSION_P(4, 5)
-# define DEPRECATED(msg) __attribute__((deprecated(msg)))
-# elif GCC_VERSION_P(3, 1)
-# define DEPRECATED(msg) __attribute__((deprecated))
-# endif
-
-# if GCC_VERSION_P(4, 0)
-# define EXECL_LIKE(ntrail) __attribute__((sentinel(ntrail)))
-# endif
-
-# if GCC_VERSION_P(4, 6)
-
- /* --- Diagnostic suppression in GCC: a tale of woe --- *
- *
- * This is extremely unpleasant, largely as a result of bugs in the GCC
- * preprocessor's handling of @_Pragma@. The fundamental problem is
- * that it's the preprocessor, and not the compiler proper, which
- * detects @_Pragma@, emitting @#pragma@ lines into its output; and it
- * does it during macro expansion, even if the macro is being expanded
- * during argument collection. Since arguments are expanded before
- * replacing the macro's invocation with its body, a pragma in an
- * argument will be emitted %%\emph{before}%% any pragmata in the body,
- * even if they appear before the argument in the body -- and even if
- * the argument doesn't actually appear anywhere at all in the body.
- *
- * Another, rather less significant, problem is that @_Pragma@'s
- * argument is a single string literal, recognized in translation phase
- * 4, before string-literal concatenation in phase 6, so we must build
- * pragma bodies as token lists and then stringify them.
- *
- * As a result, we need some subterfuge here. The @MLIB__PRAGMA_HACK@
- * macro issues a @_Pragma@ on its argument token list, which it
- * stringifies; this deals with the second problem. The first is
- * trickier: we must delay expansion of @MLIB__PRAGMA_HACK@ from the
- * argument collection phase to the body rescanning phase, and we do
- * this by splitting the invocations between @GCC_WARNING@ macro calls:
- * the name is left hanging from the previous call (or from
- * @MLIB__MUFFLE_WARNINGS@, in the first case) and the body is supplied
- * by @GCC_WARNING@, which also supplies the next @MLIB__PRAGMA_HACK@.
- * The remaining problem is to make sure we can dispose of the final
- * trailing @MLIB__PRAGMA_HACK@ harmlessly, which we do by introducing
- * an extra argument @emitp@, which may be either @t@ or @nil@; this
- * dispatches to an appropriate helper macro by means of token-pasting.
- *
- * I'm so sorry.
- */
-
-# define MLIB__PRAGMA_HACK_t(x) _Pragma(#x)
-# define MLIB__PRAGMA_HACK_nil(x)
-# define MLIB__PRAGMA_HACK(emitp, x) MLIB__PRAGMA_HACK_##emitp(x)
-# define MLIB__MUFFLE_WARNINGS(warns, body) \
+#if GCC_VERSION_P(2, 5) || CLANG_VERSION_P(3, 3)
+# define NORETURN __attribute__((__noreturn__))
+# define PRINTF_LIKE(fix, aix) __attribute__((__format__(printf, fix, aix)))
+# define SCANF_LIKE(fix, aix) __attribute__((__format__(scanf, fix, aix)))
+# define IGNORABLE __attribute__((__unused__))
+#endif
+
+#if GCC_VERSION_P(3, 4) || CLANG_VERSION_P(3, 3)
+# define MUST_CHECK __attribute__((__warn_unused_result__))
+#endif
+
+#if GCC_VERSION_P(4, 5) || CLANG_VERSION_P(3, 3)
+# define DEPRECATED(msg) __attribute__((__deprecated__(msg)))
+#elif GCC_VERSION_P(3, 1)
+# define DEPRECATED(msg) __attribute__((__deprecated__))
+#endif
+
+#if GCC_VERSION_P(4, 0) || CLANG_VERSION_P(3, 3)
+# define EXECL_LIKE(ntrail) __attribute__((__sentinel__(ntrail)))
+#endif
+
+#if CLANG_VERSION_P(3, 3)
+
+# define MLIB__PRAGMA_HACK(x) _Pragma(#x)
+# define MLIB__MUFFLE_WARNINGS(warns, body) \
+ _Pragma("clang diagnostic push") \
+ warns \
+ body \
+ _Pragma("clang diagnostic pop")
+# define CLANG_WARNING(warn) \
+ MLIB__PRAGMA_HACK(clang diagnostic ignored warn)
+# define MUFFLE_WARNINGS_DECL(warns, body) \
+ MLIB__MUFFLE_WARNINGS(warns, body)
+# define MUFFLE_WARNINGS_EXPR(warns, body) \
+ __extension__ ({ MLIB__MUFFLE_WARNINGS(warns, (body);) })
+# define MUFFLE_WARNINGS_STMT(warns, body) \
+ do { MLIB__MUFFLE_WARNINGS(warns, body) } while (0)
+
+#elif GCC_VERSION_P(4, 6)
+
+ /* --- Diagnostic suppression in GCC: a tale of woe --- *
+ *
+ * This is extremely unpleasant, largely as a result of bugs in the GCC
+ * preprocessor's handling of @_Pragma@. The fundamental problem is
+ * that it's the preprocessor, and not the compiler proper, which
+ * detects @_Pragma@, emitting @#pragma@ lines into its output; and it
+ * does it during macro expansion, even if the macro is being expanded
+ * during argument collection. Since arguments are expanded before
+ * replacing the macro's invocation with its body, a pragma in an
+ * argument will be emitted %%\emph{before}%% any pragmata in the body,
+ * even if they appear before the argument in the body -- and even if
+ * the argument doesn't actually appear anywhere at all in the body.
+ *
+ * Another, rather less significant, problem is that @_Pragma@'s
+ * argument is a single string literal, recognized in translation phase
+ * 4, before string-literal concatenation in phase 6, so we must build
+ * pragma bodies as token lists and then stringify them.
+ *
+ * As a result, we need some subterfuge here. The @MLIB__PRAGMA_HACK@
+ * macro issues a @_Pragma@ on its argument token list, which it
+ * stringifies; this deals with the second problem. The first is
+ * trickier: we must delay expansion of @MLIB__PRAGMA_HACK@ from the
+ * argument collection phase to the body rescanning phase, and we do
+ * this by splitting the invocations between @GCC_WARNING@ macro calls:
+ * the name is left hanging from the previous call (or from
+ * @MLIB__MUFFLE_WARNINGS@, in the first case) and the body is supplied
+ * by @GCC_WARNING@, which also supplies the next @MLIB__PRAGMA_HACK@.
+ * The remaining problem is to make sure we can dispose of the final
+ * trailing @MLIB__PRAGMA_HACK@ harmlessly, which we do by introducing
+ * an extra argument @emitp@, which may be either @t@ or @nil@; this
+ * dispatches to an appropriate helper macro by means of token-pasting.
+ *
+ * I'm so sorry.
+ */
+
+# define MLIB__PRAGMA_HACK_t(x) _Pragma(#x)
+# define MLIB__PRAGMA_HACK_nil(x)
+# define MLIB__PRAGMA_HACK(emitp, x) MLIB__PRAGMA_HACK_##emitp(x)
+# define MLIB__MUFFLE_WARNINGS(warns, body) \