mirror of
https://github.com/isar/libmdbx.git
synced 2025-01-10 12:04:13 +08:00
263837094e
Change-Id: Ie3db2f6439d5cc7ef3cd9ebb5b8e5d5d34f743dd
953 lines
39 KiB
C++
953 lines
39 KiB
C++
/* https://en.wikipedia.org/wiki/Operating_system_abstraction_layer */
|
||
|
||
/*
|
||
* Copyright 2015-2019 Leonid Yuriev <leo@yuriev.ru>
|
||
* and other libmdbx authors: please see AUTHORS file.
|
||
* All rights reserved.
|
||
*
|
||
* Redistribution and use in source and binary forms, with or without
|
||
* modification, are permitted only as authorized by the OpenLDAP
|
||
* Public License.
|
||
*
|
||
* A copy of this license is available in the file LICENSE in the
|
||
* top-level directory of the distribution or, alternatively, at
|
||
* <http://www.OpenLDAP.org/license.html>.
|
||
*/
|
||
|
||
#pragma once
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Microsoft compiler generates a lot of warning for self includes... */
|
||
|
||
#ifdef _MSC_VER
|
||
#pragma warning(push, 1)
|
||
#pragma warning(disable : 4548) /* expression before comma has no effect; \
|
||
expected expression with side - effect */
|
||
#pragma warning(disable : 4530) /* C++ exception handler used, but unwind \
|
||
* semantics are not enabled. Specify /EHsc */
|
||
#pragma warning(disable : 4577) /* 'noexcept' used with no exception handling \
|
||
* mode specified; termination on exception is \
|
||
* not guaranteed. Specify /EHsc */
|
||
#endif /* _MSC_VER (warnings) */
|
||
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
#if !defined(_CRT_SECURE_NO_WARNINGS)
|
||
#define _CRT_SECURE_NO_WARNINGS
|
||
#endif
|
||
#if !defined(_NO_CRT_STDIO_INLINE) && defined(MDBX_BUILD_DLL)
|
||
#define _NO_CRT_STDIO_INLINE
|
||
#endif
|
||
#endif /* Windows */
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* C99 includes */
|
||
#include <inttypes.h>
|
||
#include <stddef.h>
|
||
#include <stdint.h>
|
||
#include <stdlib.h>
|
||
|
||
#include <assert.h>
|
||
#include <fcntl.h>
|
||
#include <limits.h>
|
||
#include <stdio.h>
|
||
#include <string.h>
|
||
#include <time.h>
|
||
|
||
/* C11 stdalign.h */
|
||
#if __has_include(<stdalign.h>)
|
||
#include <stdalign.h>
|
||
#elif defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L
|
||
#define alignas(N) _Alignas(N)
|
||
#elif defined(_MSC_VER)
|
||
#define alignas(N) __declspec(align(N))
|
||
#elif __has_attribute(__aligned__) || defined(__GNUC__)
|
||
#define alignas(N) __attribute__((__aligned__(N)))
|
||
#else
|
||
#error "FIXME: Required _alignas() or equivalent."
|
||
#endif
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Systems includes */
|
||
|
||
#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || \
|
||
defined(__BSD__) || defined(__NETBSD__) || defined(__bsdi__) || \
|
||
defined(__DragonFly__) || defined(__APPLE__) || defined(__MACH__)
|
||
#include <sys/cdefs.h>
|
||
#else
|
||
#include <malloc.h>
|
||
#ifndef _POSIX_C_SOURCE
|
||
#ifdef _POSIX_SOURCE
|
||
#define _POSIX_C_SOURCE 1
|
||
#else
|
||
#define _POSIX_C_SOURCE 0
|
||
#endif
|
||
#endif
|
||
#endif /* !xBSD */
|
||
|
||
#ifndef _XOPEN_SOURCE
|
||
#define _XOPEN_SOURCE 0
|
||
#endif
|
||
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
#define WIN32_LEAN_AND_MEAN
|
||
#include <tlhelp32.h>
|
||
#include <windows.h>
|
||
#include <winnt.h>
|
||
#include <winternl.h>
|
||
#define HAVE_SYS_STAT_H
|
||
#define HAVE_SYS_TYPES_H
|
||
typedef HANDLE mdbx_thread_t;
|
||
typedef unsigned mdbx_thread_key_t;
|
||
#define MDBX_OSAL_SECTION HANDLE
|
||
#define MAP_FAILED NULL
|
||
#define HIGH_DWORD(v) ((DWORD)((sizeof(v) > 4) ? ((uint64_t)(v) >> 32) : 0))
|
||
#define THREAD_CALL WINAPI
|
||
#define THREAD_RESULT DWORD
|
||
typedef struct {
|
||
HANDLE mutex;
|
||
HANDLE event;
|
||
} mdbx_condmutex_t;
|
||
typedef CRITICAL_SECTION mdbx_fastmutex_t;
|
||
|
||
#ifdef MDBX_AVOID_CRT
|
||
#ifndef mdbx_malloc
|
||
static inline void *mdbx_malloc(size_t bytes) {
|
||
return LocalAlloc(LMEM_FIXED, bytes);
|
||
}
|
||
#endif /* mdbx_malloc */
|
||
|
||
#ifndef mdbx_calloc
|
||
static inline void *mdbx_calloc(size_t nelem, size_t size) {
|
||
return LocalAlloc(LMEM_FIXED | LMEM_ZEROINIT, nelem * size);
|
||
}
|
||
#endif /* mdbx_calloc */
|
||
|
||
#ifndef mdbx_realloc
|
||
static inline void *mdbx_realloc(void *ptr, size_t bytes) {
|
||
return LocalReAlloc(ptr, bytes, LMEM_MOVEABLE);
|
||
}
|
||
#endif /* mdbx_realloc */
|
||
|
||
#ifndef mdbx_free
|
||
#define mdbx_free LocalFree
|
||
#endif /* mdbx_free */
|
||
#else
|
||
#define mdbx_malloc malloc
|
||
#define mdbx_calloc calloc
|
||
#define mdbx_realloc realloc
|
||
#define mdbx_free free
|
||
#define mdbx_strdup _strdup
|
||
#endif /* MDBX_AVOID_CRT */
|
||
|
||
#ifndef snprintf
|
||
#define snprintf _snprintf /* ntdll */
|
||
#endif
|
||
|
||
#ifndef vsnprintf
|
||
#define vsnprintf _vsnprintf /* ntdll */
|
||
#endif
|
||
|
||
#else /*----------------------------------------------------------------------*/
|
||
|
||
#include <pthread.h>
|
||
#include <signal.h>
|
||
#include <sys/file.h>
|
||
#include <sys/mman.h>
|
||
#include <sys/param.h>
|
||
#include <sys/stat.h>
|
||
#include <sys/uio.h>
|
||
#include <unistd.h>
|
||
typedef pthread_t mdbx_thread_t;
|
||
typedef pthread_key_t mdbx_thread_key_t;
|
||
#define INVALID_HANDLE_VALUE (-1)
|
||
#define THREAD_CALL
|
||
#define THREAD_RESULT void *
|
||
typedef struct {
|
||
pthread_mutex_t mutex;
|
||
pthread_cond_t cond;
|
||
} mdbx_condmutex_t;
|
||
typedef pthread_mutex_t mdbx_fastmutex_t;
|
||
#define mdbx_malloc malloc
|
||
#define mdbx_calloc calloc
|
||
#define mdbx_realloc realloc
|
||
#define mdbx_free free
|
||
#define mdbx_strdup strdup
|
||
#endif /* Platform */
|
||
|
||
/* *INDENT-OFF* */
|
||
/* clang-format off */
|
||
#if defined(HAVE_SYS_STAT_H) || __has_include(<sys/stat.h>)
|
||
#include <sys/stat.h>
|
||
#endif
|
||
#if defined(HAVE_SYS_TYPES_H) || __has_include(<sys/types.h>)
|
||
#include <sys/types.h>
|
||
#endif
|
||
#if defined(HAVE_SYS_FILE_H) || __has_include(<sys/file.h>)
|
||
#include <sys/file.h>
|
||
#endif
|
||
/* *INDENT-ON* */
|
||
/* clang-format on */
|
||
|
||
#ifndef SSIZE_MAX
|
||
#define SSIZE_MAX INTPTR_MAX
|
||
#endif
|
||
|
||
#if !defined(MADV_DODUMP) && defined(MADV_CORE)
|
||
#define MADV_DODUMP MADV_CORE
|
||
#endif /* MADV_CORE -> MADV_DODUMP */
|
||
|
||
#if !defined(MADV_DONTDUMP) && defined(MADV_NOCORE)
|
||
#define MADV_DONTDUMP MADV_NOCORE
|
||
#endif /* MADV_NOCORE -> MADV_DONTDUMP */
|
||
|
||
#ifndef MADV_REMOVE_OR_FREE_OR_DONTNEED
|
||
#ifdef MADV_REMOVE
|
||
#define MADV_REMOVE_OR_FREE_OR_DONTNEED MADV_REMOVE
|
||
#elif defined(MADV_FREE)
|
||
#define MADV_REMOVE_OR_FREE_OR_DONTNEED MADV_FREE
|
||
#elif defined(MADV_DONTNEED)
|
||
#define MADV_REMOVE_OR_FREE_OR_DONTNEED MADV_DONTNEED
|
||
#endif
|
||
#endif /* MADV_REMOVE_OR_FREE_OR_DONTNEED */
|
||
|
||
#if defined(i386) || defined(__386) || defined(__i386) || defined(__i386__) || \
|
||
defined(i486) || defined(__i486) || defined(__i486__) || \
|
||
defined(i586) | defined(__i586) || defined(__i586__) || defined(i686) || \
|
||
defined(__i686) || defined(__i686__) || defined(_M_IX86) || \
|
||
defined(_X86_) || defined(__THW_INTEL__) || defined(__I86__) || \
|
||
defined(__INTEL__) || defined(__x86_64) || defined(__x86_64__) || \
|
||
defined(__amd64__) || defined(__amd64) || defined(_M_X64) || \
|
||
defined(_M_AMD64) || defined(__IA32__) || defined(__INTEL__)
|
||
#ifndef __ia32__
|
||
/* LY: define neutral __ia32__ for x86 and x86-64 archs */
|
||
#define __ia32__ 1
|
||
#endif /* __ia32__ */
|
||
#if !defined(__amd64__) && (defined(__x86_64) || defined(__x86_64__) || \
|
||
defined(__amd64) || defined(_M_X64))
|
||
/* LY: define trusty __amd64__ for all AMD64/x86-64 arch */
|
||
#define __amd64__ 1
|
||
#endif /* __amd64__ */
|
||
#endif /* all x86 */
|
||
|
||
#if !defined(UNALIGNED_OK)
|
||
#if (defined(__ia32__) || defined(__e2k__) || \
|
||
defined(__ARM_FEATURE_UNALIGNED)) && \
|
||
!defined(__ALIGNED__)
|
||
#define UNALIGNED_OK 1
|
||
#else
|
||
#define UNALIGNED_OK 0
|
||
#endif
|
||
#endif /* UNALIGNED_OK */
|
||
|
||
#if (-6 & 5) || CHAR_BIT != 8 || UINT_MAX < 0xffffffff || ULONG_MAX % 0xFFFF
|
||
#error \
|
||
"Sanity checking failed: Two's complement, reasonably sized integer types"
|
||
#endif
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Compiler's includes for builtins/intrinsics */
|
||
|
||
#if defined(_MSC_VER) || defined(__INTEL_COMPILER)
|
||
#include <intrin.h>
|
||
#elif __GNUC_PREREQ(4, 4) || defined(__clang__)
|
||
#if defined(__ia32__) || defined(__e2k__)
|
||
#include <x86intrin.h>
|
||
#endif /* __ia32__ */
|
||
#if defined(__ia32__)
|
||
#include <cpuid.h>
|
||
#endif /* __ia32__ */
|
||
#elif defined(__SUNPRO_C) || defined(__sun) || defined(sun)
|
||
#include <mbarrier.h>
|
||
#elif (defined(_HPUX_SOURCE) || defined(__hpux) || defined(__HP_aCC)) && \
|
||
(defined(HP_IA64) || defined(__ia64))
|
||
#include <machine/sys/inline.h>
|
||
#elif defined(__IBMC__) && defined(__powerpc)
|
||
#include <atomic.h>
|
||
#elif defined(_AIX)
|
||
#include <builtins.h>
|
||
#include <sys/atomic_op.h>
|
||
#elif (defined(__osf__) && defined(__DECC)) || defined(__alpha)
|
||
#include <c_asm.h>
|
||
#include <machine/builtins.h>
|
||
#elif defined(__MWERKS__)
|
||
/* CodeWarrior - troubles ? */
|
||
#pragma gcc_extensions
|
||
#elif defined(__SNC__)
|
||
/* Sony PS3 - troubles ? */
|
||
#elif defined(__hppa__) || defined(__hppa)
|
||
#include <machine/inline.h>
|
||
#else
|
||
#error Unsupported C compiler, please use GNU C 4.4 or newer
|
||
#endif /* Compiler */
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Byteorder */
|
||
|
||
#if !defined(__BYTE_ORDER__) || !defined(__ORDER_LITTLE_ENDIAN__) || \
|
||
!defined(__ORDER_BIG_ENDIAN__)
|
||
|
||
/* *INDENT-OFF* */
|
||
/* clang-format off */
|
||
#if defined(__GLIBC__) || defined(__GNU_LIBRARY__) || defined(__ANDROID__) || \
|
||
defined(HAVE_ENDIAN_H) || __has_include(<endian.h>)
|
||
#include <endian.h>
|
||
#elif defined(__APPLE__) || defined(__MACH__) || defined(__OpenBSD__) || \
|
||
defined(HAVE_MACHINE_ENDIAN_H) || __has_include(<machine/endian.h>)
|
||
#include <machine/endian.h>
|
||
#elif defined(HAVE_SYS_ISA_DEFS_H) || __has_include(<sys/isa_defs.h>)
|
||
#include <sys/isa_defs.h>
|
||
#elif (defined(HAVE_SYS_TYPES_H) && defined(HAVE_SYS_ENDIAN_H)) || \
|
||
(__has_include(<sys/types.h>) && __has_include(<sys/endian.h>))
|
||
#include <sys/endian.h>
|
||
#include <sys/types.h>
|
||
#elif defined(__bsdi__) || defined(__DragonFly__) || defined(__FreeBSD__) || \
|
||
defined(__NETBSD__) || defined(__NetBSD__) || \
|
||
defined(HAVE_SYS_PARAM_H) || __has_include(<sys/param.h>)
|
||
#include <sys/param.h>
|
||
#endif /* OS */
|
||
/* *INDENT-ON* */
|
||
/* clang-format on */
|
||
|
||
#if defined(__BYTE_ORDER) && defined(__LITTLE_ENDIAN) && defined(__BIG_ENDIAN)
|
||
#define __ORDER_LITTLE_ENDIAN__ __LITTLE_ENDIAN
|
||
#define __ORDER_BIG_ENDIAN__ __BIG_ENDIAN
|
||
#define __BYTE_ORDER__ __BYTE_ORDER
|
||
#elif defined(_BYTE_ORDER) && defined(_LITTLE_ENDIAN) && defined(_BIG_ENDIAN)
|
||
#define __ORDER_LITTLE_ENDIAN__ _LITTLE_ENDIAN
|
||
#define __ORDER_BIG_ENDIAN__ _BIG_ENDIAN
|
||
#define __BYTE_ORDER__ _BYTE_ORDER
|
||
#else
|
||
#define __ORDER_LITTLE_ENDIAN__ 1234
|
||
#define __ORDER_BIG_ENDIAN__ 4321
|
||
|
||
#if defined(__LITTLE_ENDIAN__) || \
|
||
(defined(_LITTLE_ENDIAN) && !defined(_BIG_ENDIAN)) || \
|
||
defined(__ARMEL__) || defined(__THUMBEL__) || defined(__AARCH64EL__) || \
|
||
defined(__MIPSEL__) || defined(_MIPSEL) || defined(__MIPSEL) || \
|
||
defined(_M_ARM) || defined(_M_ARM64) || defined(__e2k__) || \
|
||
defined(__elbrus_4c__) || defined(__elbrus_8c__) || defined(__bfin__) || \
|
||
defined(__BFIN__) || defined(__ia64__) || defined(_IA64) || \
|
||
defined(__IA64__) || defined(__ia64) || defined(_M_IA64) || \
|
||
defined(__itanium__) || defined(__ia32__) || defined(__CYGWIN__) || \
|
||
defined(_WIN64) || defined(_WIN32) || defined(__TOS_WIN__) || \
|
||
defined(__WINDOWS__)
|
||
#define __BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__
|
||
|
||
#elif defined(__BIG_ENDIAN__) || \
|
||
(defined(_BIG_ENDIAN) && !defined(_LITTLE_ENDIAN)) || \
|
||
defined(__ARMEB__) || defined(__THUMBEB__) || defined(__AARCH64EB__) || \
|
||
defined(__MIPSEB__) || defined(_MIPSEB) || defined(__MIPSEB) || \
|
||
defined(__m68k__) || defined(M68000) || defined(__hppa__) || \
|
||
defined(__hppa) || defined(__HPPA__) || defined(__sparc__) || \
|
||
defined(__sparc) || defined(__370__) || defined(__THW_370__) || \
|
||
defined(__s390__) || defined(__s390x__) || defined(__SYSC_ZARCH__)
|
||
#define __BYTE_ORDER__ __ORDER_BIG_ENDIAN__
|
||
|
||
#else
|
||
#error __BYTE_ORDER__ should be defined.
|
||
#endif /* Arch */
|
||
|
||
#endif
|
||
#endif /* __BYTE_ORDER__ || __ORDER_LITTLE_ENDIAN__ || __ORDER_BIG_ENDIAN__ */
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Memory/Compiler barriers, cache coherence */
|
||
|
||
static __inline void mdbx_compiler_barrier(void) {
|
||
#if defined(__clang__) || defined(__GNUC__)
|
||
__asm__ __volatile__("" ::: "memory");
|
||
#elif defined(_MSC_VER)
|
||
_ReadWriteBarrier();
|
||
#elif defined(__INTEL_COMPILER) /* LY: Intel Compiler may mimic GCC and MSC */
|
||
__memory_barrier();
|
||
if (type > MDBX_BARRIER_COMPILER)
|
||
#if defined(__ia64__) || defined(__ia64) || defined(_M_IA64)
|
||
__mf();
|
||
#elif defined(__i386__) || defined(__x86_64__)
|
||
_mm_mfence();
|
||
#else
|
||
#error "Unknown target for Intel Compiler, please report to us."
|
||
#endif
|
||
#elif defined(__SUNPRO_C) || defined(__sun) || defined(sun)
|
||
__compiler_barrier();
|
||
#elif (defined(_HPUX_SOURCE) || defined(__hpux) || defined(__HP_aCC)) && \
|
||
(defined(HP_IA64) || defined(__ia64))
|
||
_Asm_sched_fence(/* LY: no-arg meaning 'all expect ALU', e.g. 0x3D3D */);
|
||
#elif defined(_AIX) || defined(__ppc__) || defined(__powerpc__) || \
|
||
defined(__ppc64__) || defined(__powerpc64__)
|
||
__fence();
|
||
#else
|
||
#error "Could not guess the kind of compiler, please report to us."
|
||
#endif
|
||
}
|
||
|
||
static __inline void mdbx_memory_barrier(void) {
|
||
#if __has_extension(c_atomic) || __has_extension(cxx_atomic)
|
||
__c11_atomic_thread_fence(__ATOMIC_SEQ_CST);
|
||
#elif defined(__ATOMIC_SEQ_CST)
|
||
__atomic_thread_fence(__ATOMIC_SEQ_CST);
|
||
#elif defined(__clang__) || defined(__GNUC__)
|
||
__sync_synchronize();
|
||
#elif defined(_MSC_VER)
|
||
MemoryBarrier();
|
||
#elif defined(__INTEL_COMPILER) /* LY: Intel Compiler may mimic GCC and MSC */
|
||
#if defined(__ia64__) || defined(__ia64) || defined(_M_IA64)
|
||
__mf();
|
||
#elif defined(__i386__) || defined(__x86_64__)
|
||
_mm_mfence();
|
||
#else
|
||
#error "Unknown target for Intel Compiler, please report to us."
|
||
#endif
|
||
#elif defined(__SUNPRO_C) || defined(__sun) || defined(sun)
|
||
__machine_rw_barrier();
|
||
#elif (defined(_HPUX_SOURCE) || defined(__hpux) || defined(__HP_aCC)) && \
|
||
(defined(HP_IA64) || defined(__ia64))
|
||
_Asm_mf();
|
||
#elif defined(_AIX) || defined(__ppc__) || defined(__powerpc__) || \
|
||
defined(__ppc64__) || defined(__powerpc64__)
|
||
__lwsync();
|
||
#else
|
||
#error "Could not guess the kind of compiler, please report to us."
|
||
#endif
|
||
}
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Cache coherence and invalidation */
|
||
|
||
#ifndef MDBX_CPU_WRITEBACK_IS_COHERENT
|
||
#if defined(__ia32__) || defined(__e2k__) || defined(__hppa) || \
|
||
defined(__hppa__)
|
||
#define MDBX_CPU_WRITEBACK_IS_COHERENT 1
|
||
#else
|
||
#define MDBX_CPU_WRITEBACK_IS_COHERENT 0
|
||
#endif
|
||
#endif /* MDBX_CPU_WRITEBACK_IS_COHERENT */
|
||
|
||
#ifndef MDBX_CACHELINE_SIZE
|
||
#if defined(SYSTEM_CACHE_ALIGNMENT_SIZE)
|
||
#define MDBX_CACHELINE_SIZE SYSTEM_CACHE_ALIGNMENT_SIZE
|
||
#elif defined(__ia64__) || defined(__ia64) || defined(_M_IA64)
|
||
#define MDBX_CACHELINE_SIZE 128
|
||
#else
|
||
#define MDBX_CACHELINE_SIZE 64
|
||
#endif
|
||
#endif /* MDBX_CACHELINE_SIZE */
|
||
|
||
#if MDBX_CPU_WRITEBACK_IS_COHERENT
|
||
#define mdbx_flush_noncoherent_cpu_writeback() mdbx_compiler_barrier()
|
||
#else
|
||
#define mdbx_flush_noncoherent_cpu_writeback() mdbx_memory_barrier()
|
||
#endif
|
||
|
||
#if __has_include(<sys/cachectl.h>)
|
||
#include <sys/cachectl.h>
|
||
#elif defined(__mips) || defined(__mips__) || defined(__mips64) || \
|
||
defined(__mips64__) || defined(_M_MRX000) || defined(_MIPS_) || \
|
||
defined(__MWERKS__) || defined(__sgi)
|
||
/* MIPS should have explicit cache control */
|
||
#include <sys/cachectl.h>
|
||
#endif
|
||
|
||
#ifndef MDBX_CPU_CACHE_MMAP_NONCOHERENT
|
||
#if defined(__mips) || defined(__mips__) || defined(__mips64) || \
|
||
defined(__mips64__) || defined(_M_MRX000) || defined(_MIPS_) || \
|
||
defined(__MWERKS__) || defined(__sgi)
|
||
/* MIPS has cache coherency issues. */
|
||
#define MDBX_CPU_CACHE_MMAP_NONCOHERENT 1
|
||
#else
|
||
/* LY: assume no relevant mmap/dcache issues. */
|
||
#define MDBX_CPU_CACHE_MMAP_NONCOHERENT 0
|
||
#endif
|
||
#endif /* ndef MDBX_CPU_CACHE_MMAP_NONCOHERENT */
|
||
|
||
static __inline void mdbx_invalidate_mmap_noncoherent_cache(void *addr,
|
||
size_t nbytes) {
|
||
#if MDBX_CPU_CACHE_MMAP_NONCOHERENT
|
||
#ifdef DCACHE
|
||
/* MIPS has cache coherency issues.
|
||
* Note: for any nbytes >= on-chip cache size, entire is flushed. */
|
||
cacheflush(addr, nbytes, DCACHE);
|
||
#else
|
||
#error "Oops, cacheflush() not available"
|
||
#endif /* DCACHE */
|
||
#else /* MDBX_CPU_CACHE_MMAP_NONCOHERENT */
|
||
(void)addr;
|
||
(void)nbytes;
|
||
#endif /* MDBX_CPU_CACHE_MMAP_NONCOHERENT */
|
||
}
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* libc compatibility stuff */
|
||
|
||
#if (!defined(__GLIBC__) && __GLIBC_PREREQ(2, 1)) && \
|
||
(defined(_GNU_SOURCE) || defined(_BSD_SOURCE))
|
||
#define mdbx_asprintf asprintf
|
||
#define mdbx_vasprintf vasprintf
|
||
#else
|
||
MDBX_INTERNAL_FUNC __printf_args(2, 3) int __maybe_unused
|
||
mdbx_asprintf(char **strp, const char *fmt, ...);
|
||
MDBX_INTERNAL_FUNC int mdbx_vasprintf(char **strp, const char *fmt, va_list ap);
|
||
#endif
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* OS abstraction layer stuff */
|
||
|
||
/* max bytes to write in one call */
|
||
#define MAX_WRITE UINT32_C(0x3fff0000)
|
||
|
||
#if defined(__linux__) || defined(__gnu_linux__)
|
||
MDBX_INTERNAL_VAR uint32_t mdbx_linux_kernel_version;
|
||
#endif /* Linux */
|
||
|
||
/* Get the size of a memory page for the system.
|
||
* This is the basic size that the platform's memory manager uses, and is
|
||
* fundamental to the use of memory-mapped files. */
|
||
static __inline size_t mdbx_syspagesize(void) {
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
SYSTEM_INFO si;
|
||
GetSystemInfo(&si);
|
||
return si.dwPageSize;
|
||
#else
|
||
return sysconf(_SC_PAGE_SIZE);
|
||
#endif
|
||
}
|
||
|
||
#ifndef mdbx_strdup
|
||
LIBMDBX_API char *mdbx_strdup(const char *str);
|
||
#endif
|
||
|
||
static __inline int mdbx_get_errno(void) {
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
DWORD rc = GetLastError();
|
||
#else
|
||
int rc = errno;
|
||
#endif
|
||
return rc;
|
||
}
|
||
|
||
#ifndef mdbx_memalign_alloc
|
||
MDBX_INTERNAL_FUNC int mdbx_memalign_alloc(size_t alignment, size_t bytes,
|
||
void **result);
|
||
#endif
|
||
#ifndef mdbx_memalign_free
|
||
MDBX_INTERNAL_FUNC void mdbx_memalign_free(void *ptr);
|
||
#endif
|
||
|
||
MDBX_INTERNAL_FUNC int mdbx_condmutex_init(mdbx_condmutex_t *condmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_condmutex_lock(mdbx_condmutex_t *condmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_condmutex_unlock(mdbx_condmutex_t *condmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_condmutex_signal(mdbx_condmutex_t *condmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_condmutex_wait(mdbx_condmutex_t *condmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_condmutex_destroy(mdbx_condmutex_t *condmutex);
|
||
|
||
MDBX_INTERNAL_FUNC int mdbx_fastmutex_init(mdbx_fastmutex_t *fastmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_fastmutex_acquire(mdbx_fastmutex_t *fastmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_fastmutex_release(mdbx_fastmutex_t *fastmutex);
|
||
MDBX_INTERNAL_FUNC int mdbx_fastmutex_destroy(mdbx_fastmutex_t *fastmutex);
|
||
|
||
MDBX_INTERNAL_FUNC int mdbx_pwritev(mdbx_filehandle_t fd, struct iovec *iov,
|
||
int iovcnt, uint64_t offset,
|
||
size_t expected_written);
|
||
MDBX_INTERNAL_FUNC int mdbx_pread(mdbx_filehandle_t fd, void *buf, size_t count,
|
||
uint64_t offset);
|
||
MDBX_INTERNAL_FUNC int mdbx_pwrite(mdbx_filehandle_t fd, const void *buf,
|
||
size_t count, uint64_t offset);
|
||
|
||
MDBX_INTERNAL_FUNC int
|
||
mdbx_thread_create(mdbx_thread_t *thread,
|
||
THREAD_RESULT(THREAD_CALL *start_routine)(void *),
|
||
void *arg);
|
||
MDBX_INTERNAL_FUNC int mdbx_thread_join(mdbx_thread_t thread);
|
||
|
||
enum mdbx_syncmode_bits {
|
||
MDBX_SYNC_DATA = 1,
|
||
MDBX_SYNC_SIZE = 2,
|
||
MDBX_SYNC_IODQ = 4
|
||
};
|
||
|
||
MDBX_INTERNAL_FUNC int mdbx_filesync(mdbx_filehandle_t fd,
|
||
enum mdbx_syncmode_bits mode_bits);
|
||
MDBX_INTERNAL_FUNC int mdbx_ftruncate(mdbx_filehandle_t fd, uint64_t length);
|
||
MDBX_INTERNAL_FUNC int mdbx_fseek(mdbx_filehandle_t fd, uint64_t pos);
|
||
MDBX_INTERNAL_FUNC int mdbx_filesize(mdbx_filehandle_t fd, uint64_t *length);
|
||
MDBX_INTERNAL_FUNC int mdbx_openfile(const char *pathname, int flags,
|
||
mode_t mode, mdbx_filehandle_t *fd,
|
||
bool exclusive);
|
||
MDBX_INTERNAL_FUNC int mdbx_closefile(mdbx_filehandle_t fd);
|
||
MDBX_INTERNAL_FUNC int mdbx_removefile(const char *pathname);
|
||
|
||
typedef struct mdbx_mmap_param {
|
||
union {
|
||
void *address;
|
||
uint8_t *dxb;
|
||
struct MDBX_lockinfo *lck;
|
||
};
|
||
mdbx_filehandle_t fd;
|
||
size_t length; /* mapping length, but NOT a size of file or DB */
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
size_t current; /* mapped region size, e.g. file and DB */
|
||
uint64_t filesize;
|
||
#endif
|
||
#ifdef MDBX_OSAL_SECTION
|
||
MDBX_OSAL_SECTION section;
|
||
#endif
|
||
} mdbx_mmap_t;
|
||
|
||
MDBX_INTERNAL_FUNC int mdbx_mmap(int flags, mdbx_mmap_t *map, size_t must,
|
||
size_t limit);
|
||
MDBX_INTERNAL_FUNC int mdbx_munmap(mdbx_mmap_t *map);
|
||
MDBX_INTERNAL_FUNC int mdbx_mresize(int flags, mdbx_mmap_t *map, size_t current,
|
||
size_t wanna);
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
typedef struct {
|
||
unsigned limit, count;
|
||
HANDLE handles[31];
|
||
} mdbx_handle_array_t;
|
||
MDBX_INTERNAL_FUNC int
|
||
mdbx_suspend_threads_before_remap(MDBX_env *env, mdbx_handle_array_t **array);
|
||
MDBX_INTERNAL_FUNC int
|
||
mdbx_resume_threads_after_remap(mdbx_handle_array_t *array);
|
||
#endif /* Windows */
|
||
MDBX_INTERNAL_FUNC int mdbx_msync(mdbx_mmap_t *map, size_t offset,
|
||
size_t length, int async);
|
||
MDBX_INTERNAL_FUNC int mdbx_check4nonlocal(mdbx_filehandle_t handle, int flags);
|
||
|
||
static __inline mdbx_pid_t mdbx_getpid(void) {
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
return GetCurrentProcessId();
|
||
#else
|
||
return getpid();
|
||
#endif
|
||
}
|
||
|
||
static __inline mdbx_tid_t mdbx_thread_self(void) {
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
return GetCurrentThreadId();
|
||
#else
|
||
return pthread_self();
|
||
#endif
|
||
}
|
||
|
||
MDBX_INTERNAL_FUNC void __maybe_unused mdbx_osal_jitter(bool tiny);
|
||
MDBX_INTERNAL_FUNC uint64_t mdbx_osal_monotime(void);
|
||
MDBX_INTERNAL_FUNC uint64_t
|
||
mdbx_osal_16dot16_to_monotime(uint32_t seconds_16dot16);
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* lck stuff */
|
||
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
#undef MDBX_OSAL_LOCK
|
||
#define MDBX_OSAL_LOCK_SIGN UINT32_C(0xF10C)
|
||
#else
|
||
#define MDBX_OSAL_LOCK pthread_mutex_t
|
||
#define MDBX_OSAL_LOCK_SIGN UINT32_C(0x8017)
|
||
#endif /* MDBX_OSAL_LOCK */
|
||
|
||
/// \brief Инициализация объектов синхронизации связанных с экземпляром MDBX_env
|
||
/// как общик в LCK-файле, так и внутри текущего процесса.
|
||
/// \param
|
||
/// global_uniqueness_flag = true - означает что сейчас в системе нет других
|
||
/// процессов работающих с БД и LCK-файлом. Соответственно функция ДОЛЖНА
|
||
/// инициализировать разделяемые объекты синхронизации расположенные
|
||
/// в отображенном в память LCK-файле.
|
||
/// global_uniqueness_flag = false - означает что в системе есть хотя-бы
|
||
/// один другой процесс уже работающий с БД и LCK-файлом, в том числе
|
||
/// БД уже может быть открыта текущим процессом. Соответственно функция
|
||
/// НЕ должна инициализировать уже используемые разделяемые объекты
|
||
/// синхронизации расположенные в отображенном в память LCK-файле.
|
||
/// \return Код ошибки или 0 в случае успеха.
|
||
MDBX_INTERNAL_FUNC int mdbx_lck_init(MDBX_env *env, int global_uniqueness_flag);
|
||
|
||
/// \brief Отключение от общих межпроцесных объектов и разрушение объектов
|
||
/// синхронизации внутри текущего процесса связанных с экземпляром MDBX_env.
|
||
/// \param
|
||
/// inprocess_neighbor = NULL - если в текущем процессе нет других экземпляров
|
||
/// MDBX_env связанных с закрываемой БД. Соответственно функция ДОЛЖНА
|
||
/// своими средствами проверить, есть ли другие процесса работающие с БД
|
||
/// и LCK-файлом, и в зависимости от этого разрушить или сохранить бъекты
|
||
/// синхронизации расположенные в отображенном в память LCK-файле.
|
||
/// inprocess_neighbor = не-NULL - тогда он указывает на другой (любой
|
||
/// в случае нескольких) экземпляр MDBX_env работающей с БД и LCK-файлом
|
||
/// внутри текущего процесса. Соответственно, функция НЕ должна пытаться
|
||
/// захватывать эксклюзивную блокировки и/или пытаться разрушить общие
|
||
/// объекты синхронизации связанные с БД и LCK-файлом. Кроме этого,
|
||
/// реализация ДОЛЖНА обеспечить корректность работы других экземпляров
|
||
/// MDBX_env внутри процесса - например, восстановить POSIX-fcntl блокировки
|
||
/// после закрытия файловых дескрипторов.
|
||
/// \return Код ошибки (MDBX_PANIC) или 0 в случае успеха.
|
||
MDBX_INTERNAL_FUNC int mdbx_lck_destroy(MDBX_env *env,
|
||
MDBX_env *inprocess_neighbor);
|
||
|
||
/// \brief Подключение к общим межпроцесным объектам блокировки с попыткой
|
||
/// захвата блокировки максимального уровня (разделяемой при недоступности
|
||
/// эксклюзивной).
|
||
/// В зависимости от реализации и/или платформы (Windows) может
|
||
/// захватывать блокировку не-операционного супер-уровня (например, для
|
||
/// инициализации разделяемых объектов синхронизации), которая затем будет
|
||
/// понижена до операционно-эксклюзивной или разделяемой посредством
|
||
/// явного вызова mdbx_lck_downgrade().
|
||
/// \return
|
||
/// MDBX_RESULT_TRUE (-1) - если удалось захватить эксклюзивную блокировку и,
|
||
/// следовательно, текущий процесс является первым и единственным
|
||
/// после предыдущего использования БД.
|
||
/// MDBX_RESULT_FALSE (0) - если удалось захватить только разделяемую
|
||
/// блокировку и, следовательно, БД уже открыта и используется другими
|
||
/// процессами.
|
||
/// Иначе (не 0 и не -1) - код ошибки.
|
||
MDBX_INTERNAL_FUNC int mdbx_lck_seize(MDBX_env *env);
|
||
|
||
/// \brief Снижает уровень первоначальной захваченной блокировки до
|
||
/// операционного уровня определяемого аргументом. Смысл функции в возврате
|
||
/// в операционный режим:
|
||
/// - разблокирование других процессов ожидающих доступа, т.е если
|
||
/// (env->me_flags & MDBX_EXCLUSIVE) != 0, то другие процессы должны узнать
|
||
/// о невозможности доступа, а не ждать его.
|
||
/// - снятия блокировок мешающих работе с файлом (актуально для Windows).
|
||
/// (env->me_flags & MDBX_EXCLUSIVE) == 0 - понижение до разделяемой
|
||
/// блокировки. (env->me_flags & MDBX_EXCLUSIVE) != 0 - понижение до
|
||
/// эксклюзивной операционной блокировки.
|
||
/// \return Код ошибки или 0 в случае успеха.
|
||
MDBX_INTERNAL_FUNC int mdbx_lck_downgrade(MDBX_env *env);
|
||
|
||
/// \brief Блокирует lck-файл и/или таблицу читателей для (де)регистрации.
|
||
/// \return Код ошибки или 0 в случае успеха.
|
||
MDBX_INTERNAL_FUNC int mdbx_rdt_lock(MDBX_env *env);
|
||
|
||
/// \brief Разблокирует lck-файл и/или таблицу читателей после (де)регистрации.
|
||
MDBX_INTERNAL_FUNC void mdbx_rdt_unlock(MDBX_env *env);
|
||
|
||
/// \brief Захватывает блокировку для изменения БД (при старте пишущей
|
||
/// транзакции). Транзакции чтения при этом никак не блокируются.
|
||
/// Объявлена LIBMDBX_API так как используется в mdbx_chk.
|
||
/// \return Код ошибки или 0 в случае успеха.
|
||
LIBMDBX_API int mdbx_txn_lock(MDBX_env *env, bool dontwait);
|
||
|
||
/// \brief Освобождает блокировку по окончанию изменения БД (после завершения
|
||
/// пишущей транзакции).
|
||
/// Объявлена LIBMDBX_API так как используется в mdbx_chk.
|
||
LIBMDBX_API void mdbx_txn_unlock(MDBX_env *env);
|
||
|
||
/// \brief Устанавливает alive-флажок присутствия (индицирующую блокировку)
|
||
/// читателя для pid текущего процесса. Функции может выполнить не более
|
||
/// необходимого минимума для корректной работы mdbx_rpid_check() в других
|
||
/// процессах.
|
||
/// \return Код ошибки или 0 в случае успеха.
|
||
MDBX_INTERNAL_FUNC int mdbx_rpid_set(MDBX_env *env);
|
||
|
||
/// \brief Снимает alive-флажок присутствия (индицирующую блокировку)
|
||
/// читателя для pid текущего процесса. Функции может выполнить не более
|
||
/// необходимого минимума для корректной работы mdbx_rpid_check() в других
|
||
/// процессах.
|
||
/// \return Код ошибки или 0 в случае успеха.
|
||
MDBX_INTERNAL_FUNC int mdbx_rpid_clear(MDBX_env *env);
|
||
|
||
/// \brief Проверяет жив ли процесс-читатель с заданным pid
|
||
/// по alive-флажку присутствия (индицирующей блокировку),
|
||
/// либо любым другим способом.
|
||
/// \return
|
||
/// MDBX_RESULT_TRUE (-1) - если процесс-читатель с соответствующим pid жив
|
||
/// и работает с БД (индицирующая блокировка присутствует).
|
||
/// MDBX_RESULT_FALSE (0) - если процесс-читатель с соответствующим pid
|
||
/// отсутствует или не работает с БД (индицирующая блокировка отсутствует).
|
||
/// Иначе (не 0 и не -1) - код ошибки.
|
||
MDBX_INTERNAL_FUNC int mdbx_rpid_check(MDBX_env *env, mdbx_pid_t pid);
|
||
|
||
#if defined(_WIN32) || defined(_WIN64)
|
||
typedef union MDBX_srwlock {
|
||
struct {
|
||
long volatile readerCount;
|
||
long volatile writerCount;
|
||
};
|
||
RTL_SRWLOCK native;
|
||
} MDBX_srwlock;
|
||
|
||
typedef void(WINAPI *MDBX_srwlock_function)(MDBX_srwlock *);
|
||
MDBX_INTERNAL_VAR MDBX_srwlock_function mdbx_srwlock_Init,
|
||
mdbx_srwlock_AcquireShared, mdbx_srwlock_ReleaseShared,
|
||
mdbx_srwlock_AcquireExclusive, mdbx_srwlock_ReleaseExclusive;
|
||
|
||
typedef BOOL(WINAPI *MDBX_GetFileInformationByHandleEx)(
|
||
_In_ HANDLE hFile, _In_ FILE_INFO_BY_HANDLE_CLASS FileInformationClass,
|
||
_Out_ LPVOID lpFileInformation, _In_ DWORD dwBufferSize);
|
||
MDBX_INTERNAL_VAR MDBX_GetFileInformationByHandleEx
|
||
mdbx_GetFileInformationByHandleEx;
|
||
|
||
typedef BOOL(WINAPI *MDBX_GetVolumeInformationByHandleW)(
|
||
_In_ HANDLE hFile, _Out_opt_ LPWSTR lpVolumeNameBuffer,
|
||
_In_ DWORD nVolumeNameSize, _Out_opt_ LPDWORD lpVolumeSerialNumber,
|
||
_Out_opt_ LPDWORD lpMaximumComponentLength,
|
||
_Out_opt_ LPDWORD lpFileSystemFlags,
|
||
_Out_opt_ LPWSTR lpFileSystemNameBuffer, _In_ DWORD nFileSystemNameSize);
|
||
MDBX_INTERNAL_VAR MDBX_GetVolumeInformationByHandleW
|
||
mdbx_GetVolumeInformationByHandleW;
|
||
|
||
typedef DWORD(WINAPI *MDBX_GetFinalPathNameByHandleW)(_In_ HANDLE hFile,
|
||
_Out_ LPWSTR lpszFilePath,
|
||
_In_ DWORD cchFilePath,
|
||
_In_ DWORD dwFlags);
|
||
MDBX_INTERNAL_VAR MDBX_GetFinalPathNameByHandleW mdbx_GetFinalPathNameByHandleW;
|
||
|
||
typedef BOOL(WINAPI *MDBX_SetFileInformationByHandle)(
|
||
_In_ HANDLE hFile, _In_ FILE_INFO_BY_HANDLE_CLASS FileInformationClass,
|
||
_Out_ LPVOID lpFileInformation, _In_ DWORD dwBufferSize);
|
||
MDBX_INTERNAL_VAR MDBX_SetFileInformationByHandle
|
||
mdbx_SetFileInformationByHandle;
|
||
|
||
typedef NTSTATUS(NTAPI *MDBX_NtFsControlFile)(
|
||
IN HANDLE FileHandle, IN OUT HANDLE Event,
|
||
IN OUT PVOID /* PIO_APC_ROUTINE */ ApcRoutine, IN OUT PVOID ApcContext,
|
||
OUT PIO_STATUS_BLOCK IoStatusBlock, IN ULONG FsControlCode,
|
||
IN OUT PVOID InputBuffer, IN ULONG InputBufferLength,
|
||
OUT OPTIONAL PVOID OutputBuffer, IN ULONG OutputBufferLength);
|
||
MDBX_INTERNAL_VAR MDBX_NtFsControlFile mdbx_NtFsControlFile;
|
||
|
||
#if !defined(_WIN32_WINNT_WIN8) || _WIN32_WINNT < _WIN32_WINNT_WIN8
|
||
typedef struct _WIN32_MEMORY_RANGE_ENTRY {
|
||
PVOID VirtualAddress;
|
||
SIZE_T NumberOfBytes;
|
||
} WIN32_MEMORY_RANGE_ENTRY, *PWIN32_MEMORY_RANGE_ENTRY;
|
||
#endif /* Windows 8.x */
|
||
|
||
typedef BOOL(WINAPI *MDBX_PrefetchVirtualMemory)(
|
||
HANDLE hProcess, ULONG_PTR NumberOfEntries,
|
||
PWIN32_MEMORY_RANGE_ENTRY VirtualAddresses, ULONG Flags);
|
||
MDBX_INTERNAL_VAR MDBX_PrefetchVirtualMemory mdbx_PrefetchVirtualMemory;
|
||
|
||
typedef DWORD(WINAPI *MDBX_DiscardVirtualMemory)(PVOID VirtualAddress,
|
||
SIZE_T Size);
|
||
MDBX_INTERNAL_VAR MDBX_DiscardVirtualMemory mdbx_DiscardVirtualMemory;
|
||
|
||
#endif /* Windows */
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
/* Atomics */
|
||
|
||
#if !defined(__cplusplus) && (__STDC_VERSION__ >= 201112L) && \
|
||
!defined(__STDC_NO_ATOMICS__) && \
|
||
(__GNUC_PREREQ(4, 9) || __CLANG_PREREQ(3, 8) || \
|
||
!(defined(__GNUC__) || defined(__clang__)))
|
||
#include <stdatomic.h>
|
||
#elif defined(__GNUC__) || defined(__clang__)
|
||
/* LY: nothing required */
|
||
#elif defined(_MSC_VER)
|
||
#pragma warning(disable : 4163) /* 'xyz': not available as an intrinsic */
|
||
#pragma warning(disable : 4133) /* 'function': incompatible types - from \
|
||
'size_t' to 'LONGLONG' */
|
||
#pragma warning(disable : 4244) /* 'return': conversion from 'LONGLONG' to \
|
||
'std::size_t', possible loss of data */
|
||
#pragma warning(disable : 4267) /* 'function': conversion from 'size_t' to \
|
||
'long', possible loss of data */
|
||
#pragma intrinsic(_InterlockedExchangeAdd, _InterlockedCompareExchange)
|
||
#pragma intrinsic(_InterlockedExchangeAdd64, _InterlockedCompareExchange64)
|
||
#elif defined(__APPLE__)
|
||
#include <libkern/OSAtomic.h>
|
||
#else
|
||
#error FIXME atomic-ops
|
||
#endif
|
||
|
||
static __inline uint32_t mdbx_atomic_add32(volatile uint32_t *p, uint32_t v) {
|
||
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
||
assert(atomic_is_lock_free(p));
|
||
return atomic_fetch_add((_Atomic uint32_t *)p, v);
|
||
#elif defined(__GNUC__) || defined(__clang__)
|
||
return __sync_fetch_and_add(p, v);
|
||
#else
|
||
#ifdef _MSC_VER
|
||
STATIC_ASSERT(sizeof(volatile long) == sizeof(volatile uint32_t));
|
||
return _InterlockedExchangeAdd((volatile long *)p, v);
|
||
#endif
|
||
#ifdef __APPLE__
|
||
return OSAtomicAdd32(v, (volatile int32_t *)p);
|
||
#endif
|
||
#endif
|
||
}
|
||
|
||
static __inline uint64_t mdbx_atomic_add64(volatile uint64_t *p, uint64_t v) {
|
||
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
||
assert(atomic_is_lock_free(p));
|
||
return atomic_fetch_add((_Atomic uint64_t *)p, v);
|
||
#elif defined(__GNUC__) || defined(__clang__)
|
||
return __sync_fetch_and_add(p, v);
|
||
#else
|
||
#ifdef _MSC_VER
|
||
#ifdef _WIN64
|
||
return _InterlockedExchangeAdd64((volatile int64_t *)p, v);
|
||
#else
|
||
return InterlockedExchangeAdd64((volatile int64_t *)p, v);
|
||
#endif
|
||
#endif /* _MSC_VER */
|
||
#ifdef __APPLE__
|
||
return OSAtomicAdd64(v, (volatile int64_t *)p);
|
||
#endif
|
||
#endif
|
||
}
|
||
|
||
#define mdbx_atomic_sub32(p, v) mdbx_atomic_add32(p, 0 - (v))
|
||
#define mdbx_atomic_sub64(p, v) mdbx_atomic_add64(p, 0 - (v))
|
||
|
||
static __inline bool mdbx_atomic_compare_and_swap32(volatile uint32_t *p,
|
||
uint32_t c, uint32_t v) {
|
||
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
||
assert(atomic_is_lock_free(p));
|
||
return atomic_compare_exchange_strong((_Atomic uint32_t *)p, &c, v);
|
||
#elif defined(__GNUC__) || defined(__clang__)
|
||
return __sync_bool_compare_and_swap(p, c, v);
|
||
#else
|
||
#ifdef _MSC_VER
|
||
STATIC_ASSERT(sizeof(volatile long) == sizeof(volatile uint32_t));
|
||
return c == _InterlockedCompareExchange((volatile long *)p, v, c);
|
||
#endif
|
||
#ifdef __APPLE__
|
||
return c == OSAtomicCompareAndSwap32Barrier(c, v, (volatile int32_t *)p);
|
||
#endif
|
||
#endif
|
||
}
|
||
|
||
static __inline bool mdbx_atomic_compare_and_swap64(volatile uint64_t *p,
|
||
uint64_t c, uint64_t v) {
|
||
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
||
assert(atomic_is_lock_free(p));
|
||
return atomic_compare_exchange_strong((_Atomic uint64_t *)p, &c, v);
|
||
#elif defined(__GNUC__) || defined(__clang__)
|
||
return __sync_bool_compare_and_swap(p, c, v);
|
||
#else
|
||
#ifdef _MSC_VER
|
||
return c == _InterlockedCompareExchange64((volatile int64_t *)p, v, c);
|
||
#endif
|
||
#ifdef __APPLE__
|
||
return c == OSAtomicCompareAndSwap64Barrier(c, v, (volatile uint64_t *)p);
|
||
#endif
|
||
#endif
|
||
}
|
||
|
||
/*----------------------------------------------------------------------------*/
|
||
|
||
#if defined(_MSC_VER) && _MSC_VER >= 1900
|
||
/* LY: MSVC 2015/2017/2019 has buggy/inconsistent PRIuPTR/PRIxPTR macros
|
||
* for internal format-args checker. */
|
||
#undef PRIuPTR
|
||
#undef PRIiPTR
|
||
#undef PRIdPTR
|
||
#undef PRIxPTR
|
||
#define PRIuPTR "Iu"
|
||
#define PRIiPTR "Ii"
|
||
#define PRIdPTR "Id"
|
||
#define PRIxPTR "Ix"
|
||
#define PRIuSIZE "zu"
|
||
#define PRIiSIZE "zi"
|
||
#define PRIdSIZE "zd"
|
||
#define PRIxSIZE "zx"
|
||
#endif /* fix PRI*PTR for _MSC_VER */
|
||
|
||
#ifndef PRIuSIZE
|
||
#define PRIuSIZE PRIuPTR
|
||
#define PRIiSIZE PRIiPTR
|
||
#define PRIdSIZE PRIdPTR
|
||
#define PRIxSIZE PRIxPTR
|
||
#endif /* PRI*SIZE macros for MSVC */
|
||
|
||
#ifdef _MSC_VER
|
||
#pragma warning(pop)
|
||
#endif
|