29 # if !defined (__cplusplus) && (__STDC_VERSION__ >= 201112L) \
30 && !defined (__STDC_NO_ATOMICS__)
33 # include <stdatomic.h>
37 # define ATOMIC_FLAG_INIT false
39 # define ATOMIC_VAR_INIT(value) (value)
41 # define atomic_init(obj, value) \
42 do { *(obj) = (value); } while(0)
44 # define kill_dependency(y) \
47 # define atomic_thread_fence(order) \
50 # define atomic_signal_fence(order) \
53 # define atomic_is_lock_free(obj) \
99 # if defined (__GCC_HAVE_SYNC_COMPARE_AND_SWAP_4) || (defined (__clang__) && (defined (__x86_64__) || defined (__i386__)))
103 # define atomic_store(object,desired) \
105 *(object) = (desired); \
106 __sync_synchronize(); \
109 # define atomic_store_explicit(object,desired,order) \
110 atomic_store(object,desired)
112 # define atomic_load(object) \
113 (__sync_synchronize(), *(object))
115 # define atomic_load_explicit(object,order) \
118 # define atomic_exchange(object,desired) \
120 typeof (object) _obj = (object); \
121 typeof (*object) _old; \
123 _old = atomic_load(_obj); \
124 while (!__sync_bool_compare_and_swap(_obj, _old, (desired))); \
128 # define atomic_exchange_explicit(object,desired,order) \
129 atomic_exchange(object,desired)
131 # define atomic_compare_exchange(object,expected,desired) \
133 typeof (object) _exp = (expected); \
134 typeof (*object) _old = *_exp; \
135 *_exp = __sync_val_compare_and_swap((object), _old, (desired)); \
139 # define atomic_compare_exchange_strong(object,expected,desired) \
140 atomic_compare_exchange(object, expected, desired)
142 # define atomic_compare_exchange_strong_explicit(object,expected,desired,order) \
143 atomic_compare_exchange_strong(object, expected, desired)
145 # define atomic_compare_exchange_weak(object,expected,desired) \
146 atomic_compare_exchange(object, expected, desired)
148 # define atomic_compare_exchange_weak_explicit(object,expected,desired,order) \
149 atomic_compare_exchange_weak(object, expected, desired)
151 # define atomic_fetch_add(object,operand) \
152 __sync_fetch_and_add(object, operand)
154 # define atomic_fetch_add_explicit(object,operand,order) \
155 atomic_fetch_add(object,operand)
157 # define atomic_fetch_sub(object,operand) \
158 __sync_fetch_and_sub(object, operand)
160 # define atomic_fetch_sub_explicit(object,operand,order) \
161 atomic_fetch_sub(object,operand)
163 # define atomic_fetch_or(object,operand) \
164 __sync_fetch_and_or(object, operand)
166 # define atomic_fetch_or_explicit(object,operand,order) \
167 atomic_fetch_or(object,operand)
169 # define atomic_fetch_xor(object,operand) \
170 __sync_fetch_and_sub(object, operand)
172 # define atomic_fetch_xor_explicit(object,operand,order) \
173 atomic_fetch_sub(object,operand)
175 # define atomic_fetch_and(object,operand) \
176 __sync_fetch_and_and(object, operand)
178 # define atomic_fetch_and_explicit(object,operand,order) \
179 atomic_fetch_and(object,operand)
181 # define atomic_flag_test_and_set(object) \
182 atomic_exchange(object, true)
184 # define atomic_flag_test_and_set_explicit(object,order) \
185 atomic_flag_test_and_set(object)
187 # define atomic_flag_clear(object) \
188 atomic_store(object, false)
190 # define atomic_flag_clear_explicit(object,order) \
191 atomic_flag_clear(object)
193 # elif defined (__GNUC__)
197 # define atomic_store(object,desired) \
199 typeof (object) _obj = (object); \
200 typeof (*object) _des = (desired); \
201 vlc_global_lock(VLC_ATOMIC_MUTEX); \
203 vlc_global_unlock(VLC_ATOMIC_MUTEX); \
205 # define atomic_store_explicit(object,desired,order) \
206 atomic_store(object,desired)
208 # define atomic_load(object) \
210 typeof (object) _obj = (object); \
211 typeof (*object) _old; \
212 vlc_global_lock(VLC_ATOMIC_MUTEX); \
214 vlc_global_unlock(VLC_ATOMIC_MUTEX); \
217 # define atomic_load_explicit(object,order) \
220 # define atomic_exchange(object,desired) \
222 typeof (object) _obj = (object); \
223 typeof (*object) _des = (desired); \
224 typeof (*object) _old; \
225 vlc_global_lock(VLC_ATOMIC_MUTEX); \
228 vlc_global_unlock(VLC_ATOMIC_MUTEX); \
231 # define atomic_exchange_explicit(object,desired,order) \
232 atomic_exchange(object,desired)
234 # define atomic_compare_exchange_strong(object,expected,desired) \
236 typeof (object) _obj = (object); \
237 typeof (object) _exp = (expected); \
238 typeof (*object) _des = (desired); \
240 vlc_global_lock(VLC_ATOMIC_MUTEX); \
241 ret = *_obj == *_exp; \
246 vlc_global_unlock(VLC_ATOMIC_MUTEX); \
249 # define atomic_compare_exchange_strong_explicit(object,expected,desired,order) \
250 atomic_compare_exchange_strong(object, expected, desired)
251 # define atomic_compare_exchange_weak(object,expected,desired) \
252 atomic_compare_exchange_strong(object, expected, desired)
253 # define atomic_compare_exchange_weak_explicit(object,expected,desired,order) \
254 atomic_compare_exchange_weak(object, expected, desired)
256 # define atomic_fetch_OP(object,desired,op) \
258 typeof (object) _obj = (object); \
259 typeof (*object) _des = (desired); \
260 typeof (*object) _old; \
261 vlc_global_lock(VLC_ATOMIC_MUTEX); \
263 *_obj = (*_obj) op (_des); \
264 vlc_global_unlock(VLC_ATOMIC_MUTEX); \
268 # define atomic_fetch_add(object,operand) \
269 atomic_fetch_OP(object,operand,+)
270 # define atomic_fetch_add_explicit(object,operand,order) \
271 atomic_fetch_add(object,operand)
273 # define atomic_fetch_sub(object,operand) \
274 atomic_fetch_OP(object,operand,-)
275 # define atomic_fetch_sub_explicit(object,operand,order) \
276 atomic_fetch_sub(object,operand)
278 # define atomic_fetch_or(object,operand) \
279 atomic_fetch_OP(object,operand,|)
280 # define atomic_fetch_or_explicit(object,operand,order) \
281 atomic_fetch_or(object,operand)
283 # define atomic_fetch_xor(object,operand) \
284 atomic_fetch_OP(object,operand,^)
285 # define atomic_fetch_xor_explicit(object,operand,order) \
286 atomic_fetch_sub(object,operand)
288 # define atomic_fetch_and(object,operand) \
289 atomic_fetch_OP(object,operand,&)
290 # define atomic_fetch_and_explicit(object,operand,order) \
291 atomic_fetch_and(object,operand)
293 # define atomic_flag_test_and_set(object) \
294 atomic_exchange(object, true)
296 # define atomic_flag_test_and_set_explicit(object,order) \
297 atomic_flag_test_and_set(object)
299 # define atomic_flag_clear(object) \
300 atomic_store(object, false)
302 # define atomic_flag_clear_explicit(object,order) \
303 atomic_flag_clear(object)
306 # error FIXME: implement atomic operations for this compiler.
319 # define VLC_ATOMIC_INIT(val) { (val) }
324 return atomic_load(&atom->
u);
329 atomic_store(&atom->
u, v);
335 return atomic_fetch_add(&atom->
u, v) + v;
340 return atomic_fetch_sub (&atom->
u, v) - v;
355 return atomic_exchange(&atom->
u, v);
359 uintptr_t u, uintptr_t v)
361 atomic_compare_exchange_strong(&atom->
u, &u, v);
370 union {
float f; uint32_t i; } u;
371 u.i = atomic_load(atom);
378 union {
float f; uint32_t i; } u;
380 atomic_store(atom, u.i);