X-Git-Url: https://git.lttng.org/?a=blobdiff_plain;f=include%2Flttng%2Fust-compiler.h;h=b7fd2c1c42bff2bac5d6bfc1c4f403a61fa82234;hb=3615ef97a1075d302b5faad557154785135b7eb7;hp=674b519cd11991ee4a04b42abdb081e86d2d7843;hpb=89350fda665ceeba49bb7dbabe3ce68e4aa5040d;p=lttng-ust.git diff --git a/include/lttng/ust-compiler.h b/include/lttng/ust-compiler.h index 674b519c..b7fd2c1c 100644 --- a/include/lttng/ust-compiler.h +++ b/include/lttng/ust-compiler.h @@ -32,8 +32,8 @@ * g++ 4.8 and prior do not support C99 compound literals. Therefore, * force allocating those on the heap with these C++ compilers. */ -#if defined (__cplusplus) && defined (__GNUC__) && \ - (__GNUC__ < 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ <= 8)) +#if defined (__cplusplus) && !defined (__clang__) && defined (__GNUC__) && \ + ((__GNUC__ < 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ <= 8))) # ifndef LTTNG_UST_ALLOCATE_COMPOUND_LITERAL_ON_HEAP # define LTTNG_UST_ALLOCATE_COMPOUND_LITERAL_ON_HEAP # endif @@ -68,7 +68,7 @@ #ifdef __cplusplus #define lttng_ust_static_assert(predicate, msg, c_identifier_msg) \ static_assert(predicate, msg) -#elif __STDC_VERSION__ >= 201112L +#elif defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L) #define lttng_ust_static_assert(predicate, msg, c_identifier_msg) \ _Static_assert(predicate, msg) #else @@ -86,4 +86,61 @@ typedef char lttng_ust_static_assert_##c_identifier_msg[2*!!(predicate)-1] #endif +/* Combine two tokens. */ +#define LTTNG_UST_COMPILER__COMBINE_TOKENS(_tokena, _tokenb) \ + _tokena##_tokenb +#define LTTNG_UST_COMPILER_COMBINE_TOKENS(_tokena, _tokenb) \ + LTTNG_UST_COMPILER__COMBINE_TOKENS(_tokena, _tokenb) +/* + * Wrap constructor and destructor functions to invoke them as functions with + * the constructor/destructor GNU C attributes when building as C, or as the + * constructor/destructor of a variable defined within an anonymous namespace + * when building as C++. + */ +#ifdef __cplusplus +#define LTTNG_UST_DECLARE_CONSTRUCTOR_DESTRUCTOR(name, constructor_func, \ + destructor_func, ...) \ +namespace lttng { \ +namespace ust { \ +namespace details { \ +class LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, \ + name) { \ +public: \ + LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() __VA_ARGS__; \ + ~LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() __VA_ARGS__; \ +}; \ +LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)::LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() \ +{ \ + constructor_func(); \ +} \ +LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)::~LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() \ +{ \ + destructor_func(); \ +} \ +} \ +} \ +} \ + \ +namespace { \ +const lttng::ust::details::LTTNG_UST_COMPILER_COMBINE_TOKENS( \ + lttng_ust_constructor_destructor_, name) \ + LTTNG_UST_COMPILER_COMBINE_TOKENS(name, registration_instance); \ +} +#else /* __cplusplus */ +#define LTTNG_UST_DECLARE_CONSTRUCTOR_DESTRUCTOR(name, constructor_func, \ + destructor_func, ...) \ + static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_, name)(void) \ + __attribute__((constructor)) __VA_ARGS__; \ + static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_, name)(void) \ + { \ + constructor_func(); \ + } \ + static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_destructor_, name)(void) \ + __attribute__((destructor)) __VA_ARGS__; \ + static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_destructor_, name)(void) \ + { \ + destructor_func(); \ + } +#endif + #endif /* _LTTNG_UST_COMPILER_H */