1 #ifndef _URCU_TLS_COMPAT_H
2 #define _URCU_TLS_COMPAT_H
7 * Userspace RCU library - Thread-Local Storage Compatibility Header
9 * Copyright 2012 - Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
11 * This library is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU Lesser General Public
13 * License as published by the Free Software Foundation; either
14 * version 2.1 of the License, or (at your option) any later version.
16 * This library is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 * Lesser General Public License for more details.
21 * You should have received a copy of the GNU Lesser General Public
22 * License along with this library; if not, write to the Free Software
23 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
27 #include <urcu/config.h>
28 #include <urcu/compiler.h>
29 #include <urcu/arch.h>
38 * Default to '__thread' on all C and C++ compilers except MSVC. While C11 has
39 * '_Thread_local' and C++11 has 'thread_local', only '__thread' seems to have
40 * a compatible implementation when linking public extern symbols across
41 * language boundaries.
43 * For more details, see 'https://gcc.gnu.org/onlinedocs/gcc/Thread-Local.html'.
46 # define URCU_TLS_STORAGE_CLASS __declspec(thread)
48 # define URCU_TLS_STORAGE_CLASS __thread
52 * Hint: How to define/declare TLS variables of compound types
53 * such as array or function pointers?
55 * Answer: Use typedef to assign a type_name to the compound type.
56 * Example: Define a TLS variable which is an int array with len=4:
58 * typedef int my_int_array_type[4];
59 * DEFINE_URCU_TLS(my_int_array_type, var_name);
62 * typedef void (*call_rcu_flavor)(struct rcu_head *, XXXX);
63 * DECLARE_URCU_TLS(call_rcu_flavor, p_call_rcu);
65 * NOTE: URCU_TLS() is NOT async-signal-safe, you can't use it
66 * inside any function which can be called from signal handler.
68 * But if pthread_getspecific() is async-signal-safe in your
69 * platform, you can make URCU_TLS() async-signal-safe via:
70 * ensuring the first call to URCU_TLS() of a given TLS variable of
71 * all threads is called earliest from a non-signal handler function.
73 * Example: In any thread, the first call of URCU_TLS(rcu_reader)
74 * is called from rcu_register_thread(), so we can ensure all later
75 * URCU_TLS(rcu_reader) in any thread is async-signal-safe.
77 * Moreover, URCU_TLS variables should not be touched from signal
78 * handlers setup with with sigaltstack(2).
81 # define DECLARE_URCU_TLS(type, name) \
82 URCU_TLS_STORAGE_CLASS type name
84 # define DEFINE_URCU_TLS(type, name) \
85 URCU_TLS_STORAGE_CLASS type name
87 # define DEFINE_URCU_TLS_INIT(type, name, init) \
88 URCU_TLS_STORAGE_CLASS type name = (init)
90 # define URCU_TLS(name) (name)
92 #else /* #ifndef CONFIG_RCU_TLS */
95 * The *_1() macros ensure macro parameters are expanded.
102 pthread_mutex_t init_mutex
;
106 # define DECLARE_URCU_TLS_1(type, name) \
107 type *__tls_access_ ## name(void)
108 # define DECLARE_URCU_TLS(type, name) \
109 DECLARE_URCU_TLS_1(type, name)
112 * Note: we don't free memory at process exit, since it will be dealt
115 # define DEFINE_URCU_TLS_INIT_1(type, name, do_init) \
116 type *__tls_access_ ## name(void) \
118 static struct urcu_tls __tls_ ## name = { \
120 .init_mutex = PTHREAD_MUTEX_INITIALIZER,\
123 __typeof__(type) *__tls_p; \
124 if (!__tls_ ## name.init_done) { \
125 /* Mutex to protect concurrent init */ \
126 pthread_mutex_lock(&__tls_ ## name.init_mutex); \
127 if (!__tls_ ## name.init_done) { \
128 (void) pthread_key_create(&__tls_ ## name.key, \
130 cmm_smp_wmb(); /* create key before write init_done */ \
131 __tls_ ## name.init_done = 1; \
133 pthread_mutex_unlock(&__tls_ ## name.init_mutex); \
135 cmm_smp_rmb(); /* read init_done before getting key */ \
136 __tls_p = (__typeof__(type) *) pthread_getspecific(__tls_ ## name.key); \
137 if (caa_unlikely(__tls_p == NULL)) { \
138 __tls_p = (__typeof__(type) *) calloc(1, sizeof(type)); \
140 (void) pthread_setspecific(__tls_ ## name.key, \
146 # define _URCU_TLS_INIT(init) \
149 # define DEFINE_URCU_TLS_INIT(type, name, init) \
150 DEFINE_URCU_TLS_INIT_1(type, name, _URCU_TLS_INIT(init))
152 # define DEFINE_URCU_TLS(type, name) \
153 DEFINE_URCU_TLS_INIT_1(type, name, /* empty */)
155 # define URCU_TLS_1(name) (*__tls_access_ ## name())
157 # define URCU_TLS(name) URCU_TLS_1(name)
159 #endif /* #else #ifndef CONFIG_RCU_TLS */
165 #endif /* _URCU_TLS_COMPAT_H */