2 * Copyright (c) Meta Platforms, Inc. and affiliates.
5 * This source code is licensed under both the BSD-style license (found in the
6 * LICENSE file in the root directory of this source tree) and the GPLv2 (found
7 * in the COPYING file in the root directory of this source tree).
8 * You may select, at your option, one of the above-listed licenses.
11 /* Note : this module is expected to remain private, do not expose it */
13 #ifndef ERROR_H_MODULE
14 #define ERROR_H_MODULE
16 #if defined (__cplusplus)
21 /* ****************************************
23 ******************************************/
24 #include "../zstd_errors.h" /* enum list */
27 #include "zstd_deps.h" /* size_t */
30 /* ****************************************
32 ******************************************/
34 # define ERR_STATIC static __attribute__((unused))
35 #elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */)
36 # define ERR_STATIC static inline
37 #elif defined(_MSC_VER)
38 # define ERR_STATIC static __inline
40 # define ERR_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */
44 /*-****************************************
45 * Customization (error_public.h)
46 ******************************************/
47 typedef ZSTD_ErrorCode ERR_enum;
48 #define PREFIX(name) ZSTD_error_##name
51 /*-****************************************
52 * Error codes handling
53 ******************************************/
54 #undef ERROR /* already defined on Visual Studio */
55 #define ERROR(name) ZSTD_ERROR(name)
56 #define ZSTD_ERROR(name) ((size_t)-PREFIX(name))
58 ERR_STATIC unsigned ERR_isError(size_t code) { return (code > ERROR(maxCode)); }
60 ERR_STATIC ERR_enum ERR_getErrorCode(size_t code) { if (!ERR_isError(code)) return (ERR_enum)0; return (ERR_enum) (0-code); }
62 /* check and forward error code */
63 #define CHECK_V_F(e, f) \
69 #define CHECK_F(f) do { CHECK_V_F(_var_err__, f); } while (0)
72 /*-****************************************
74 ******************************************/
76 const char* ERR_getErrorString(ERR_enum code); /* error_private.c */
78 ERR_STATIC const char* ERR_getErrorName(size_t code)
80 return ERR_getErrorString(ERR_getErrorCode(code));
84 * Ignore: this is an internal helper.
86 * This is a helper function to help force C99-correctness during compilation.
87 * Under strict compilation modes, variadic macro arguments can't be empty.
88 * However, variadic function arguments can be. Using a function therefore lets
89 * us statically check that at least one (string) argument was passed,
90 * independent of the compilation flags.
92 static INLINE_KEYWORD UNUSED_ATTR
93 void _force_has_format_string(const char *format, ...) {
98 * Ignore: this is an internal helper.
100 * We want to force this function invocation to be syntactically correct, but
101 * we don't want to force runtime evaluation of its arguments.
103 #define _FORCE_HAS_FORMAT_STRING(...) \
106 _force_has_format_string(__VA_ARGS__); \
110 #define ERR_QUOTE(str) #str
113 * Return the specified error if the condition evaluates to true.
115 * In debug modes, prints additional information.
116 * In order to do that (particularly, printing the conditional that failed),
117 * this can't just wrap RETURN_ERROR().
119 #define RETURN_ERROR_IF(cond, err, ...) \
122 RAWLOG(3, "%s:%d: ERROR!: check %s failed, returning %s", \
123 __FILE__, __LINE__, ERR_QUOTE(cond), ERR_QUOTE(ERROR(err))); \
124 _FORCE_HAS_FORMAT_STRING(__VA_ARGS__); \
125 RAWLOG(3, ": " __VA_ARGS__); \
132 * Unconditionally return the specified error.
134 * In debug modes, prints additional information.
136 #define RETURN_ERROR(err, ...) \
138 RAWLOG(3, "%s:%d: ERROR!: unconditional check failed, returning %s", \
139 __FILE__, __LINE__, ERR_QUOTE(ERROR(err))); \
140 _FORCE_HAS_FORMAT_STRING(__VA_ARGS__); \
141 RAWLOG(3, ": " __VA_ARGS__); \
147 * If the provided expression evaluates to an error code, returns that error code.
149 * In debug modes, prints additional information.
151 #define FORWARD_IF_ERROR(err, ...) \
153 size_t const err_code = (err); \
154 if (ERR_isError(err_code)) { \
155 RAWLOG(3, "%s:%d: ERROR!: forwarding error in %s: %s", \
156 __FILE__, __LINE__, ERR_QUOTE(err), ERR_getErrorName(err_code)); \
157 _FORCE_HAS_FORMAT_STRING(__VA_ARGS__); \
158 RAWLOG(3, ": " __VA_ARGS__); \
164 #if defined (__cplusplus)
168 #endif /* ERROR_H_MODULE */