mdbx: avoid memory-model from variables for C11 atomics (workaround for lcc 1.26).

This commit is contained in:
Леонид Юрьев (Leonid Yuriev) 2022-06-12 20:04:35 +03:00
parent c082eb7d8a
commit 3817236b68
2 changed files with 54 additions and 35 deletions

View File

@ -877,6 +877,7 @@ size_t __hot mdbx_e2k_strnlen_bug_workaround(const char *s, size_t maxlen) {
/*------------------------------------------------------------------------------ /*------------------------------------------------------------------------------
* safe read/write volatile 64-bit fields on 32-bit architectures. */ * safe read/write volatile 64-bit fields on 32-bit architectures. */
#ifndef atomic_store64
MDBX_MAYBE_UNUSED static __always_inline uint64_t MDBX_MAYBE_UNUSED static __always_inline uint64_t
atomic_store64(MDBX_atomic_uint64_t *p, const uint64_t value, atomic_store64(MDBX_atomic_uint64_t *p, const uint64_t value,
enum MDBX_memory_order order) { enum MDBX_memory_order order) {
@ -900,7 +901,9 @@ atomic_store64(MDBX_atomic_uint64_t *p, const uint64_t value,
#endif /* !MDBX_64BIT_ATOMIC */ #endif /* !MDBX_64BIT_ATOMIC */
return value; return value;
} }
#endif /* atomic_store64 */
#ifndef atomic_load64
MDBX_MAYBE_UNUSED static MDBX_MAYBE_UNUSED static
#if MDBX_64BIT_ATOMIC #if MDBX_64BIT_ATOMIC
__always_inline __always_inline
@ -940,6 +943,7 @@ MDBX_MAYBE_UNUSED static
} }
#endif /* !MDBX_64BIT_ATOMIC */ #endif /* !MDBX_64BIT_ATOMIC */
} }
#endif /* atomic_load64 */
static __always_inline void atomic_yield(void) { static __always_inline void atomic_yield(void) {
#if defined(_WIN32) || defined(_WIN64) #if defined(_WIN32) || defined(_WIN64)
@ -1062,6 +1066,12 @@ static __always_inline uint64_t safe64_txnid_next(uint64_t txnid) {
return txnid; return txnid;
} }
#if defined(MDBX_HAVE_C11ATOMICS) && defined(__LCC__)
#define safe64_reset(p, single_writer) \
atomic_store64(p, UINT64_MAX, \
(single_writer) ? mo_AcquireRelease \
: mo_SequentialConsistency)
#else
static __always_inline void safe64_reset(MDBX_atomic_uint64_t *p, static __always_inline void safe64_reset(MDBX_atomic_uint64_t *p,
bool single_writer) { bool single_writer) {
#if !MDBX_64BIT_CAS #if !MDBX_64BIT_CAS
@ -1089,6 +1099,7 @@ static __always_inline void safe64_reset(MDBX_atomic_uint64_t *p,
assert(p->weak >= SAFE64_INVALID_THRESHOLD); assert(p->weak >= SAFE64_INVALID_THRESHOLD);
mdbx_jitter4testing(true); mdbx_jitter4testing(true);
} }
#endif /* LCC && MDBX_HAVE_C11ATOMICS */
static __always_inline bool safe64_reset_compare(MDBX_atomic_uint64_t *p, static __always_inline bool safe64_reset_compare(MDBX_atomic_uint64_t *p,
txnid_t compare) { txnid_t compare) {

View File

@ -261,49 +261,54 @@ typedef union {
#define MDBX_c11a_rw(type, ptr) (&(ptr)->c11a) #define MDBX_c11a_rw(type, ptr) (&(ptr)->c11a)
#endif /* Crutches for C11 atomic compiler's bugs */ #endif /* Crutches for C11 atomic compiler's bugs */
static __always_inline memory_order mo_c11_store(enum MDBX_memory_order fence) { #define mo_c11_store(fence) \
switch (fence) { (((fence) == mo_Relaxed) ? memory_order_relaxed \
default: : ((fence) == mo_AcquireRelease) ? memory_order_release \
assert(false); : memory_order_seq_cst)
__unreachable(); #define mo_c11_load(fence) \
case mo_Relaxed: (((fence) == mo_Relaxed) ? memory_order_relaxed \
return memory_order_relaxed; : ((fence) == mo_AcquireRelease) ? memory_order_acquire \
case mo_AcquireRelease: : memory_order_seq_cst)
return memory_order_release;
case mo_SequentialConsistency:
return memory_order_seq_cst;
}
}
static __always_inline memory_order mo_c11_load(enum MDBX_memory_order fence) {
switch (fence) {
default:
assert(false);
__unreachable();
case mo_Relaxed:
return memory_order_relaxed;
case mo_AcquireRelease:
return memory_order_acquire;
case mo_SequentialConsistency:
return memory_order_seq_cst;
}
}
#endif /* MDBX_HAVE_C11ATOMICS */ #endif /* MDBX_HAVE_C11ATOMICS */
#ifndef __cplusplus #ifndef __cplusplus
MDBX_MAYBE_UNUSED static __always_inline void
mdbx_memory_fence(enum MDBX_memory_order order, bool write) {
#ifdef MDBX_HAVE_C11ATOMICS #ifdef MDBX_HAVE_C11ATOMICS
atomic_thread_fence(write ? mo_c11_store(order) : mo_c11_load(order)); #define mdbx_memory_fence(order, write) \
#else /* MDBX_HAVE_C11ATOMICS */ atomic_thread_fence((write) ? mo_c11_store(order) : mo_c11_load(order))
mdbx_compiler_barrier(); #else /* MDBX_HAVE_C11ATOMICS */
if (write && #define mdbx_memory_fence(order, write) \
order > (MDBX_CPU_WRITEBACK_INCOHERENT ? mo_Relaxed : mo_AcquireRelease)) do { \
mdbx_memory_barrier(); mdbx_compiler_barrier(); \
if (write && order > (MDBX_CPU_WRITEBACK_INCOHERENT ? mo_Relaxed \
: mo_AcquireRelease)) \
mdbx_memory_barrier(); \
} while (0)
#endif /* MDBX_HAVE_C11ATOMICS */ #endif /* MDBX_HAVE_C11ATOMICS */
}
#if defined(MDBX_HAVE_C11ATOMICS) && defined(__LCC__)
#define atomic_store32(p, value, order) \
({ \
const uint32_t value_to_store = (value); \
atomic_store_explicit(MDBX_c11a_rw(uint32_t, p), value_to_store, \
mo_c11_store(order)); \
value_to_store; \
})
#define atomic_load32(p, order) \
atomic_load_explicit(MDBX_c11a_ro(uint32_t, p), mo_c11_load(order))
#define atomic_store64(p, value, order) \
({ \
const uint64_t value_to_store = (value); \
atomic_store_explicit(MDBX_c11a_rw(uint64_t, p), value_to_store, \
mo_c11_store(order)); \
value_to_store; \
})
#define atomic_load64(p, order) \
atomic_load_explicit(MDBX_c11a_ro(uint64_t, p), mo_c11_load(order))
#endif /* LCC && MDBX_HAVE_C11ATOMICS */
#ifndef atomic_store32
MDBX_MAYBE_UNUSED static __always_inline uint32_t MDBX_MAYBE_UNUSED static __always_inline uint32_t
atomic_store32(MDBX_atomic_uint32_t *p, const uint32_t value, atomic_store32(MDBX_atomic_uint32_t *p, const uint32_t value,
enum MDBX_memory_order order) { enum MDBX_memory_order order) {
@ -319,7 +324,9 @@ atomic_store32(MDBX_atomic_uint32_t *p, const uint32_t value,
#endif /* MDBX_HAVE_C11ATOMICS */ #endif /* MDBX_HAVE_C11ATOMICS */
return value; return value;
} }
#endif /* atomic_store32 */
#ifndef atomic_load32
MDBX_MAYBE_UNUSED static __always_inline uint32_t MDBX_MAYBE_UNUSED static __always_inline uint32_t
atomic_load32(const MDBX_atomic_uint32_t *p, enum MDBX_memory_order order) { atomic_load32(const MDBX_atomic_uint32_t *p, enum MDBX_memory_order order) {
STATIC_ASSERT(sizeof(MDBX_atomic_uint32_t) == 4); STATIC_ASSERT(sizeof(MDBX_atomic_uint32_t) == 4);
@ -334,6 +341,7 @@ atomic_load32(const MDBX_atomic_uint32_t *p, enum MDBX_memory_order order) {
return value; return value;
#endif /* MDBX_HAVE_C11ATOMICS */ #endif /* MDBX_HAVE_C11ATOMICS */
} }
#endif /* atomic_load32 */
#endif /* !__cplusplus */ #endif /* !__cplusplus */