infix
A JIT-Powered FFI Library for C
Loading...
Searching...
No Matches
error.c
Go to the documentation of this file.
1
40#include <infix/infix.h>
41#include <stdarg.h>
42#include <stdio.h> // For snprintf
43#include <string.h>
44
45// Use a portable mechanism for thread-local storage (TLS).
46// The order of checks is critical for cross-platform compatibility.
47#if defined(__OpenBSD__)
48// OpenBSD has known issues with TLS cleanup in some linking scenarios (segfault on exit).
49// We disable TLS entirely on this platform to ensure stability, at the cost of thread-safety.
50#define INFIX_TLS
51#elif defined(_MSC_VER)
52// Microsoft Visual C++
53#define INFIX_TLS __declspec(thread)
54#elif defined(_WIN32) && defined(__clang__)
55// Clang on Windows: check if behaving like MSVC or GCC.
56// If using MSVC codegen/headers, use declspec.
57#define INFIX_TLS __declspec(thread)
58#elif defined(__GNUC__)
59// MinGW (GCC on Windows) and standard GCC/Clang on *nix.
60// MinGW prefers __thread or _Thread_local over __declspec(thread).
61#define INFIX_TLS __thread
62#elif defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L && !defined(__STDC_NO_THREADS__)
63// Fallback to C11 standard
64#define INFIX_TLS _Thread_local
65#else
66// Fallback for compilers that do not support TLS. This is not thread-safe.
67#warning "Compiler does not support thread-local storage; error handling will not be thread-safe."
68#define INFIX_TLS
69#endif
70
71// A portable macro for safe string copying to prevent buffer overflows.
72#if defined(_MSC_VER)
73#define _INFIX_SAFE_STRNCPY(dest, src, count) strncpy_s(dest, sizeof(dest), src, count)
74#else
75#define _INFIX_SAFE_STRNCPY(dest, src, count) \
76 do { \
77 strncpy(dest, src, (count)); \
78 (dest)[(sizeof(dest)) - 1] = '\0'; \
79 } while (0)
80#endif
81
90
100
108 switch (code) {
110 return "Success";
112 return "An unknown error occurred";
114 return "Out of memory";
116 return "Failed to allocate executable memory";
118 return "Failed to change memory protection flags";
120 return "Unexpected token or character";
122 return "Unterminated aggregate (missing '}', '>', ']', or ')')'";
124 return "Invalid type keyword";
126 return "Function signature missing '->' or return type";
128 return "Integer overflow detected during layout calculation";
130 return "Type definition is too deeply nested";
132 return "Named type was declared with empty angle brackets";
134 return "The current platform ABI is not supported";
136 return "A data type was too large to be handled by the ABI";
138 return "Named type not found in registry or is an undefined forward declaration";
140 return "Aggregate contains an illegal member type (e.g., a struct with a void member)";
142 return "The requested dynamic library could not be found";
144 return "The requested symbol was not found in the library";
146 return "Loading the dynamic library failed";
147 default:
148 return "An unknown or unspecified error occurred";
149 }
150}
151
165void _infix_set_error(infix_error_category_t category, infix_error_code_t code, size_t position) {
166 g_infix_last_error.category = category;
168 g_infix_last_error.position = position;
170 // Check if we can generate a rich parser error message.
171 if (category == INFIX_CATEGORY_PARSER && g_infix_last_signature_context != nullptr) {
172 // Generate a rich, GCC-style error message for parser failures.
173 const char * signature = g_infix_last_signature_context;
174 size_t sig_len = strlen(signature);
175 const size_t radius = 20; // Number of characters to show around the error position.
176 // Calculate the start and end of the snippet to display.
177 size_t start = (position > radius) ? (position - radius) : 0;
178 size_t end = (position + radius < sig_len) ? (position + radius) : sig_len;
179 // Add indicators if the snippet is truncated.
180 const char * start_indicator = (start > 0) ? "... " : "";
181 const char * end_indicator = (end < sig_len) ? " ..." : "";
182 size_t start_indicator_len = (start > 0) ? 4 : 0;
183 // Create the code snippet line.
184 char snippet[128];
185 snprintf(snippet,
186 sizeof(snippet),
187 "%s%.*s%s",
188 start_indicator,
189 (int)(end - start),
190 signature + start,
191 end_indicator);
192 // Create the pointer line with a caret '^' under the error.
193 char pointer[128];
194 size_t caret_pos = position - start + start_indicator_len;
195 snprintf(pointer, sizeof(pointer), "%*s^", (int)caret_pos, "");
196 // Build the final multi-line message piece by piece to avoid buffer overflows.
197 char * p = g_infix_last_error.message;
198 size_t remaining = sizeof(g_infix_last_error.message);
199 int written;
200 // Write the snippet and pointer lines.
201 written = snprintf(p, remaining, "\n\n %s\n %s", snippet, pointer);
202 if (written < 0 || (size_t)written >= remaining) {
203 // Fallback to a simple message on snprintf failure or buffer overflow.
204 const char * msg = _get_error_message_for_code(code);
206 return;
207 }
208 p += written;
209 remaining -= written;
210 // Append the standard error description.
211 snprintf(p, remaining, "\n\nError: %s", _get_error_message_for_code(code));
212 }
213 else {
214 // For non-parser errors, just copy the standard message.
215 const char * msg = _get_error_message_for_code(code);
217 }
218}
219
236 long system_code,
237 const char * msg) {
238 g_infix_last_error.category = category;
242 if (msg)
244 else {
245 const char * default_msg = _get_error_message_for_code(code);
247 }
248}
249
266
clock_t start
Definition 901_call_overhead.c:47
clock_t end
Definition 901_call_overhead.c:47
static const char * _get_error_message_for_code(infix_error_code_t code)
Definition error.c:107
void _infix_set_error(infix_error_category_t category, infix_error_code_t code, size_t position)
Sets the thread-local error state with detailed information.
Definition error.c:165
INFIX_TLS const char * g_infix_last_signature_context
A thread-local pointer to the full signature string being parsed.
Definition error.c:99
void _infix_set_system_error(infix_error_category_t category, infix_error_code_t code, long system_code, const char *msg)
Sets the thread-local error state for a system-level error.
Definition error.c:234
#define _INFIX_SAFE_STRNCPY(dest, src, count)
Definition error.c:75
void _infix_clear_error(void)
Clears the thread-local error state.
Definition error.c:258
static INFIX_TLS infix_error_details_t g_infix_last_error
The thread-local variable that stores the details of the last error.
Definition error.c:89
#define INFIX_TLS
Definition error.c:68
infix_error_code_t
Enumerates specific error codes.
Definition infix.h:1223
infix_error_details_t infix_get_last_error(void)
Retrieves detailed information about the last error that occurred on the current thread.
Definition error.c:271
infix_error_category_t
Enumerates the high-level categories of errors that can occur.
Definition infix.h:1213
@ INFIX_CODE_PROTECTION_FAILURE
Definition infix.h:1230
@ INFIX_CODE_SUCCESS
Definition infix.h:1225
@ INFIX_CODE_LIBRARY_NOT_FOUND
Definition infix.h:1246
@ INFIX_CODE_INVALID_MEMBER_TYPE
Definition infix.h:1243
@ INFIX_CODE_UNRESOLVED_NAMED_TYPE
Definition infix.h:1242
@ INFIX_CODE_INTEGER_OVERFLOW
Definition infix.h:1236
@ INFIX_CODE_TYPE_TOO_LARGE
Definition infix.h:1241
@ INFIX_CODE_UNEXPECTED_TOKEN
Definition infix.h:1232
@ INFIX_CODE_MISSING_RETURN_TYPE
Definition infix.h:1235
@ INFIX_CODE_EMPTY_MEMBER_NAME
Definition infix.h:1238
@ INFIX_CODE_EXECUTABLE_MEMORY_FAILURE
Definition infix.h:1229
@ INFIX_CODE_UNKNOWN
Definition infix.h:1226
@ INFIX_CODE_SYMBOL_NOT_FOUND
Definition infix.h:1247
@ INFIX_CODE_RECURSION_DEPTH_EXCEEDED
Definition infix.h:1237
@ INFIX_CODE_UNSUPPORTED_ABI
Definition infix.h:1240
@ INFIX_CODE_UNTERMINATED_AGGREGATE
Definition infix.h:1233
@ INFIX_CODE_LIBRARY_LOAD_FAILED
Definition infix.h:1248
@ INFIX_CODE_INVALID_KEYWORD
Definition infix.h:1234
@ INFIX_CODE_OUT_OF_MEMORY
Definition infix.h:1228
@ INFIX_CATEGORY_PARSER
Definition infix.h:1217
@ INFIX_CATEGORY_NONE
Definition infix.h:1214
size_t position
Definition infix.h:1257
infix_error_category_t category
Definition infix.h:1255
char message[256]
Definition infix.h:1259
long system_error_code
Definition infix.h:1258
infix_error_code_t code
Definition infix.h:1256
The public interface for the infix FFI library.
Internal data structures, function prototypes, and constants.
Provides detailed, thread-local information about the last error that occurred.
Definition infix.h:1254