2007-12-30 17:41:49 +01:00
|
|
|
/***************************************************************************
|
|
|
|
|
2009-08-17 12:41:51 +02:00
|
|
|
gb_error.h
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2009-08-17 12:41:51 +02:00
|
|
|
(c) 2000-2009 Benoît Minisini <gambas@users.sourceforge.net>
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
2009-08-17 12:41:51 +02:00
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
2007-12-30 17:41:49 +01:00
|
|
|
any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2009-08-17 12:41:51 +02:00
|
|
|
GNU General Public License for more details.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2009-08-17 12:41:51 +02:00
|
|
|
You should have received a copy of the GNU General Public License
|
2007-12-30 17:41:49 +01:00
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#ifndef __GB_ERROR_H
|
|
|
|
#define __GB_ERROR_H
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <setjmp.h>
|
|
|
|
|
2010-05-25 13:19:00 +02:00
|
|
|
#include "gb_common.h"
|
2007-12-30 17:41:49 +01:00
|
|
|
#include "gb_limit.h"
|
|
|
|
|
2009-02-04 23:51:20 +01:00
|
|
|
//#define DEBUG_ERROR 1
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
enum {
|
|
|
|
E_ABORT = -2,
|
|
|
|
E_CUSTOM = -1,
|
|
|
|
E_UNKNOWN = 0,
|
|
|
|
E_MEMORY,
|
|
|
|
E_CLASS,
|
|
|
|
E_STACK,
|
|
|
|
E_NEPARAM,
|
|
|
|
E_TMPARAM,
|
|
|
|
E_TYPE,
|
|
|
|
E_OVERFLOW,
|
|
|
|
E_ILLEGAL,
|
|
|
|
E_NFUNC,
|
|
|
|
E_CSTATIC,
|
|
|
|
E_NSYMBOL,
|
|
|
|
E_NOBJECT,
|
|
|
|
E_NULL,
|
|
|
|
E_STATIC,
|
|
|
|
E_NREAD,
|
|
|
|
E_NWRITE,
|
|
|
|
E_NPROPERTY,
|
|
|
|
E_NRETURN,
|
|
|
|
E_MATH,
|
|
|
|
E_ARG,
|
|
|
|
E_BOUND,
|
|
|
|
E_NDIM,
|
|
|
|
E_NARRAY,
|
|
|
|
E_MAIN,
|
|
|
|
E_NNEW,
|
|
|
|
E_ZERO,
|
|
|
|
E_LIBRARY,
|
|
|
|
E_EVENT,
|
|
|
|
E_IOBJECT,
|
|
|
|
E_ENUM,
|
|
|
|
E_UCONV,
|
|
|
|
E_CONV,
|
|
|
|
E_DATE,
|
|
|
|
E_BADPATH,
|
|
|
|
E_OPEN,
|
|
|
|
E_PROJECT,
|
|
|
|
E_FULL,
|
|
|
|
E_EXIST,
|
|
|
|
E_EOF,
|
|
|
|
E_FORMAT,
|
|
|
|
E_DYNAMIC,
|
|
|
|
E_SYSTEM,
|
|
|
|
E_ACCESS,
|
|
|
|
E_TOOLONG,
|
|
|
|
E_NEXIST,
|
|
|
|
E_DIR,
|
|
|
|
E_READ,
|
|
|
|
E_WRITE,
|
|
|
|
E_NDIR,
|
|
|
|
E_REGEXP,
|
|
|
|
E_ARCH,
|
|
|
|
E_REGISTER,
|
|
|
|
E_CLOSED,
|
|
|
|
E_VIRTUAL,
|
|
|
|
E_STOP,
|
|
|
|
E_STRING,
|
|
|
|
E_EVAL,
|
|
|
|
E_LOCK,
|
|
|
|
E_PARENT,
|
|
|
|
E_EXTLIB,
|
|
|
|
E_EXTSYM,
|
2008-01-19 18:52:05 +01:00
|
|
|
E_BYREF,
|
2009-09-17 22:58:27 +02:00
|
|
|
E_OVERRIDE,
|
2010-05-19 14:43:57 +02:00
|
|
|
E_VKEY,
|
2010-07-14 03:50:18 +02:00
|
|
|
E_SARRAY,
|
|
|
|
E_EXTCB
|
2007-12-30 17:41:49 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
typedef
|
|
|
|
struct {
|
2008-01-25 16:01:02 +01:00
|
|
|
short code;
|
|
|
|
bool free;
|
2007-12-30 17:41:49 +01:00
|
|
|
void *cp;
|
|
|
|
void *fp;
|
|
|
|
void *pc;
|
2008-01-25 16:01:02 +01:00
|
|
|
char *msg;
|
2010-05-28 09:47:49 +02:00
|
|
|
int bt_count;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
ERROR_INFO;
|
|
|
|
|
|
|
|
typedef
|
2010-06-18 02:14:53 +02:00
|
|
|
struct _ERROR_CONTEXT {
|
|
|
|
struct _ERROR_CONTEXT *prev;
|
|
|
|
jmp_buf env;
|
|
|
|
int ret;
|
2010-08-03 13:00:42 +02:00
|
|
|
struct _ERROR_HANDLER *handler;
|
2010-06-18 02:14:53 +02:00
|
|
|
ERROR_INFO info;
|
|
|
|
}
|
|
|
|
ERROR_CONTEXT;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-06-18 02:14:53 +02:00
|
|
|
typedef
|
|
|
|
struct _ERROR_HANDLER {
|
|
|
|
struct _ERROR_HANDLER *prev;
|
|
|
|
ERROR_CONTEXT *context;
|
|
|
|
void (*handler)();
|
|
|
|
}
|
|
|
|
ERROR_HANDLER;
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
#ifndef __GB_ERROR_C
|
2008-01-25 16:01:02 +01:00
|
|
|
EXTERN ERROR_CONTEXT *ERROR_current;
|
2008-01-27 15:00:04 +01:00
|
|
|
EXTERN ERROR_INFO ERROR_last;
|
2010-06-18 02:14:53 +02:00
|
|
|
EXTERN ERROR_HANDLER *ERROR_handler;
|
2007-12-30 17:41:49 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define ERROR_LEAVE_DONE ((ERROR_CONTEXT *)-1)
|
|
|
|
|
2009-02-04 23:51:20 +01:00
|
|
|
#if DEBUG_ERROR
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
#define TRY \
|
|
|
|
{ \
|
|
|
|
ERROR_CONTEXT __err_context; \
|
|
|
|
{ \
|
|
|
|
ERROR_CONTEXT *__err = &__err_context; \
|
2009-02-04 23:51:20 +01:00
|
|
|
fprintf(stderr, "TRY %s\n", __FUNCTION__); \
|
2007-12-30 17:41:49 +01:00
|
|
|
ERROR_enter(__err); \
|
2008-01-25 16:01:02 +01:00
|
|
|
__err->ret = setjmp(__err->env); \
|
|
|
|
if (__err->ret == 0)
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2008-01-25 16:01:02 +01:00
|
|
|
/*#define CATCH \
|
|
|
|
fprintf(stderr, "%p == %p ? %d\n", ERROR_current, __err, __err->ret); \
|
|
|
|
if (__err->ret != 0 && (__err->ret = 2))*/
|
2009-02-04 23:51:20 +01:00
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
#define CATCH \
|
2009-02-04 23:51:20 +01:00
|
|
|
if (__err->ret) fprintf(stderr, "CATCH %p\n", __err); \
|
|
|
|
if (__err->ret)
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
#define END_TRY \
|
2009-02-04 23:51:20 +01:00
|
|
|
fprintf(stderr, "END TRY %s\n", __FUNCTION__); \
|
2008-01-25 16:01:02 +01:00
|
|
|
ERROR_leave(__err); \
|
2007-12-30 17:41:49 +01:00
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
2009-02-04 23:51:20 +01:00
|
|
|
#define PROPAGATE() fprintf(stderr, "PROPAGATE %s\n", __FUNCTION__), ERROR_propagate()
|
|
|
|
|
|
|
|
#else /* DEBUG_ERROR */
|
|
|
|
|
|
|
|
#define TRY \
|
|
|
|
{ \
|
|
|
|
ERROR_CONTEXT __err_context; \
|
|
|
|
{ \
|
|
|
|
ERROR_CONTEXT *__err = &__err_context; \
|
|
|
|
ERROR_enter(__err); \
|
|
|
|
__err->ret = setjmp(__err->env); \
|
|
|
|
if (__err->ret == 0)
|
|
|
|
|
|
|
|
#define CATCH \
|
|
|
|
else
|
|
|
|
|
|
|
|
#define END_TRY \
|
|
|
|
ERROR_leave(__err); \
|
|
|
|
} \
|
|
|
|
}
|
2008-01-25 16:01:02 +01:00
|
|
|
|
|
|
|
#define PROPAGATE() ERROR_propagate()
|
2009-02-04 23:51:20 +01:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2010-06-18 02:14:53 +02:00
|
|
|
#define ON_ERROR(_handler) \
|
|
|
|
{ \
|
|
|
|
ERROR_HANDLER __handler; \
|
|
|
|
__handler.handler = (_handler); \
|
|
|
|
__handler.prev = ERROR_handler; \
|
|
|
|
__handler.context = ERROR_current; \
|
|
|
|
ERROR_handler = &__handler;
|
2010-08-03 13:00:42 +02:00
|
|
|
//fprintf(stderr, "%s.%d: ERROR_handler -> %p @ %p\n", __FUNCTION__, __LINE__, ERROR_handler, ERROR_current);
|
2010-06-18 02:14:53 +02:00
|
|
|
|
|
|
|
#define END_ERROR \
|
|
|
|
ERROR_handler = __handler.prev; \
|
|
|
|
}
|
2010-08-03 13:00:42 +02:00
|
|
|
//fprintf(stderr, "%s.%d: ERROR_handler <- %p @ %p (%p)\n", __FUNCTION__, __LINE__, ERROR_handler, ERROR_handler ? ERROR_handler->context : NULL, ERROR_current);
|
2009-02-04 23:51:20 +01:00
|
|
|
|
|
|
|
#define ERROR __err
|
2008-01-25 16:01:02 +01:00
|
|
|
|
2008-01-27 15:00:04 +01:00
|
|
|
const char *ERROR_get(void);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-05-22 20:02:34 +02:00
|
|
|
//void ERROR_enter(ERROR_CONTEXT *err);
|
|
|
|
|
|
|
|
#define ERROR_enter(_err) \
|
|
|
|
do { \
|
|
|
|
_err->prev = ERROR_current; \
|
|
|
|
_err->info.code = 0; \
|
2010-08-03 13:00:42 +02:00
|
|
|
_err->handler = ERROR_handler; \
|
2010-05-22 20:02:34 +02:00
|
|
|
ERROR_current = _err; \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
|
|
|
|
//void ERROR_leave(ERROR_CONTEXT *err);
|
|
|
|
|
|
|
|
#define ERROR_leave(_err) \
|
|
|
|
do { \
|
2010-05-25 13:19:00 +02:00
|
|
|
ERROR_CONTEXT *_prev = (_err); \
|
|
|
|
if (_prev->prev != ERROR_LEAVE_DONE) \
|
2010-05-22 20:02:34 +02:00
|
|
|
{ \
|
2010-05-25 13:19:00 +02:00
|
|
|
ERROR_current = _prev->prev; \
|
2010-05-22 20:02:34 +02:00
|
|
|
if (ERROR_current) \
|
|
|
|
{ \
|
2010-05-25 13:19:00 +02:00
|
|
|
if (_prev->info.code) \
|
2010-05-22 20:02:34 +02:00
|
|
|
{ \
|
|
|
|
ERROR_reset(&ERROR_current->info); \
|
2010-05-25 13:19:00 +02:00
|
|
|
ERROR_current->info = _prev->info; \
|
2010-05-22 20:02:34 +02:00
|
|
|
} \
|
|
|
|
} \
|
|
|
|
else \
|
2010-05-25 13:19:00 +02:00
|
|
|
ERROR_reset(&_prev->info); \
|
|
|
|
_prev->prev = ERROR_LEAVE_DONE; \
|
2010-05-22 20:02:34 +02:00
|
|
|
} \
|
|
|
|
} while(0)
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2008-01-27 15:00:04 +01:00
|
|
|
void ERROR_define(const char *pattern, char *arg[]);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-05-27 00:01:28 +02:00
|
|
|
void ERROR_propagate(void) NORETURN;
|
2008-01-27 15:00:04 +01:00
|
|
|
void THROW(int code, ...) NORETURN;
|
|
|
|
void THROW_SYSTEM(int err, const char *path);
|
2010-05-27 00:01:28 +02:00
|
|
|
void THROW_ILLEGAL(void) NORETURN;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2008-01-27 15:00:04 +01:00
|
|
|
void ERROR_panic(const char *error, ...) NORETURN;
|
2010-08-28 14:33:05 +02:00
|
|
|
void ERROR_warning(const char *warning, ...);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2008-01-27 15:00:04 +01:00
|
|
|
void ERROR_print(void);
|
|
|
|
void ERROR_print_at(FILE *where, bool msgonly, bool newline);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2008-01-27 15:00:04 +01:00
|
|
|
void ERROR_save(ERROR_INFO *save);
|
|
|
|
void ERROR_restore(ERROR_INFO *save);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-05-27 00:01:28 +02:00
|
|
|
void ERROR_clear(void);
|
2008-01-27 15:00:04 +01:00
|
|
|
void ERROR_reset(ERROR_INFO *info);
|
2010-05-27 00:01:28 +02:00
|
|
|
void ERROR_lock(void);
|
|
|
|
void ERROR_unlock(void);
|
|
|
|
void ERROR_set_last(void);
|
2010-01-25 18:53:58 +01:00
|
|
|
void ERROR_deprecated(const char *msg);
|
2008-01-27 15:00:04 +01:00
|
|
|
|
|
|
|
#define ERROR_exit() ERROR_reset(&ERROR_last)
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
#endif
|