mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-20 05:58:21 +08:00
mdbx: rework unaligned access.
The three points: - disentangle C11-atomic fences/barriers and pure-functions (with `__attribute__((__pure__))`) to avoid compiler misoptimization; - fix hypotetic unaligned access to 64-bit dwords on ARM with `__ARM_FEATURE_UNALIGNED` defined; - reasonable paranoia that makes clarity for code readers.
This commit is contained in:
parent
cbbfaf3202
commit
ef7b4289c0
113
src/core.c
113
src/core.c
@ -124,12 +124,17 @@ static __always_inline void poke_u8(uint8_t *const __restrict ptr,
|
|||||||
MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint16_t
|
MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint16_t
|
||||||
unaligned_peek_u16(const unsigned expected_alignment, const void *const ptr) {
|
unaligned_peek_u16(const unsigned expected_alignment, const void *const ptr) {
|
||||||
assert((uintptr_t)ptr % expected_alignment == 0);
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
if (MDBX_UNALIGNED_OK || (expected_alignment % sizeof(uint16_t)) == 0)
|
if (MDBX_UNALIGNED_OK >= 2 || (expected_alignment % sizeof(uint16_t)) == 0)
|
||||||
return *(const uint16_t *)ptr;
|
return *(const uint16_t *)ptr;
|
||||||
else {
|
else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
return *(const __unaligned uint16_t *)ptr;
|
||||||
|
#else
|
||||||
uint16_t v;
|
uint16_t v;
|
||||||
memcpy(&v, ptr, sizeof(v));
|
memcpy(&v, ptr, sizeof(v));
|
||||||
return v;
|
return v;
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,16 +142,22 @@ static __always_inline void
|
|||||||
unaligned_poke_u16(const unsigned expected_alignment,
|
unaligned_poke_u16(const unsigned expected_alignment,
|
||||||
void *const __restrict ptr, const uint16_t v) {
|
void *const __restrict ptr, const uint16_t v) {
|
||||||
assert((uintptr_t)ptr % expected_alignment == 0);
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
if (MDBX_UNALIGNED_OK || (expected_alignment % sizeof(v)) == 0)
|
if (MDBX_UNALIGNED_OK >= 2 || (expected_alignment % sizeof(v)) == 0)
|
||||||
*(uint16_t *)ptr = v;
|
*(uint16_t *)ptr = v;
|
||||||
else
|
else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
*((uint16_t __unaligned *)ptr) = v;
|
||||||
|
#else
|
||||||
memcpy(ptr, &v, sizeof(v));
|
memcpy(ptr, &v, sizeof(v));
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint32_t unaligned_peek_u32(
|
MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint32_t unaligned_peek_u32(
|
||||||
const unsigned expected_alignment, const void *const __restrict ptr) {
|
const unsigned expected_alignment, const void *const __restrict ptr) {
|
||||||
assert((uintptr_t)ptr % expected_alignment == 0);
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
if (MDBX_UNALIGNED_OK || (expected_alignment % sizeof(uint32_t)) == 0)
|
if (MDBX_UNALIGNED_OK >= 4 || (expected_alignment % sizeof(uint32_t)) == 0)
|
||||||
return *(const uint32_t *)ptr;
|
return *(const uint32_t *)ptr;
|
||||||
else if ((expected_alignment % sizeof(uint16_t)) == 0) {
|
else if ((expected_alignment % sizeof(uint16_t)) == 0) {
|
||||||
const uint16_t lo =
|
const uint16_t lo =
|
||||||
@ -155,9 +166,14 @@ MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint32_t unaligned_peek_u32(
|
|||||||
((const uint16_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__];
|
((const uint16_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__];
|
||||||
return lo | (uint32_t)hi << 16;
|
return lo | (uint32_t)hi << 16;
|
||||||
} else {
|
} else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
return *(const __unaligned uint32_t *)ptr;
|
||||||
|
#else
|
||||||
uint32_t v;
|
uint32_t v;
|
||||||
memcpy(&v, ptr, sizeof(v));
|
memcpy(&v, ptr, sizeof(v));
|
||||||
return v;
|
return v;
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -165,20 +181,26 @@ static __always_inline void
|
|||||||
unaligned_poke_u32(const unsigned expected_alignment,
|
unaligned_poke_u32(const unsigned expected_alignment,
|
||||||
void *const __restrict ptr, const uint32_t v) {
|
void *const __restrict ptr, const uint32_t v) {
|
||||||
assert((uintptr_t)ptr % expected_alignment == 0);
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
if (MDBX_UNALIGNED_OK || (expected_alignment % sizeof(v)) == 0)
|
if (MDBX_UNALIGNED_OK >= 4 || (expected_alignment % sizeof(v)) == 0)
|
||||||
*(uint32_t *)ptr = v;
|
*(uint32_t *)ptr = v;
|
||||||
else if ((expected_alignment % sizeof(uint16_t)) == 0) {
|
else if ((expected_alignment % sizeof(uint16_t)) == 0) {
|
||||||
((uint16_t *)ptr)[__BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__] = (uint16_t)v;
|
((uint16_t *)ptr)[__BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__] = (uint16_t)v;
|
||||||
((uint16_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__] =
|
((uint16_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__] =
|
||||||
(uint16_t)(v >> 16);
|
(uint16_t)(v >> 16);
|
||||||
} else
|
} else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
*((uint32_t __unaligned *)ptr) = v;
|
||||||
|
#else
|
||||||
memcpy(ptr, &v, sizeof(v));
|
memcpy(ptr, &v, sizeof(v));
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint64_t unaligned_peek_u64(
|
MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint64_t unaligned_peek_u64(
|
||||||
const unsigned expected_alignment, const void *const __restrict ptr) {
|
const unsigned expected_alignment, const void *const __restrict ptr) {
|
||||||
assert((uintptr_t)ptr % expected_alignment == 0);
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
if (MDBX_UNALIGNED_OK || (expected_alignment % sizeof(uint64_t)) == 0)
|
if (MDBX_UNALIGNED_OK >= 8 || (expected_alignment % sizeof(uint64_t)) == 0)
|
||||||
return *(const uint64_t *)ptr;
|
return *(const uint64_t *)ptr;
|
||||||
else if ((expected_alignment % sizeof(uint32_t)) == 0) {
|
else if ((expected_alignment % sizeof(uint32_t)) == 0) {
|
||||||
const uint32_t lo =
|
const uint32_t lo =
|
||||||
@ -187,9 +209,35 @@ MDBX_NOTHROW_PURE_FUNCTION static __always_inline uint64_t unaligned_peek_u64(
|
|||||||
((const uint32_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__];
|
((const uint32_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__];
|
||||||
return lo | (uint64_t)hi << 32;
|
return lo | (uint64_t)hi << 32;
|
||||||
} else {
|
} else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
return *(const __unaligned uint64_t *)ptr;
|
||||||
|
#else
|
||||||
uint64_t v;
|
uint64_t v;
|
||||||
memcpy(&v, ptr, sizeof(v));
|
memcpy(&v, ptr, sizeof(v));
|
||||||
return v;
|
return v;
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static __always_inline uint64_t
|
||||||
|
unaligned_peek_u64_volatile(const unsigned expected_alignment,
|
||||||
|
const volatile void *const __restrict ptr) {
|
||||||
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
|
assert(expected_alignment % sizeof(uint32_t) == 0);
|
||||||
|
if (MDBX_UNALIGNED_OK >= 8 || (expected_alignment % sizeof(uint64_t)) == 0)
|
||||||
|
return *(volatile const uint64_t *)ptr;
|
||||||
|
else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
return *(const volatile __unaligned uint64_t *)ptr;
|
||||||
|
#else
|
||||||
|
const uint32_t lo = ((const volatile uint32_t *)
|
||||||
|
ptr)[__BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__];
|
||||||
|
const uint32_t hi = ((const volatile uint32_t *)
|
||||||
|
ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__];
|
||||||
|
return lo | (uint64_t)hi << 32;
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -197,14 +245,20 @@ static __always_inline void
|
|||||||
unaligned_poke_u64(const unsigned expected_alignment,
|
unaligned_poke_u64(const unsigned expected_alignment,
|
||||||
void *const __restrict ptr, const uint64_t v) {
|
void *const __restrict ptr, const uint64_t v) {
|
||||||
assert((uintptr_t)ptr % expected_alignment == 0);
|
assert((uintptr_t)ptr % expected_alignment == 0);
|
||||||
if (MDBX_UNALIGNED_OK || (expected_alignment % sizeof(v)) == 0)
|
if (MDBX_UNALIGNED_OK >= 8 || (expected_alignment % sizeof(v)) == 0)
|
||||||
*(uint64_t *)ptr = v;
|
*(uint64_t *)ptr = v;
|
||||||
else if ((expected_alignment % sizeof(uint32_t)) == 0) {
|
else if ((expected_alignment % sizeof(uint32_t)) == 0) {
|
||||||
((uint32_t *)ptr)[__BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__] = (uint32_t)v;
|
((uint32_t *)ptr)[__BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__] = (uint32_t)v;
|
||||||
((uint32_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__] =
|
((uint32_t *)ptr)[__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__] =
|
||||||
(uint32_t)(v >> 32);
|
(uint32_t)(v >> 32);
|
||||||
} else
|
} else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
*((uint64_t __unaligned *)ptr) = v;
|
||||||
|
#else
|
||||||
memcpy(ptr, &v, sizeof(v));
|
memcpy(ptr, &v, sizeof(v));
|
||||||
|
#endif /* _MSC_VER || __unaligned */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define UNALIGNED_PEEK_8(ptr, struct, field) \
|
#define UNALIGNED_PEEK_8(ptr, struct, field) \
|
||||||
@ -5500,26 +5554,23 @@ static bool meta_weak_acceptable(const MDBX_env *env, const MDBX_meta *meta,
|
|||||||
#define METAPAGE(env, n) page_meta(pgno2page(env, n))
|
#define METAPAGE(env, n) page_meta(pgno2page(env, n))
|
||||||
#define METAPAGE_END(env) METAPAGE(env, NUM_METAS)
|
#define METAPAGE_END(env) METAPAGE(env, NUM_METAS)
|
||||||
|
|
||||||
static __inline txnid_t meta_txnid(const MDBX_env *env, const MDBX_meta *meta,
|
static __inline txnid_t mdbx_meta_txnid_stable(const MDBX_env *env,
|
||||||
const bool allow_volatile) {
|
const MDBX_meta *meta) {
|
||||||
mdbx_memory_fence(mo_AcquireRelease, false);
|
mdbx_memory_fence(mo_AcquireRelease, false);
|
||||||
txnid_t a = unaligned_peek_u64(4, &meta->mm_txnid_a);
|
txnid_t a = unaligned_peek_u64(4, &meta->mm_txnid_a);
|
||||||
txnid_t b = unaligned_peek_u64(4, &meta->mm_txnid_b);
|
txnid_t b = unaligned_peek_u64(4, &meta->mm_txnid_b);
|
||||||
if (allow_volatile)
|
|
||||||
return (a == b) ? a : 0;
|
|
||||||
mdbx_assert(env, a == b);
|
mdbx_assert(env, a == b);
|
||||||
(void)env;
|
(void)env;
|
||||||
return a;
|
return a;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline txnid_t mdbx_meta_txnid_stable(const MDBX_env *env,
|
|
||||||
const MDBX_meta *meta) {
|
|
||||||
return meta_txnid(env, meta, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
static __inline txnid_t mdbx_meta_txnid_fluid(const MDBX_env *env,
|
static __inline txnid_t mdbx_meta_txnid_fluid(const MDBX_env *env,
|
||||||
const MDBX_meta *meta) {
|
volatile const MDBX_meta *meta) {
|
||||||
return meta_txnid(env, meta, true);
|
(void)env;
|
||||||
|
mdbx_memory_fence(mo_AcquireRelease, false);
|
||||||
|
txnid_t a = unaligned_peek_u64_volatile(4, &meta->mm_txnid_a);
|
||||||
|
txnid_t b = unaligned_peek_u64_volatile(4, &meta->mm_txnid_b);
|
||||||
|
return (a == b) ? a : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static __inline void mdbx_meta_update_begin(const MDBX_env *env,
|
static __inline void mdbx_meta_update_begin(const MDBX_env *env,
|
||||||
@ -5662,7 +5713,7 @@ static MDBX_meta *mdbx_meta_head(const MDBX_env *env) {
|
|||||||
|
|
||||||
static txnid_t mdbx_recent_committed_txnid(const MDBX_env *env) {
|
static txnid_t mdbx_recent_committed_txnid(const MDBX_env *env) {
|
||||||
while (true) {
|
while (true) {
|
||||||
const MDBX_meta *head = mdbx_meta_head(env);
|
const volatile MDBX_meta *head = mdbx_meta_head(env);
|
||||||
const txnid_t recent = mdbx_meta_txnid_fluid(env, head);
|
const txnid_t recent = mdbx_meta_txnid_fluid(env, head);
|
||||||
mdbx_compiler_barrier();
|
mdbx_compiler_barrier();
|
||||||
if (likely(head == mdbx_meta_head(env) &&
|
if (likely(head == mdbx_meta_head(env) &&
|
||||||
@ -5673,7 +5724,7 @@ static txnid_t mdbx_recent_committed_txnid(const MDBX_env *env) {
|
|||||||
|
|
||||||
static txnid_t mdbx_recent_steady_txnid(const MDBX_env *env) {
|
static txnid_t mdbx_recent_steady_txnid(const MDBX_env *env) {
|
||||||
while (true) {
|
while (true) {
|
||||||
const MDBX_meta *head = mdbx_meta_steady(env);
|
const volatile MDBX_meta *head = mdbx_meta_steady(env);
|
||||||
const txnid_t recent = mdbx_meta_txnid_fluid(env, head);
|
const txnid_t recent = mdbx_meta_txnid_fluid(env, head);
|
||||||
mdbx_compiler_barrier();
|
mdbx_compiler_barrier();
|
||||||
if (likely(head == mdbx_meta_steady(env) &&
|
if (likely(head == mdbx_meta_steady(env) &&
|
||||||
@ -7087,7 +7138,7 @@ retry:;
|
|||||||
goto retry;
|
goto retry;
|
||||||
}
|
}
|
||||||
env->me_txn0->mt_txnid = head_txnid;
|
env->me_txn0->mt_txnid = head_txnid;
|
||||||
mdbx_assert(env, head_txnid == meta_txnid(env, head, false));
|
mdbx_assert(env, head_txnid == mdbx_meta_txnid_fluid(env, head));
|
||||||
mdbx_assert(env, head_txnid == mdbx_recent_committed_txnid(env));
|
mdbx_assert(env, head_txnid == mdbx_recent_committed_txnid(env));
|
||||||
mdbx_find_oldest(env->me_txn0);
|
mdbx_find_oldest(env->me_txn0);
|
||||||
flags |= MDBX_SHRINK_ALLOWED;
|
flags |= MDBX_SHRINK_ALLOWED;
|
||||||
@ -7563,7 +7614,7 @@ static int mdbx_txn_renew0(MDBX_txn *txn, const unsigned flags) {
|
|||||||
/* Seek & fetch the last meta */
|
/* Seek & fetch the last meta */
|
||||||
if (likely(/* not recovery mode */ env->me_stuck_meta < 0)) {
|
if (likely(/* not recovery mode */ env->me_stuck_meta < 0)) {
|
||||||
while (1) {
|
while (1) {
|
||||||
MDBX_meta *const meta = mdbx_meta_head(env);
|
const volatile MDBX_meta *const meta = mdbx_meta_head(env);
|
||||||
mdbx_jitter4testing(false);
|
mdbx_jitter4testing(false);
|
||||||
const txnid_t snap = mdbx_meta_txnid_fluid(env, meta);
|
const txnid_t snap = mdbx_meta_txnid_fluid(env, meta);
|
||||||
mdbx_jitter4testing(false);
|
mdbx_jitter4testing(false);
|
||||||
@ -7572,7 +7623,7 @@ static int mdbx_txn_renew0(MDBX_txn *txn, const unsigned flags) {
|
|||||||
atomic_store32(&r->mr_snapshot_pages_used, meta->mm_geo.next,
|
atomic_store32(&r->mr_snapshot_pages_used, meta->mm_geo.next,
|
||||||
mo_Relaxed);
|
mo_Relaxed);
|
||||||
atomic_store64(&r->mr_snapshot_pages_retired,
|
atomic_store64(&r->mr_snapshot_pages_retired,
|
||||||
unaligned_peek_u64(4, meta->mm_pages_retired),
|
unaligned_peek_u64_volatile(4, meta->mm_pages_retired),
|
||||||
mo_Relaxed);
|
mo_Relaxed);
|
||||||
safe64_write(&r->mr_txnid, snap);
|
safe64_write(&r->mr_txnid, snap);
|
||||||
mdbx_jitter4testing(false);
|
mdbx_jitter4testing(false);
|
||||||
@ -7589,10 +7640,16 @@ static int mdbx_txn_renew0(MDBX_txn *txn, const unsigned flags) {
|
|||||||
/* Snap the state from current meta-head */
|
/* Snap the state from current meta-head */
|
||||||
txn->mt_txnid = snap;
|
txn->mt_txnid = snap;
|
||||||
txn->mt_geo = meta->mm_geo;
|
txn->mt_geo = meta->mm_geo;
|
||||||
memcpy(txn->mt_dbs, meta->mm_dbs, CORE_DBS * sizeof(MDBX_db));
|
STATIC_ASSERT(CORE_DBS == 2);
|
||||||
|
txn->mt_dbs[0] = meta->mm_dbs[0];
|
||||||
|
txn->mt_dbs[1] = meta->mm_dbs[1];
|
||||||
txn->mt_canary = meta->mm_canary;
|
txn->mt_canary = meta->mm_canary;
|
||||||
|
|
||||||
/* LY: Retry on a race, ITS#7970. */
|
/* LY: Retry on a race, ITS#7970.
|
||||||
|
* The barrier is not needed here since C11-atomics are used,
|
||||||
|
* but it is reasonable paranoia to avoid compiler misoptimization
|
||||||
|
* and makes clarity for code readers. */
|
||||||
|
mdbx_compiler_barrier();
|
||||||
if (likely(meta == mdbx_meta_head(env) &&
|
if (likely(meta == mdbx_meta_head(env) &&
|
||||||
snap == mdbx_meta_txnid_fluid(env, meta) &&
|
snap == mdbx_meta_txnid_fluid(env, meta) &&
|
||||||
snap >= atomic_load64(&env->me_lck->mti_oldest_reader,
|
snap >= atomic_load64(&env->me_lck->mti_oldest_reader,
|
||||||
@ -11253,7 +11310,7 @@ mdbx_env_set_geometry(MDBX_env *env, intptr_t size_lower, intptr_t size_now,
|
|||||||
}
|
}
|
||||||
MDBX_meta *head = mdbx_meta_head(env);
|
MDBX_meta *head = mdbx_meta_head(env);
|
||||||
if (!inside_txn) {
|
if (!inside_txn) {
|
||||||
env->me_txn0->mt_txnid = meta_txnid(env, head, false);
|
env->me_txn0->mt_txnid = mdbx_meta_txnid_stable(env, head);
|
||||||
mdbx_find_oldest(env->me_txn0);
|
mdbx_find_oldest(env->me_txn0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -361,16 +361,28 @@
|
|||||||
#endif /* MDBX_64BIT_CAS */
|
#endif /* MDBX_64BIT_CAS */
|
||||||
|
|
||||||
#ifndef MDBX_UNALIGNED_OK
|
#ifndef MDBX_UNALIGNED_OK
|
||||||
#ifdef _MSC_VER
|
#if defined(__ALIGNED__) || defined(__SANITIZE_UNDEFINED__)
|
||||||
#define MDBX_UNALIGNED_OK 1 /* avoid MSVC misoptimization */
|
#define MDBX_UNALIGNED_OK 0 /* no unaligned access allowed */
|
||||||
|
#elif defined(__ARM_FEATURE_UNALIGNED)
|
||||||
|
#define MDBX_UNALIGNED_OK 4 /* ok unaligned for 32-bit words */
|
||||||
#elif __CLANG_PREREQ(5, 0) || __GNUC_PREREQ(5, 0)
|
#elif __CLANG_PREREQ(5, 0) || __GNUC_PREREQ(5, 0)
|
||||||
#define MDBX_UNALIGNED_OK 0 /* expecting optimization is well done */
|
/* expecting an optimization will well done, also this
|
||||||
#elif (defined(__ia32__) || defined(__ARM_FEATURE_UNALIGNED)) && \
|
* hushes false-positives from UBSAN (undefined behaviour sanitizer) */
|
||||||
!defined(__ALIGNED__)
|
|
||||||
#define MDBX_UNALIGNED_OK 1
|
|
||||||
#else
|
|
||||||
#define MDBX_UNALIGNED_OK 0
|
#define MDBX_UNALIGNED_OK 0
|
||||||
|
#elif defined(__e2k__) || defined(__elbrus__)
|
||||||
|
#if __iset__ > 4
|
||||||
|
#define MDBX_UNALIGNED_OK 8 /* ok unaligned for 64-bit words */
|
||||||
|
#else
|
||||||
|
#define MDBX_UNALIGNED_OK 4 /* ok unaligned for 32-bit words */
|
||||||
#endif
|
#endif
|
||||||
|
#elif defined(__ia32__)
|
||||||
|
#define MDBX_UNALIGNED_OK 8 /* ok unaligned for 64-bit words */
|
||||||
|
#else
|
||||||
|
#define MDBX_UNALIGNED_OK 0 /* no unaligned access allowed */
|
||||||
|
#endif
|
||||||
|
#elif MDBX_UNALIGNED_OK == 1
|
||||||
|
#undef MDBX_UNALIGNED_OK
|
||||||
|
#define MDBX_UNALIGNED_OK 32 /* any unaligned access allowed */
|
||||||
#endif /* MDBX_UNALIGNED_OK */
|
#endif /* MDBX_UNALIGNED_OK */
|
||||||
|
|
||||||
#ifndef MDBX_CACHELINE_SIZE
|
#ifndef MDBX_CACHELINE_SIZE
|
||||||
|
32
test/utils.h
32
test/utils.h
@ -148,35 +148,31 @@ static __inline uint16_t bswap16(uint16_t v) { return v << 8 | v >> 8; }
|
|||||||
namespace unaligned {
|
namespace unaligned {
|
||||||
|
|
||||||
template <typename T> static __inline T load(const void *ptr) {
|
template <typename T> static __inline T load(const void *ptr) {
|
||||||
#if defined(_MSC_VER) && \
|
if (MDBX_UNALIGNED_OK >= sizeof(T))
|
||||||
(defined(_M_ARM64) || defined(_M_X64) || defined(_M_IA64))
|
|
||||||
return *(const T __unaligned *)ptr;
|
|
||||||
#elif MDBX_UNALIGNED_OK
|
|
||||||
return *(const T *)ptr;
|
return *(const T *)ptr;
|
||||||
|
else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
|
return *(const T __unaligned *)ptr;
|
||||||
#else
|
#else
|
||||||
T local;
|
T local;
|
||||||
#if defined(__GNUC__) || defined(__clang__)
|
|
||||||
__builtin_memcpy(&local, (const T *)ptr, sizeof(T));
|
|
||||||
#else
|
|
||||||
memcpy(&local, (const T *)ptr, sizeof(T));
|
memcpy(&local, (const T *)ptr, sizeof(T));
|
||||||
#endif /* __GNUC__ || __clang__ */
|
|
||||||
return local;
|
return local;
|
||||||
#endif /* MDBX_UNALIGNED_OK */
|
#endif /* _MSC_VER || __unaligned */
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T> static __inline void store(void *ptr, const T &value) {
|
template <typename T> static __inline void store(void *ptr, const T &value) {
|
||||||
#if defined(_MSC_VER) && \
|
if (MDBX_UNALIGNED_OK >= sizeof(T))
|
||||||
(defined(_M_ARM64) || defined(_M_X64) || defined(_M_IA64))
|
*(T *)ptr = value;
|
||||||
|
else {
|
||||||
|
#if defined(__unaligned) || defined(_M_ARM) || defined(_M_ARM64) || \
|
||||||
|
defined(_M_X64) || defined(_M_IA64)
|
||||||
*((T __unaligned *)ptr) = value;
|
*((T __unaligned *)ptr) = value;
|
||||||
#elif MDBX_UNALIGNED_OK
|
|
||||||
*(volatile T *)ptr = value;
|
|
||||||
#else
|
|
||||||
#if defined(__GNUC__) || defined(__clang__)
|
|
||||||
__builtin_memcpy(ptr, &value, sizeof(T));
|
|
||||||
#else
|
#else
|
||||||
memcpy(ptr, &value, sizeof(T));
|
memcpy(ptr, &value, sizeof(T));
|
||||||
#endif /* __GNUC__ || __clang__ */
|
#endif /* _MSC_VER || __unaligned */
|
||||||
#endif /* MDBX_UNALIGNED_OK */
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} /* namespace unaligned */
|
} /* namespace unaligned */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user