Fix: trace events in C constructors/destructors
[lttng-ust.git] / include / lttng / ust-compiler.h
1 /*
2 * SPDX-License-Identifier: MIT
3 *
4 * Copyright (C) 2011-2012 Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
5 * Copyright (C) 2011-2012 Paul Woegerer <paul_woegerer@mentor.com>
6 */
7
8 #ifndef _LTTNG_UST_COMPILER_H
9 #define _LTTNG_UST_COMPILER_H
10
11 #include <assert.h>
12
13 /*
14 * By default, LTTng-UST uses the priority 150 for the tracepoint and probe
15 * provider constructors to trace tracepoints located within
16 * constructors/destructors with a higher priority value within the same
17 * module. This priority can be overridden by the application.
18 */
19 #ifndef LTTNG_UST_CONSTRUCTOR_PRIO
20 #define LTTNG_UST_CONSTRUCTOR_PRIO 150
21 #endif
22
23 #define lttng_ust_notrace __attribute__((no_instrument_function))
24
25 /*
26 * Clang supports the no_sanitize variable attribute on global variables.
27 * GCC only supports the no_sanitize_address function attribute, which is
28 * not what we need.
29 */
30 #if defined(__clang__)
31 # if __has_feature(address_sanitizer)
32 # define __lttng_ust_variable_attribute_no_sanitize_address \
33 __attribute__((no_sanitize("address")))
34 # else
35 # define __lttng_ust_variable_attribute_no_sanitize_address
36 # endif
37 #else
38 # define __lttng_ust_variable_attribute_no_sanitize_address
39 #endif
40
41 /*
42 * g++ 4.8 and prior do not support C99 compound literals. Therefore,
43 * force allocating those on the heap with these C++ compilers.
44 */
45 #if defined (__cplusplus) && !defined (__clang__) && defined (__GNUC__) && \
46 ((__GNUC__ < 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ <= 8)))
47 # ifndef LTTNG_UST_ALLOCATE_COMPOUND_LITERAL_ON_HEAP
48 # define LTTNG_UST_ALLOCATE_COMPOUND_LITERAL_ON_HEAP
49 # endif
50 #endif
51
52 /*
53 * Compound literals with static storage are needed by LTTng.
54 * Compound literals are part of the C99 and C11 standards, but not
55 * part of the C++ standards. However, those are supported by both g++ and
56 * clang. In order to be strictly C++11 compliant, defining
57 * LTTNG_UST_ALLOCATE_COMPOUND_LITERAL_ON_HEAP before including this header
58 * allocates those on the heap in C++.
59 *
60 * Example use:
61 * static struct mystruct *var = LTTNG_UST_COMPOUND_LITERAL(struct mystruct, { 1, 2, 3 });
62 */
63 #if defined (__cplusplus) && defined (LTTNG_UST_ALLOCATE_COMPOUND_LITERAL_ON_HEAP)
64 #define LTTNG_UST_COMPOUND_LITERAL(type, ...) new (type) __VA_ARGS__
65 #else
66 #define LTTNG_UST_COMPOUND_LITERAL(type, ...) (type[]) { __VA_ARGS__ }
67 #endif
68
69 /*
70 * Compile time assertion.
71 * - predicate: boolean expression to evaluate,
72 * - msg: string to print to the user on failure when `static_assert()` is
73 * supported,
74 * - c_identifier_msg: message to be included in the typedef to emulate a
75 * static assertion. This parameter must be a valid C identifier as it will
76 * be used as a typedef name.
77 */
78 #ifdef __cplusplus
79 #define lttng_ust_static_assert(predicate, msg, c_identifier_msg) \
80 static_assert(predicate, msg)
81 #elif defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L)
82 #define lttng_ust_static_assert(predicate, msg, c_identifier_msg) \
83 _Static_assert(predicate, msg)
84 #else
85 /*
86 * Evaluates the predicate and emit a compilation error on failure.
87 *
88 * If the predicate evaluates to true, this macro emits a typedef of an array
89 * of size 0.
90 *
91 * If the predicate evaluates to false, this macro emits a typedef of an array
92 * of negative size which is invalid in C and forces a compiler error. The msg
93 * parameter is used in the tentative typedef so it is printed to the user.
94 */
95 #define lttng_ust_static_assert(predicate, msg, c_identifier_msg) \
96 typedef char lttng_ust_static_assert_##c_identifier_msg[2*!!(predicate)-1]
97 #endif
98
99 /* Combine two tokens. */
100 #define LTTNG_UST_COMPILER__COMBINE_TOKENS(_tokena, _tokenb) \
101 _tokena##_tokenb
102 #define LTTNG_UST_COMPILER_COMBINE_TOKENS(_tokena, _tokenb) \
103 LTTNG_UST_COMPILER__COMBINE_TOKENS(_tokena, _tokenb)
104 /*
105 * Wrap constructor and destructor functions to invoke them as functions with
106 * the constructor/destructor GNU C attributes when building as C, or as the
107 * constructor/destructor of a variable defined within an anonymous namespace
108 * when building as C++.
109 */
110 #ifdef __cplusplus
111 #define LTTNG_UST_DECLARE_CONSTRUCTOR_DESTRUCTOR(name, constructor_func, \
112 destructor_func, ...) \
113 namespace lttng { \
114 namespace ust { \
115 namespace details { \
116 class LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, \
117 name) { \
118 public: \
119 LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() __VA_ARGS__; \
120 ~LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() __VA_ARGS__; \
121 }; \
122 LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)::LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() \
123 { \
124 constructor_func(); \
125 } \
126 LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)::~LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_destructor_, name)() \
127 { \
128 destructor_func(); \
129 } \
130 } \
131 } \
132 } \
133 \
134 namespace { \
135 const lttng::ust::details::LTTNG_UST_COMPILER_COMBINE_TOKENS( \
136 lttng_ust_constructor_destructor_, name) \
137 LTTNG_UST_COMPILER_COMBINE_TOKENS(name, registration_instance); \
138 }
139 #else /* __cplusplus */
140 #define LTTNG_UST_DECLARE_CONSTRUCTOR_DESTRUCTOR(name, constructor_func, \
141 destructor_func, ...) \
142 static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_, name)(void) \
143 __attribute__((constructor(LTTNG_UST_CONSTRUCTOR_PRIO))) __VA_ARGS__; \
144 static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_constructor_, name)(void) \
145 { \
146 constructor_func(); \
147 } \
148 static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_destructor_, name)(void) \
149 __attribute__((destructor(LTTNG_UST_CONSTRUCTOR_PRIO))) __VA_ARGS__; \
150 static void LTTNG_UST_COMPILER_COMBINE_TOKENS(lttng_ust_destructor_, name)(void) \
151 { \
152 destructor_func(); \
153 }
154 #endif
155
156 #endif /* _LTTNG_UST_COMPILER_H */
This page took 0.035975 seconds and 4 git commands to generate.