/* https://en.wikipedia.org/wiki/Operating_system_abstraction_layer */ /* * Copyright 2015-2017 Leonid Yuriev * 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 * . */ #pragma once #ifdef _MSC_VER #pragma warning(push, 1) #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) */ #include #include #include #include #include #include #include #include #include #include #include #ifndef _POSIX_C_SOURCE #ifdef _POSIX_SOURCE #define _POSIX_C_SOURCE 1 #else #define _POSIX_C_SOURCE 0 #endif #endif #ifndef _XOPEN_SOURCE #define _XOPEN_SOURCE 0 #endif #if defined(_WIN32) || defined(_WIN64) #include #include #define HAVE_SYS_STAT_H #define HAVE_SYS_TYPES_H typedef HANDLE mdbx_mutex_t; typedef HANDLE mdbx_cond_t; typedef HANDLE mdbx_thread_t; typedef unsigned mdbx_thread_key_t; typedef SSIZE_T ssize_t; #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 #else #include #include #include #include #include #include typedef pthread_mutex_t mdbx_mutex_t; typedef pthread_cond_t mdbx_cond_t; 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 * #endif /* Platform */ #ifndef SSIZE_MAX #define SSIZE_MAX INTPTR_MAX #endif #ifdef HAVE_SYS_STAT_H #include #endif #ifdef HAVE_SYS_TYPES_H #include #endif #ifdef HAVE_SYS_FILE_H #include #endif /*----------------------------------------------------------------------------*/ #ifdef _MSC_VER #if _MSC_FULL_VER < 190024215 #if _MSC_FULL_VER < 180040629 && defined(_M_IX86) #error Please use Visual Studio 2015 (MSC 19.0) or newer for 32-bit target. #else #pragma message( \ "It is recommended to use Visual Studio 2015 (MSC 19.0) or newer.") #endif #endif #include #elif __GNUC_PREREQ(4, 4) || defined(__clang__) #if defined(__i386__) || defined(__x86_64__) #include #include #endif #elif defined(__INTEL_COMPILER) #include #elif defined(__SUNPRO_C) || defined(__sun) || defined(sun) #include #elif (defined(_HPUX_SOURCE) || defined(__hpux) || defined(__HP_aCC)) && \ (defined(HP_IA64) || defined(__ia64)) #include #elif defined(__IBMC__) && defined(__powerpc) #include #elif defined(_AIX) #include #include #elif (defined(__osf__) && defined(__DECC)) || defined(__alpha) #include #include #elif defined(__MWERKS__) /* CodeWarrior - troubles ? */ #pragma gcc_extensions #elif defined(__SNC__) /* Sony PS3 - troubles ? */ #else #error Unknown C compiler, please use GNU C 5.x or newer #endif /* Compiler */ /*----------------------------------------------------------------------------*/ /* Byteorder */ #if !defined(__BYTE_ORDER__) || !defined(__ORDER_LITTLE_ENDIAN__) || \ !defined(__ORDER_BIG_ENDIAN__) #if defined(HAVE_ENDIAN_H) #include #elif defined(HAVE_SYS_PARAM_H) #include /* for endianness */ #elif defined(HAVE_NETINET_IN_H) && defined(HAVE_RESOLV_H) #include #include /* defines BYTE_ORDER on HPUX and Solaris */ #endif #ifdef _MSC_VER #pragma warning(pop) #endif #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 #else #define __ORDER_LITTLE_ENDIAN__ 1234 #define __ORDER_BIG_ENDIAN__ 4321 #if defined(__LITTLE_ENDIAN__) || defined(_LITTLE_ENDIAN) || \ defined(__ARMEL__) || defined(__THUMBEL__) || defined(__AARCH64EL__) || \ defined(__MIPSEL__) || defined(_MIPSEL) || defined(__MIPSEL) || \ defined(__i386) || defined(__x86_64__) || defined(_M_IX86) || \ defined(_M_X64) || defined(i386) || defined(_X86_) || defined(__i386__) || \ defined(_X86_64_) || defined(_M_ARM) || defined(_M_ARM64) || \ defined(__e2k__) #define __BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__ #elif defined(__BIG_ENDIAN__) || defined(_BIG_ENDIAN) || defined(__ARMEB__) || \ defined(__THUMBEB__) || defined(__AARCH64EB__) || defined(__MIPSEB__) || \ defined(_MIPSEB) || defined(__MIPSEB) || defined(_M_IA64) #define __BYTE_ORDER__ __ORDER_BIG_ENDIAN__ #else #error __BYTE_ORDER__ should be defined. #endif #endif #endif #if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__ && \ __BYTE_ORDER__ != __ORDER_BIG_ENDIAN__ #error Unsupported byte order. #endif /*----------------------------------------------------------------------------*/ /* Cache coherence */ #if defined(__i386__) || defined(__x86_64__) || defined(_M_AMD64) || \ defined(_M_IX86) || defined(__i386) || defined(__amd64) || \ defined(i386) || defined(__x86_64) || defined(_AMD64_) || defined(_M_X64) #define MDBX_CACHE_IS_COHERENT 1 #elif defined(__hppa) || defined(__hppa__) #define MDBX_CACHE_IS_COHERENT 1 #endif #ifndef MDBX_CACHE_IS_COHERENT #define MDBX_CACHE_IS_COHERENT 0 #endif #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 */ #ifndef __cache_aligned #define __cache_aligned __aligned(MDBX_CACHELINE_SIZE) #endif /*----------------------------------------------------------------------------*/ /* Memory/Compiler barriers */ 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 } #if MDBX_CACHE_IS_COHERENT #define mdbx_coherent_barrier() mdbx_compiler_barrier() #else #define mdbx_coherent_barrier() mdbx_memory_barrier() #endif #if defined(__mips) && defined(__linux) /* Only MIPS has explicit cache control */ #include #endif static __inline void mdbx_invalidate_cache(void *addr, size_t nbytes) { mdbx_coherent_barrier(); #if defined(__mips) && defined(__linux) /* MIPS has cache coherency issues. * Note: for any nbytes >= on-chip cache size, entire is flushed. */ cacheflush(addr, nbytes, DCACHE); #elif defined(_M_MRX000) || defined(_MIPS_) #error "Sorry, cacheflush() for MIPS not implemented" #else /* LY: assume no relevant mmap/dcache issues. */ (void)addr; (void)nbytes; #endif } /*----------------------------------------------------------------------------*/ /* max bytes to write in one call */ #define MAX_WRITE UINT32_C(0x3fff0000) /* 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 } static __inline char *mdbx_strdup(const char *str) { #ifdef _MSC_VER return _strdup(str); #else return strdup(str); #endif } int mdbx_memalign_alloc(size_t alignment, size_t bytes, void **result); void mdbx_memalign_free(void *ptr); int mdbx_mutex_init(mdbx_mutex_t *mutex); int mdbx_mutex_destroy(mdbx_mutex_t *mutex); int mdbx_mutex_lock(mdbx_mutex_t *mutex); int mdbx_mutex_unlock(mdbx_mutex_t *mutex); int mdbx_cond_init(mdbx_cond_t *cond); int mdbx_cond_destroy(mdbx_cond_t *cond); int mdbx_cond_signal(mdbx_cond_t *cond); int mdbx_cond_wait(mdbx_cond_t *cond, mdbx_mutex_t *mutex); int mdbx_pwritev(mdbx_filehandle_t fd, struct iovec *iov, int iovcnt, off_t offset, size_t expected_written); int mdbx_pread(mdbx_filehandle_t fd, void *buf, size_t count, off_t offset); int mdbx_pwrite(mdbx_filehandle_t fd, const void *buf, size_t count, off_t offset); int mdbx_write(mdbx_filehandle_t fd, const void *buf, size_t count); int mdbx_msync(void *addr, size_t length, int async); int mdbx_thread_create(mdbx_thread_t *thread, THREAD_RESULT(THREAD_CALL *start_routine)(void *), void *arg); int mdbx_thread_join(mdbx_thread_t thread); mdbx_tid_t mdbx_thread_self(void); int mdbx_thread_key_create(mdbx_thread_key_t *key); void mdbx_thread_key_delete(mdbx_thread_key_t key); void *mdbx_thread_rthc_get(mdbx_thread_key_t key); void mdbx_thread_rthc_set(mdbx_thread_key_t key, const void *value); int mdbx_filesync(mdbx_filehandle_t fd, bool fullsync); int mdbx_ftruncate(mdbx_filehandle_t fd, off_t length); int mdbx_filesize(mdbx_filehandle_t fd, off_t *length); int mdbx_openfile(const char *pathname, int flags, mode_t mode, mdbx_filehandle_t *fd); int mdbx_closefile(mdbx_filehandle_t fd); int mdbx_mremap_size(void **address, size_t old_size, size_t new_size); int mdbx_mmap(void **address, size_t length, int rw, mdbx_filehandle_t fd); int mdbx_munmap(void *address, size_t length); int mdbx_mlock(const void *address, size_t length); static __inline mdbx_pid_t mdbx_getpid(void) { #if defined(_WIN32) || defined(_WIN64) return GetCurrentProcessId(); #else return getpid(); #endif } void mdbx_osal_jitter(bool tiny); /*----------------------------------------------------------------------------*/ #ifndef mdbx_assert_fail void mdbx_assert_fail(MDB_env *env, const char *msg, const char *func, int line); #endif /* mdbx_assert_fail */ #if __GLIBC_PREREQ(2, 1) #define mdbx_asprintf asprintf #else int mdbx_asprintf(char **strp, const char *fmt, ...); #endif /*----------------------------------------------------------------------------*/ #if defined(_WIN32) || defined(_WIN64) #undef MDBX_OSAL_LOCK #define MDBX_OSAL_LOCK_SIGN MDBX_TETRAD('f', 'l', 'c', 'k') #else #define MDBX_OSAL_LOCK pthread_mutex_t #define MDBX_OSAL_LOCK_SIGN MDBX_TETRAD('P', 'T', 'M', 'X') #endif int mdbx_lck_init(MDB_env *env); int mdbx_lck_seize(MDB_env *env); int mdbx_lck_downgrade(MDB_env *env); void mdbx_lck_destroy(MDB_env *env); int mdbx_rdt_lock(MDB_env *env); void mdbx_rdt_unlock(MDB_env *env); int mdbx_txn_lock(MDB_env *env); void mdbx_txn_unlock(MDB_env *env); int mdbx_rpid_set(MDB_env *env); int mdbx_rpid_clear(MDB_env *env); /* Checks reader by pid. * * Returns: * MDBX_RESULT_TRUE, if pid is live (unable to acquire lock) * MDBX_RESULT_FALSE, if pid is dead (lock acquired) * or otherwise the errcode. */ int mdbx_rpid_check(MDB_env *env, mdbx_pid_t pid); /*----------------------------------------------------------------------------*/ #ifdef _MSC_VER #ifndef snprintf #define snprintf(buffer, buffer_size, format, ...) \ _snprintf_s(buffer, buffer_size, _TRUNCATE, format, __VA_ARGS__) #endif /* snprintf */ #ifndef vsnprintf #define vsnprintf(buffer, buffer_size, format, args) \ _vsnprintf_s(buffer, buffer_size, _TRUNCATE, format, args) #endif /* vsnprintf */ #endif /* _MSC_VER */ /*----------------------------------------------------------------------------*/ #if (__STDC_VERSION__ >= 201112L) && !defined(__STDC_NO_ATOMICS__) && \ (__GNUC_PREREQ(4, 9) || __CLANG_PREREQ(3, 8) || \ !(defined(__GNUC__) || defined(__clang__))) #include #elif defined(__GNUC__) || defined(__clang__) /* LY: nothing required */ #elif defined(_MSC_VER) #pragma warning(push) #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 intrinsic(_InterlockedExchangeAdd, _InterlockedCompareExchange) #pragma intrinsic(_InterlockedExchangeAdd64, _InterlockedCompareExchange64) #elif defined(__APPLE__) #include #else #error FIXME atomic-ops #endif static __inline size_t mdbx_atomic_add(volatile size_t *p, size_t v) { #ifdef ATOMIC_VAR_INIT return atomic_fetch_add(p, v); #elif defined(__GNUC__) || defined(__clang__) return __sync_fetch_and_add(p, v); #else switch (sizeof(size_t)) { case 4: #ifdef _MSC_VER return _InterlockedExchangeAdd(p, v); #endif #ifdef __APPLE__ return OSAtomicAdd32(v, (volatile int32_t *)p); #endif case 8: #ifdef _MSC_VER return _InterlockedExchangeAdd64(p, v); #endif #ifdef __APPLE__ return OSAtomicAdd64(v, (volatile int64_t *)p); #endif } while (1) ; #endif } #define mdbx_atomic_sub(p, v) mdbx_atomic_add(p, -(v)) static __inline bool mdbx_atomic_compare_and_swap(volatile size_t *p, size_t c, size_t v) { #ifdef ATOMIC_VAR_INIT return atomic_compare_exchange_strong(p, &c, v); #elif defined(__GNUC__) || defined(__clang__) return __sync_bool_compare_and_swap(p, c, v); #else switch (sizeof(size_t)) { case 4: #ifdef _MSC_VER return c == _InterlockedCompareExchange(p, v, c); #endif #ifdef __APPLE__ return c == OSAtomicCompareAndSwap32Barrier(c, v, (volatile int32_t *)p); #endif case 8: #ifdef _MSC_VER return c == _InterlockedCompareExchange64(p, v, c); #endif #ifdef __APPLE__ return c == OSAtomicCompareAndSwap64Barrier(c, v, (volatile int32_t *)p); #endif } while (1) ; #endif } #ifdef _MSC_VER #pragma warning(pop) #endif