exception.h 4.6 KB
Newer Older
weixin_48148422's avatar
weixin_48148422 已提交
1
/*
weixin_48148422's avatar
weixin_48148422 已提交
2
 * Copyright (c) 2020 TAOS Data, Inc. <jhtao@taosdata.com>
weixin_48148422's avatar
weixin_48148422 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
 *
 * This program is free software: you can use, redistribute, and/or modify
 * it under the terms of the GNU Affero General Public License, version 3
 * or later ("AGPL"), as published by the Free Software Foundation.
 *
 * 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.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */

#ifndef TDENGINE_EXCEPTION_H
#define TDENGINE_EXCEPTION_H

#include <setjmp.h>
weixin_48148422's avatar
weixin_48148422 已提交
20
#include <stdint.h>
weixin_48148422's avatar
weixin_48148422 已提交
21
#include <stdbool.h>
weixin_48148422's avatar
weixin_48148422 已提交
22
#include <assert.h>
weixin_48148422's avatar
weixin_48148422 已提交
23 24 25 26 27

#ifdef __cplusplus
extern "C" {
#endif

weixin_48148422's avatar
weixin_48148422 已提交
28
/*
29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
 * cleanup actions
 */
typedef struct SCleanupAction {
    bool failOnly;
    uint8_t wrapper;
    uint16_t reserved;
    void* func;
    union {
        void* Ptr;
        bool Bool;
        char Char;
        int8_t Int8;
        uint8_t Uint8;
        int16_t Int16;
        uint16_t Uint16;
        int Int;
        unsigned int Uint;
        int32_t Int32;
        uint32_t Uint32;
        int64_t Int64;
        uint64_t Uint64;
        float Float;
        double Double;
    } arg1, arg2;
} SCleanupAction;

weixin_48148422's avatar
weixin_48148422 已提交
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70

/*
 * exception hander registration
 */
typedef struct SExceptionNode {
   struct SExceptionNode* prev;
   jmp_buf jb;
   int32_t code;
   int32_t maxCleanupAction;
   int32_t numCleanupAction;
   SCleanupAction* cleanupActions;
} SExceptionNode;

////////////////////////////////////////////////////////////////////////////////
// functions & macros for auto-cleanup

71 72 73 74 75
void cleanupPush_void_ptr_ptr   ( bool failOnly, void* func, void* arg1, void* arg2 );
void cleanupPush_void_ptr_bool  ( bool failOnly, void* func, void* arg1, bool arg2 );
void cleanupPush_void_ptr       ( bool failOnly, void* func, void* arg );
void cleanupPush_int_int        ( bool failOnly, void* func, int arg );
void cleanupPush_void           ( bool failOnly, void* func );
weixin_48148422's avatar
weixin_48148422 已提交
76
void cleanupPush_int_ptr        ( bool failOnly, void* func, void* arg );
77 78

int32_t cleanupGetActionCount();
weixin_48148422's avatar
weixin_48148422 已提交
79 80
void cleanupExecuteTo( int32_t anchor, bool failed );
void cleanupExecute( SExceptionNode* node, bool failed );
81 82 83 84 85 86

#define CLEANUP_PUSH_VOID_PTR_PTR( failOnly, func, arg1, arg2 )  cleanupPush_void_ptr_ptr( (failOnly), (void*)(func), (void*)(arg1), (void*)(arg2) )
#define CLEANUP_PUSH_VOID_PTR_BOOL( failOnly, func, arg1, arg2 ) cleanupPush_void_ptr_bool( (failOnly), (void*)(func), (void*)(arg1), (bool)(arg2) )
#define CLEANUP_PUSH_VOID_PTR( failOnly, func, arg )             cleanupPush_void_ptr( (failOnly), (void*)(func), (void*)(arg) )
#define CLEANUP_PUSH_INT_INT( failOnly, func, arg )              cleanupPush_void_ptr( (failOnly), (void*)(func), (int)(arg) )
#define CLEANUP_PUSH_VOID( failOnly, func )                      cleanupPush_void( (failOnly), (void*)(func) )
weixin_48148422's avatar
weixin_48148422 已提交
87
#define CLEANUP_PUSH_INT_PTR( failOnly, func, arg )              cleanupPush_int_ptr( (failOnly), (void*)(func), (void*)(arg) )
88 89
#define CLEANUP_PUSH_FREE( failOnly, arg )                       cleanupPush_void_ptr( (failOnly), free, (void*)(arg) )
#define CLEANUP_PUSH_CLOSE( failOnly, arg )                      cleanupPush_int_int( (failOnly), close, (int)(arg) )
weixin_48148422's avatar
weixin_48148422 已提交
90
#define CLEANUP_PUSH_FCLOSE( failOnly, arg )                     cleanupPush_int_ptr( (failOnly), fclose, (void*)(arg) )
91

weixin_48148422's avatar
weixin_48148422 已提交
92 93
#define CLEANUP_GET_ANCHOR()          cleanupGetActionCount()
#define CLEANUP_EXECUTE_TO( anchor, failed )  cleanupExecuteTo( (anchor), (failed) )
94

weixin_48148422's avatar
weixin_48148422 已提交
95 96 97

////////////////////////////////////////////////////////////////////////////////
// functions & macros for exception handling
weixin_48148422's avatar
weixin_48148422 已提交
98

weixin_48148422's avatar
weixin_48148422 已提交
99
void exceptionPushNode( SExceptionNode* node );
100
int32_t exceptionPopNode();
weixin_48148422's avatar
weixin_48148422 已提交
101
void exceptionThrow( int32_t code );
weixin_48148422's avatar
weixin_48148422 已提交
102

103 104
#define TRY(maxCleanupActions) do { \
    SExceptionNode exceptionNode = { 0 }; \
weixin_48148422's avatar
weixin_48148422 已提交
105 106
    SCleanupAction cleanupActions[(maxCleanupActions) > 0 ? (maxCleanupActions) : 1]; \
    exceptionNode.maxCleanupAction = (maxCleanupActions) > 0 ? (maxCleanupActions) : 1; \
107 108 109 110
    exceptionNode.cleanupActions = cleanupActions; \
    exceptionPushNode( &exceptionNode ); \
    int caughtException = setjmp( exceptionNode.jb ); \
    if( caughtException == 0 )
weixin_48148422's avatar
weixin_48148422 已提交
111

weixin_48148422's avatar
weixin_48148422 已提交
112
#define CATCH( code ) int32_t code = exceptionPopNode(); \
weixin_48148422's avatar
weixin_48148422 已提交
113
    if( caughtException == 1 )
weixin_48148422's avatar
weixin_48148422 已提交
114

weixin_48148422's avatar
weixin_48148422 已提交
115
#define FINALLY( code ) int32_t code = exceptionPopNode();
weixin_48148422's avatar
weixin_48148422 已提交
116 117 118

#define END_TRY } while( 0 );

119
#define THROW( x )          exceptionThrow( (x) )
weixin_48148422's avatar
weixin_48148422 已提交
120 121
#define CAUGHT_EXCEPTION()  ((bool)(caughtException == 1))
#define CLEANUP_EXECUTE()   cleanupExecute( &exceptionNode, CAUGHT_EXCEPTION() )
weixin_48148422's avatar
weixin_48148422 已提交
122 123 124 125 126 127

#ifdef __cplusplus
}
#endif

#endif