Ruby 4.1.0dev (2026-05-26 revision d6fa8e3e0f876114bf4ab9c8961a8e10e32ac9db)
ruby_atomic.h (d6fa8e3e0f876114bf4ab9c8961a8e10e32ac9db)
1#ifndef INTERNAL_ATOMIC_H
2#define INTERNAL_ATOMIC_H
3
4#include "ruby/atomic.h"
5#ifdef HAVE_STDATOMIC_H
6# include <stdatomic.h>
7#endif
8
9#define RUBY_ATOMIC_VALUE_LOAD(x) rbimpl_atomic_value_load(&(x), RBIMPL_ATOMIC_SEQ_CST)
10
11/* shim macros only */
12#define ATOMIC_ADD(var, val) RUBY_ATOMIC_ADD(var, val)
13#define ATOMIC_CAS(var, oldval, newval) RUBY_ATOMIC_CAS(var, oldval, newval)
14#define ATOMIC_DEC(var) RUBY_ATOMIC_DEC(var)
15#define ATOMIC_EXCHANGE(var, val) RUBY_ATOMIC_EXCHANGE(var, val)
16#define ATOMIC_FETCH_ADD(var, val) RUBY_ATOMIC_FETCH_ADD(var, val)
17#define ATOMIC_FETCH_SUB(var, val) RUBY_ATOMIC_FETCH_SUB(var, val)
18#define ATOMIC_INC(var) RUBY_ATOMIC_INC(var)
19#define ATOMIC_OR(var, val) RUBY_ATOMIC_OR(var, val)
20#define ATOMIC_PTR_CAS(var, oldval, newval) RUBY_ATOMIC_PTR_CAS(var, oldval, newval)
21#define ATOMIC_PTR_EXCHANGE(var, val) RUBY_ATOMIC_PTR_EXCHANGE(var, val)
22#define ATOMIC_SET(var, val) RUBY_ATOMIC_SET(var, val)
23#define ATOMIC_SIZE_ADD(var, val) RUBY_ATOMIC_SIZE_ADD(var, val)
24#define ATOMIC_SIZE_CAS(var, oldval, newval) RUBY_ATOMIC_SIZE_CAS(var, oldval, newval)
25#define ATOMIC_SIZE_DEC(var) RUBY_ATOMIC_SIZE_DEC(var)
26#define ATOMIC_SIZE_EXCHANGE(var, val) RUBY_ATOMIC_SIZE_EXCHANGE(var, val)
27#define ATOMIC_SIZE_INC(var) RUBY_ATOMIC_SIZE_INC(var)
28#define ATOMIC_SIZE_SUB(var, val) RUBY_ATOMIC_SIZE_SUB(var, val)
29#define ATOMIC_SUB(var, val) RUBY_ATOMIC_SUB(var, val)
30#define ATOMIC_VALUE_CAS(var, oldval, val) RUBY_ATOMIC_VALUE_CAS(var, oldval, val)
31#define ATOMIC_VALUE_EXCHANGE(var, val) RUBY_ATOMIC_VALUE_EXCHANGE(var, val)
32
33#define ATOMIC_LOAD_RELAXED(var) rbimpl_atomic_load(&(var), RBIMPL_ATOMIC_RELAXED)
34
35typedef RBIMPL_ALIGNAS(8) uint64_t rbimpl_atomic_uint64_t;
36
37static inline uint64_t
38rbimpl_atomic_u64_load_relaxed(const volatile rbimpl_atomic_uint64_t *value)
39{
40#if defined(HAVE_GCC_ATOMIC_BUILTINS_64)
41 return __atomic_load_n(value, __ATOMIC_RELAXED);
42#elif defined(_WIN32)
43 uint64_t val = *value;
44 return InterlockedCompareExchange64(RBIMPL_CAST((uint64_t *)value), val, val);
45#elif defined(__sun) && defined(HAVE_ATOMIC_H) && (defined(_LP64) || defined(_I32LPx))
46 uint64_t val = *value;
47 return atomic_cas_64(value, val, val);
48#else
49 // TODO: stdatomic
50
51 return *value;
52#endif
53}
54#define ATOMIC_U64_LOAD_RELAXED(var) rbimpl_atomic_u64_load_relaxed(&(var))
55
56static inline void
57rbimpl_atomic_u64_set_relaxed(volatile rbimpl_atomic_uint64_t *address, uint64_t value)
58{
59#if defined(HAVE_GCC_ATOMIC_BUILTINS_64)
60 __atomic_store_n(address, value, __ATOMIC_RELAXED);
61#elif defined(_WIN32)
62 InterlockedExchange64(address, value);
63#elif defined(__sun) && defined(HAVE_ATOMIC_H) && (defined(_LP64) || defined(_I32LPx))
64 atomic_swap_64(address, value);
65#else
66 // TODO: stdatomic
67
68 *address = value;
69#endif
70}
71#define ATOMIC_U64_SET_RELAXED(var, val) rbimpl_atomic_u64_set_relaxed(&(var), val)
72
73static inline uint64_t
74rbimpl_atomic_u64_fetch_add_relaxed(volatile rbimpl_atomic_uint64_t *value, uint64_t addend)
75{
76#if defined(HAVE_GCC_ATOMIC_BUILTINS_64)
77 return __atomic_fetch_add(value, addend, __ATOMIC_RELAXED);
78#elif defined(_WIN32)
79 return (uint64_t)InterlockedExchangeAdd64((LONG64 *)value, (LONG64)addend);
80#elif defined(__sun) && defined(HAVE_ATOMIC_H) && (defined(_LP64) || defined(_I32LPx))
81 return atomic_add_64_nv(value, addend) - addend;
82#else
83 // TODO: stdatomic
84 uint64_t prev = *value;
85 *value = prev + addend;
86 return prev;
87#endif
88}
89#define ATOMIC_U64_FETCH_ADD_RELAXED(var, val) rbimpl_atomic_u64_fetch_add_relaxed(&(var), val)
90
91static inline uint64_t
92rbimpl_atomic_u64_load_acquire(const volatile rbimpl_atomic_uint64_t *value)
93{
94#if defined(HAVE_GCC_ATOMIC_BUILTINS_64)
95 return __atomic_load_n(value, __ATOMIC_ACQUIRE);
96#else
97 return rbimpl_atomic_u64_load_relaxed(value);
98#endif
99}
100#define ATOMIC_U64_LOAD_ACQUIRE(var) rbimpl_atomic_u64_load_acquire(&(var))
101
102static inline void
103rbimpl_atomic_u64_set_release(volatile rbimpl_atomic_uint64_t *address, uint64_t value)
104{
105#if defined(HAVE_GCC_ATOMIC_BUILTINS_64)
106 __atomic_store_n(address, value, __ATOMIC_RELEASE);
107#else
108 rbimpl_atomic_u64_set_relaxed(address, value);
109#endif
110}
111#define ATOMIC_U64_SET_RELEASE(var, val) rbimpl_atomic_u64_set_release(&(var), val)
112
113#endif
Atomic operations.
#define RBIMPL_ALIGNAS
Wraps (or simulates) alignas.
Definition stdalign.h:57