src/lib/error.h
author Tero Marttila <terom@fixme.fi>
Wed, 27 May 2009 23:57:48 +0300
branchnew-lib-errors
changeset 217 7728d6ec3abf
parent 216 a10ba529ae39
child 218 5229a5d098b2
permissions -rw-r--r--
nexus.c compiles
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     1
#ifndef LIBQMSK_ERROR_H
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     2
#define LIBQMSK_ERROR_H
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     3
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     4
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     5
 * @file
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     6
 *
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     7
 * Support for error handling, using explicit function return codes and error info contexts.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     8
 *
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
     9
 * This is designed to support multiple "namespaces" of errors, which are entirely independent of eachother. These
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    10
 * namespaces can then reference one another, to build a 'tree' of useable error codes. Hence, each 'level' of this
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    11
 * tree has it's own set of error codes, and then error info contexts contain a stack of these error codes, which can
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    12
 * then be used to trace the error down.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    13
 */
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
    14
#include <stdbool.h>
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    15
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    16
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    17
 * The type used to represent a scalar error code, there are only unique per level.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    18
 *
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    19
 * Note that this type is signed to avoid nastyness with negating error codes, but all valid ERR_* codes are *always*
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    20
 * greater than zero in value.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    21
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    22
typedef signed short err_t;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    23
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    24
/** 
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    25
 * Standardized err_t code to mean no error, evaluates as logical false.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    26
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    27
static const err_t SUCCESS = 0;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    28
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    29
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    30
 * Extra information about an error, used to reference error codes in external libraries etc.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    31
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    32
union error_extra {
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    33
    /** (signed) integer code */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    34
    int int_;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    35
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    36
    /** Constant string */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    37
    const char *str;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    38
};
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    39
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    40
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    41
 * Errors can contain 'extra' information to reference things like error codes from external libraries; when these are
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    42
 * used, a 'type' must be specified to handle these.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    43
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    44
struct error_extra_type {
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    45
    /** Short name */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    46
    const char *name;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    47
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    48
    /** Conversion func, must return a static pointer */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    49
    const char * (*msg_func) (const struct error_extra_type *type, const union error_extra *extra);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    50
};
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    51
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    52
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    53
 * error_extra_type for libc errno values
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    54
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    55
const struct error_extra_type error_extra_errno;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    56
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    57
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    58
 * error_extra_type for string error values
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    59
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    60
const struct error_extra_type error_extra_string;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    61
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    62
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    63
 * Description of an error code
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    64
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    65
struct error_type {
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    66
    /** The actual code */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    67
    err_t code;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    68
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    69
    /** The short name of the error */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    70
    const char *name;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    71
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    72
    /** The type of any contained extra info */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    73
    const struct error_extra_type *extra_type;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    74
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    75
    /** Any linked error_code table for looking up sub-errors */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    76
    const struct error_list *sublist;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    77
};
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    78
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    79
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    80
 * Helper macros to define error_type's
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    81
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    82
#define ERROR_TYPE(code, name) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    83
    { (code), (name), NULL, NULL }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    84
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    85
#define ERROR_TYPE_ERRNO(code, name) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    86
    { (code), (name), &error_extra_errno, NULL }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    87
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    88
#define ERROR_TYPE_STRING(code, name) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    89
    { (code), (name), &error_extra_string, NULL }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    90
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    91
#define ERROR_TYPE_CUSTOM(code, name, type) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    92
    { (code), (name), (type), NULL }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    93
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    94
#define ERROR_TYPE_SUB(code, name, sub) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    95
    { (code), (name), NULL, (sub) }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    96
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    97
#define ERROR_TYPE_END \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    98
    { 0, NULL, NULL, NULL }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
    99
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   100
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   101
 * List of error types
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   102
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   103
struct error_list {
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   104
    /** Name of sublib */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   105
    const char *name;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   106
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   107
    /** The list */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   108
    struct error_type list[];
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   109
};
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   110
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   111
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   112
 * Helper macro to define an error_list
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   113
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   114
#define ERROR_LIST(name, ...) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   115
    { (name), __VA_ARGS__, ERROR_TYPE_END }
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   116
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   117
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   118
 * Maximum number of nesting levels supported for errors
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   119
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   120
#define ERROR_DEPTH_MAX 8
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   121
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   122
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   123
 * Information about an actual error
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   124
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   125
typedef struct error_state {
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   126
    /** The stack of error codes */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   127
    struct error_item {
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   128
        /** This level's error code */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   129
        err_t code;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   130
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   131
        /** Optional table for interpreting error_code, can be used instead of error_code::next */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   132
        const struct error_list *list;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   133
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   134
    } stack[ERROR_DEPTH_MAX];
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   135
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   136
    /** Current top of stack for accumulating errors, NULL means empty stack */
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   137
    struct error_item *cur;
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   138
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   139
    /** Type info for external error info */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   140
    const struct error_extra_type *extra_type;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   141
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   142
    /** Value of external error info */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   143
    union error_extra extra_value;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   144
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   145
} error_t;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   146
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   147
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   148
 * Look up the error_type for the given table and scalar code.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   149
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   150
const struct error_type* error_lookup_code (const struct error_list *list, err_t code);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   151
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   152
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   153
 * Look up the error_type for for the given root table and error_info.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   154
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   155
const struct error_type* error_lookup (const error_t *err);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   156
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   157
/**
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   158
 * Return the error name for the given code.
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   159
 */
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   160
const char* error_name (const struct error_list *list, err_t code);
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   161
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   162
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   163
 * Maximum length of messages returned by error_msg
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   164
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   165
#define ERROR_MSG_MAX 1024
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   166
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   167
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   168
 * Return a pointer to a statically allocated string describing the given full error
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   169
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   170
const char* error_msg (const error_t *err);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   171
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   172
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   173
 * Compare the given errors for equivalency
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   174
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   175
bool error_cmp_eq (const error_t *a, const error_t *b);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   176
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   177
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   178
/**
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   179
 * Reset the given error state to an empty state.
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   180
 *
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   181
 * This is just the same as memset() with zero.
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   182
 */
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   183
void error_reset (error_t *err);
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   184
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   185
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   186
 * Evaluates to the current top of the error stack, 
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   187
 */
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   188
static struct error_item* error_top (error_t *err)
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   189
{
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   190
    return err->cur ? err->cur : err->stack;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   191
}
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   192
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   193
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   194
 * Evaluate to the most recent error code pushed, or SUCCESS if none
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   195
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   196
static inline err_t error_code (error_t *err)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   197
{
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   198
    return err->cur ? err->cur->code : SUCCESS;
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   199
}
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   200
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   201
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   202
 * Reset the error state, push the given initial err.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   203
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   204
void error_set (error_t *err, const struct error_list *list, err_t code);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   205
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   206
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   207
 * Reset, push err, and set extra.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   208
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   209
void error_set_extra (error_t *err, const struct error_list *list, err_t code, const struct error_extra_type *type, union error_extra extra);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   210
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   211
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   212
 * Push an additional level of error info onto the error state's stack. If it doesn't fit, XXX: it is ignored.
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   213
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   214
void error_push (error_t *err, const struct error_list *list, err_t code);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   215
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   216
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   217
 * Copy value of error
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   218
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   219
void error_copy (error_t *dst, const error_t *src);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   220
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   221
/** Evalutes to the lvalue representing the error code at the top of the stack */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   222
#define ERROR_CODE(err_state) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   223
    (error_top(err_state)->code)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   224
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   225
/** Push a new error code onto the given state */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   226
#define PUSH_ERROR(err_state, err_list, err_code) ({ \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   227
        error_push(err_state, err_list, err_code); \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   228
        err_code; })
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   229
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   230
/** Set the error to the given value */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   231
#define SET_ERROR(err_state, err_list, err_code) ({ \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   232
        error_set(err_state, err_list, err_code); \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   233
        err_code; })
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   234
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   235
/* Helper macro to use error_set_extra */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   236
#define _ERROR_SET_EXTRA(err_state, err_list, err_code, err_extra_type, _err_extra_field_, err_extra_value) \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   237
        error_set_extra(err_state, err_list, err_code, err_extra_type, ((union error_extra) { ._err_extra_field_ = (err_extra_value)}))
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   238
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   239
/** Set the error with extra info as integer */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   240
#define SET_ERROR_EXTRA(err_state, err_list, err_code, err_extra_type, err_extra_int) ({ \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   241
        _ERROR_SET_EXTRA(err_state, err_list, err_code, err_extra_type, int_, err_extra);  \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   242
        err_code; })
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   243
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   244
/** Set the error with extra info as the libc errno */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   245
#define SET_ERROR_ERRNO(err_state, err_list, err_code) ({ \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   246
        _ERROR_SET_EXTRA(err_state, err_list, err_code, &error_extra_errno, int_, errno); \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   247
        err_code; })
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   248
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   249
/** Set the error with extra info as the given static string */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   250
#define SET_ERROR_STR(err_state, err_list, err_code, err_str) ({ \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   251
        _ERROR_SET_EXTRA(err_state, err_list, err_code, &error_extra_string, str, err_str); \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   252
        err_code; })
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   253
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   254
/** Set the error with the contents of the given error */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   255
#define SET_ERROR_COPY(err_state_dst, err_state_src) ({ \
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   256
        error_code(err_state_dst); })
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   257
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   258
/** Same as above, but also do a 'goto error' jump */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   259
#define JUMP_PUSH_ERROR(...)        do { PUSH_ERROR(__VA_ARGS__);       goto error; } while (0)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   260
#define JUMP_SET_ERROR(...)         do { SET_ERROR(__VA_ARGS__);        goto error; } while (0)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   261
#define JUMP_SET_ERROR_EXTRA(...)   do { SET_ERROR_EXTRA(__VA_ARGS__);  goto error; } while (0)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   262
#define JUMP_SET_ERROR_ERRNO(...)   do { SET_ERROR_ERRNO(__VA_ARGS__);  goto error; } while (0)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   263
#define JUMP_SET_ERROR_STR(...)     do { SET_ERROR_STR(__VA_ARGS__);    goto error; } while (0)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   264
#define JUMP_SET_ERROR_INFO(...)    do { SET_ERROR_INFO(__VA_ARGS__);   goto error; } while (0)
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   265
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   266
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   267
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   268
 * Abort execution of process with error message
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   269
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   270
void _error_abort (const char *file, const char *line, const char *func, const char *fmt, ...);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   271
#define error_abort(...) _error_abort(__FILE__, __LINE__, __func__, __VA_ARGS__);
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   272
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   273
/**
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   274
 * Used to mark unexpcted conditions for switch-default. The given val must be an integer, as passed to switch
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   275
 */
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   276
#define NOT_REACHED(val) error_abort("%s = %#x", #val, (int) (val));
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   277
217
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   278
/**
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   279
 * General-purpose errors that may be useful and don't belong in any more specific namespace.
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   280
 */
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   281
enum general_error_code {
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   282
    ERR_GENERAL_NONE,
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   283
    ERR_MEM,                ///< memory allocation error
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   284
    ERR_NOT_IMPLEMENTED,    ///< function not implmented: <func>
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   285
    ERR_MISC,               ///< miscellaneous error: <error>
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   286
    ERR_CMD_OPT,            ///< invalid command line option: <error> - XXX: replace with something getopt
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   287
    ERR_UNKNOWN,            ///< unknown error
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   288
};
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   289
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   290
const struct error_list general_errors;
7728d6ec3abf nexus.c compiles
Tero Marttila <terom@fixme.fi>
parents: 216
diff changeset
   291
216
a10ba529ae39 initial error code
Tero Marttila <terom@fixme.fi>
parents:
diff changeset
   292
#endif /* LIBQMSK_ERROR_H */