2026-01-04 22:56:31 -05:00
|
|
|
#include "akerror.h"
|
2026-01-10 10:20:35 -05:00
|
|
|
#if defined(AKERR_USE_STDLIB) && AKERR_USE_STDLIB == 1
|
2026-01-04 22:56:31 -05:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
#include <stdio.h>
|
2026-01-10 10:20:35 -05:00
|
|
|
#endif // AKERR_USE_STDLIB
|
2025-07-20 21:40:14 -04:00
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_ErrorContext __akerr_last_ditch;
|
|
|
|
|
akerr_ErrorContext *__akerr_last_ignored;
|
|
|
|
|
akerr_ErrorUnhandledErrorHandler akerr_handler_unhandled_error;
|
|
|
|
|
akerr_ErrorLogFunction akerr_log_method = NULL;
|
2025-07-20 21:40:14 -04:00
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
char __AKERR_ERROR_NAMES[AKERR_MAX_ERR_VALUE+1][AKERR_MAX_ERROR_NAME_LENGTH];
|
2025-07-20 21:40:14 -04:00
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_ErrorContext AKERR_ARRAY_ERROR[AKERR_MAX_ARRAY_ERROR];
|
2025-07-20 21:40:14 -04:00
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
void akerr_default_logger(const char *fmt, ...)
|
2026-01-04 22:56:31 -05:00
|
|
|
{
|
2026-01-10 10:20:35 -05:00
|
|
|
#if defined(AKERR_USE_STDLIB) && AKERR_USE_STDLIB == 1
|
2026-01-04 22:56:31 -05:00
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
|
|
va_start(ap, fmt);
|
|
|
|
|
vfprintf(stderr, fmt, ap);
|
|
|
|
|
va_end(ap);
|
|
|
|
|
#else
|
|
|
|
|
return;
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
void akerr_init()
|
2025-07-20 21:40:14 -04:00
|
|
|
{
|
|
|
|
|
static int inited = 0;
|
|
|
|
|
if ( inited == 0 ) {
|
2026-01-10 10:20:35 -05:00
|
|
|
for (int i = 0; i < AKERR_MAX_ARRAY_ERROR; i++ ) {
|
|
|
|
|
memset((void *)&AKERR_ARRAY_ERROR[i], 0x00, sizeof(akerr_ErrorContext));
|
|
|
|
|
AKERR_ARRAY_ERROR[i].arrayid = i;
|
|
|
|
|
AKERR_ARRAY_ERROR[i].stacktracebufptr = (char *)&AKERR_ARRAY_ERROR[i].stacktracebuf;
|
2025-07-20 21:40:14 -04:00
|
|
|
}
|
2026-01-10 10:20:35 -05:00
|
|
|
__akerr_last_ignored = NULL;
|
|
|
|
|
memset((void *)&__akerr_last_ditch, 0x00, sizeof(akerr_ErrorContext));
|
|
|
|
|
__akerr_last_ditch.stacktracebufptr = (char *)&__akerr_last_ditch.stacktracebuf;
|
|
|
|
|
if ( akerr_log_method == NULL ) {
|
|
|
|
|
akerr_log_method = &akerr_default_logger;
|
2026-01-04 22:56:31 -05:00
|
|
|
}
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_handler_unhandled_error = &akerr_default_handler_unhandled_error;
|
|
|
|
|
memset((void *)&__AKERR_ERROR_NAMES[0], 0x00, ((AKERR_MAX_ERR_VALUE+1) * AKERR_MAX_ERROR_NAME_LENGTH));
|
2025-08-02 15:07:08 -04:00
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_name_for_status(AKERR_NULLPOINTER, "Null Pointer Error");
|
|
|
|
|
akerr_name_for_status(AKERR_OUTOFBOUNDS, "Out Of Bounds Error");
|
|
|
|
|
akerr_name_for_status(AKERR_API, "API Error");
|
|
|
|
|
akerr_name_for_status(AKERR_ATTRIBUTE, "Attribute Error");
|
|
|
|
|
akerr_name_for_status(AKERR_TYPE, "Type Error");
|
|
|
|
|
akerr_name_for_status(AKERR_KEY, "Key Error");
|
|
|
|
|
akerr_name_for_status(AKERR_HEAP, "Heap Error");
|
|
|
|
|
akerr_name_for_status(AKERR_INDEX, "Index Error");
|
|
|
|
|
akerr_name_for_status(AKERR_FORMAT, "Format Error");
|
|
|
|
|
akerr_name_for_status(AKERR_IO, "Input Output Error");
|
|
|
|
|
akerr_name_for_status(AKERR_REGISTRY, "Registry Error");
|
|
|
|
|
akerr_name_for_status(AKERR_VALUE, "Value Error");
|
|
|
|
|
akerr_name_for_status(AKERR_BEHAVIOR, "Behavior Error");
|
|
|
|
|
akerr_name_for_status(AKERR_RELATIONSHIP, "Relationship Error");
|
2025-08-02 15:07:08 -04:00
|
|
|
|
2025-07-20 21:40:14 -04:00
|
|
|
inited = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
void akerr_default_handler_unhandled_error(akerr_ErrorContext *errctx)
|
2025-07-20 21:40:14 -04:00
|
|
|
{
|
|
|
|
|
if ( errctx == NULL ) {
|
|
|
|
|
exit(1);
|
|
|
|
|
}
|
|
|
|
|
exit(errctx->status);
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_ErrorContext *akerr_next_error()
|
2025-07-20 21:40:14 -04:00
|
|
|
{
|
2026-01-10 10:20:35 -05:00
|
|
|
for (int i = 0; i < AKERR_MAX_ARRAY_ERROR; i++ ) {
|
|
|
|
|
if ( AKERR_ARRAY_ERROR[i].refcount == 0 ) {
|
|
|
|
|
return &AKERR_ARRAY_ERROR[i];
|
2025-07-20 21:40:14 -04:00
|
|
|
}
|
|
|
|
|
}
|
2026-01-10 10:20:35 -05:00
|
|
|
return (akerr_ErrorContext *)NULL;
|
2025-07-20 21:40:14 -04:00
|
|
|
}
|
|
|
|
|
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_ErrorContext *akerr_release_error(akerr_ErrorContext *err)
|
2025-07-20 21:40:14 -04:00
|
|
|
{
|
|
|
|
|
int oldid = 0;
|
|
|
|
|
if ( err == NULL ) {
|
2026-01-10 10:20:35 -05:00
|
|
|
akerr_ErrorContext *errctx = &__akerr_last_ditch;
|
|
|
|
|
FAIL_RETURN(errctx, AKERR_NULLPOINTER, "akerr_release_error got NULL context pointer");
|
2025-07-20 21:40:14 -04:00
|
|
|
}
|
|
|
|
|
if ( err->refcount > 0 ) {
|
|
|
|
|
err->refcount -= 1;
|
|
|
|
|
}
|
|
|
|
|
if ( err->refcount == 0 ) {
|
2026-01-10 09:50:17 -05:00
|
|
|
oldid = err->arrayid;
|
2026-01-10 10:20:35 -05:00
|
|
|
memset(err, 0x00, sizeof(akerr_ErrorContext));
|
2025-07-20 21:40:14 -04:00
|
|
|
err->stacktracebufptr = (char *)&err->stacktracebuf;
|
2026-01-10 09:50:17 -05:00
|
|
|
err->arrayid = oldid;
|
2025-07-20 21:40:14 -04:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-02 15:07:08 -04:00
|
|
|
|
|
|
|
|
// returns or sets the name for the given status.
|
|
|
|
|
// Call with name = NULL to retrieve a status.
|
2026-01-10 10:20:35 -05:00
|
|
|
char *akerr_name_for_status(int status, char *name)
|
2025-07-20 21:40:14 -04:00
|
|
|
{
|
2026-01-10 10:20:35 -05:00
|
|
|
if ( status > AKERR_MAX_ERR_VALUE ) {
|
2025-08-02 15:07:08 -04:00
|
|
|
return "Unknown Error";
|
|
|
|
|
}
|
|
|
|
|
if ( name != NULL ) {
|
2026-01-10 10:20:35 -05:00
|
|
|
strncpy((char *)&__AKERR_ERROR_NAMES[status], name, AKERR_MAX_ERROR_NAME_LENGTH);
|
2025-08-02 15:07:08 -04:00
|
|
|
}
|
2026-01-10 10:20:35 -05:00
|
|
|
return (char *)&__AKERR_ERROR_NAMES[status];
|
2025-07-20 21:40:14 -04:00
|
|
|
}
|