diff options
Diffstat (limited to 'core/error_macros.h')
-rw-r--r-- | core/error_macros.h | 222 |
1 files changed, 222 insertions, 0 deletions
diff --git a/core/error_macros.h b/core/error_macros.h new file mode 100644 index 0000000000..6cd38755c0 --- /dev/null +++ b/core/error_macros.h @@ -0,0 +1,222 @@ +/*************************************************************************/ +/* error_macros.h */ +/*************************************************************************/ +/* This file is part of: */ +/* GODOT ENGINE */ +/* http://www.godotengine.org */ +/*************************************************************************/ +/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */ +/* */ +/* Permission is hereby granted, free of charge, to any person obtaining */ +/* a copy of this software and associated documentation files (the */ +/* "Software"), to deal in the Software without restriction, including */ +/* without limitation the rights to use, copy, modify, merge, publish, */ +/* distribute, sublicense, and/or sell copies of the Software, and to */ +/* permit persons to whom the Software is furnished to do so, subject to */ +/* the following conditions: */ +/* */ +/* The above copyright notice and this permission notice shall be */ +/* included in all copies or substantial portions of the Software. */ +/* */ +/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ +/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ +/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/ +/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ +/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ +/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ +/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ +/*************************************************************************/ +#ifndef ERROR_MACROS_H +#define ERROR_MACROS_H + + +/** + * Error macros. Unlike exceptions and asserts, these macros try to mantain consistency and stability + * inside the code. It is recommended to always return processable data, so in case of an error, the + * engine can stay working well. + * In most cases, bugs and/or invalid data are not fatal and should never allow a perfectly running application + * to fail or crash. + */ + +/** + * Pointer to the error macro priting function. Reassign to any function to have errors printed + */ + +/** Function used by the error macros */ + +// function, file, line, error, explanation + +enum ErrorHandlerType { + ERR_HANDLER_ERROR, + ERR_HANDLER_WARNING, + ERR_HANDLER_SCRIPT +}; + +typedef void (*ErrorHandlerFunc)(void*,const char*,const char*,int p_line,const char *, const char *,ErrorHandlerType p_type); +void _err_set_last_error(const char* p_err); +void _err_clear_last_error(); + +struct ErrorHandlerList { + + ErrorHandlerFunc errfunc; + void *userdata; + + ErrorHandlerList*next; + + ErrorHandlerList() { errfunc=0; next=0; userdata=0; } +}; + +void add_error_handler(ErrorHandlerList *p_handler); +void remove_error_handler(ErrorHandlerList *p_handler); + +void _err_print_error(const char* p_function,const char* p_file,int p_line,const char *p_error,ErrorHandlerType p_type=ERR_HANDLER_ERROR); + +#ifndef _STR +#define _STR(m_x) #m_x +#define _MKSTR(m_x) _STR(m_x) +#endif + +#define _FNL __FILE__":" + +/** An index has failed if m_index<0 or m_index >=m_size, the function exists */ + +extern bool _err_error_exists; + +#ifdef DEBUG_ENABLED +/** Print a warning string. + */ +#define ERR_EXPLAINC(m_reason) {_err_set_last_error(m_reason); _err_error_exists=true;} +#define ERR_EXPLAIN(m_string) {_err_set_last_error(String(m_string).utf8().get_data()); _err_error_exists=true;} + +#else + +#define ERR_EXPLAIN( m_text ) +#define ERR_EXPLAINC( m_text ) + +#endif + +#ifdef __GNUC__ +//#define FUNCTION_STR __PRETTY_FUNCTION__ - too annoying +#define FUNCTION_STR __FUNCTION__ +#else +#define FUNCTION_STR __FUNCTION__ +#endif + +#define ERR_FAIL_INDEX(m_index,m_size) \ + do {if ((m_index)<0 || (m_index)>=(m_size)) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Index "_STR(m_index)" out of size ("_STR(m_size)")."); \ + return; \ + } else _err_error_exists=false; } while(0); \ + +/** An index has failed if m_index<0 or m_index >=m_size, the function exists. + * This function returns an error value, if returning Error, please select the most + * appropriate error condition from error_macros.h + */ + +#define ERR_FAIL_INDEX_V(m_index,m_size,m_retval) \ + do {if ((m_index)<0 || (m_index)>=(m_size)) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Index "_STR(m_index)" out of size ("_STR(m_size)")."); \ + return m_retval; \ + } else _err_error_exists=false;} while (0); + + /** An error condition happened (m_cond tested true) (WARNING this is the opposite as assert(). + * the function will exit. + */ + + #define ERR_FAIL_NULL(m_param) \ + { if ( !m_param ) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Parameter ' "_STR(m_param)" ' is null."); \ + return; \ + }else _err_error_exists=false; } \ + + +#define ERR_FAIL_NULL_V(m_param,m_retval) \ + { if ( !m_param ) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Parameter ' "_STR(m_param)" ' is null."); \ + return m_retval; \ + }else _err_error_exists=false; } \ + +/** An error condition happened (m_cond tested true) (WARNING this is the opposite as assert(). + * the function will exit. + */ + +#define ERR_FAIL_COND(m_cond) \ + { if ( m_cond ) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Condition ' "_STR(m_cond)" ' is true."); \ + return; \ + }else _err_error_exists=false; } \ + +/** An error condition happened (m_cond tested true) (WARNING this is the opposite as assert(). + * the function will exit. + * This function returns an error value, if returning Error, please select the most + * appropriate error condition from error_macros.h + */ + +#define ERR_FAIL_COND_V(m_cond,m_retval) \ + { if ( m_cond ) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Condition ' "_STR(m_cond)" ' is true. returned: "_STR(m_retval)); \ + return m_retval; \ + }else _err_error_exists=false; } \ + +/** An error condition happened (m_cond tested true) (WARNING this is the opposite as assert(). + * the loop will skip to the next iteration. + */ + +#define ERR_CONTINUE(m_cond) \ + { if ( m_cond ) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Condition ' "_STR(m_cond)" ' is true. Continuing..:"); \ + continue;\ + } else _err_error_exists=false;} \ + +/** An error condition happened (m_cond tested true) (WARNING this is the opposite as assert(). + * the loop will break + */ + +#define ERR_BREAK(m_cond) \ + { if ( m_cond ) { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Condition ' "_STR(m_cond)" ' is true. Breaking..:"); \ + break;\ + } else _err_error_exists=false;} \ + +/** Print an error string and return + */ + +#define ERR_FAIL() \ +{ \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Method/Function Failed."); \ + _err_error_exists=false;\ + return;\ +} \ + +/** Print an error string and return with value + */ + +#define ERR_FAIL_V(m_value) \ +{ \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,"Method/Function Failed, returning: "__STR(m_value)); \ + _err_error_exists=false; \ + return m_value;\ +} \ + +/** Print an error string. + */ + +#define ERR_PRINT(m_string) \ + { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,m_string); \ + _err_error_exists=false;\ + } \ + + +/** Print a warning string. + */ + +#define WARN_PRINT(m_string) \ + { \ + _err_print_error(FUNCTION_STR,__FILE__,__LINE__,m_string,ERR_HANDLER_WARNING); \ + _err_error_exists=false;\ + } \ + + + +#endif |