2017-07-26 11:48:34 +03:00
|
|
|
|
/* https://en.wikipedia.org/wiki/Operating_system_abstraction_layer */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Copyright 2015-2017 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
|
|
|
|
|
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
/* Microsoft compiler generates a lot of warning for self includes... */
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
#pragma warning(push, 1)
|
2017-07-03 12:50:48 +03:00
|
|
|
|
#pragma warning(disable : 4548) /* expression before comma has no effect; \
|
|
|
|
|
expected expression with side - effect */
|
2017-05-24 13:59:50 +03:00
|
|
|
|
#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 */
|
|
|
|
|
#if !defined(_CRT_SECURE_NO_WARNINGS)
|
|
|
|
|
#define _CRT_SECURE_NO_WARNINGS
|
|
|
|
|
#endif
|
|
|
|
|
#endif /* _MSC_VER (warnings) */
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
/* C99 includes */
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#include <inttypes.h>
|
|
|
|
|
#include <stddef.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <malloc.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
/* Systems includes */
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
#include <windows.h>
|
|
|
|
|
#include <winnt.h>
|
|
|
|
|
#define HAVE_SYS_STAT_H
|
|
|
|
|
#define HAVE_SYS_TYPES_H
|
|
|
|
|
typedef HANDLE mdbx_thread_t;
|
|
|
|
|
typedef unsigned mdbx_thread_key_t;
|
2017-06-21 01:34:56 +03:00
|
|
|
|
#define MDBX_OSAL_SECTION HANDLE
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#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
|
2017-05-23 18:40:21 +03:00
|
|
|
|
typedef struct {
|
|
|
|
|
HANDLE mutex;
|
|
|
|
|
HANDLE event;
|
|
|
|
|
} mdbx_condmutex_t;
|
2017-05-23 21:54:06 +03:00
|
|
|
|
typedef CRITICAL_SECTION mdbx_fastmutex_t;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
#include <pthread.h>
|
2017-07-26 15:56:39 +03:00
|
|
|
|
#include <signal.h>
|
2017-05-18 14:00:40 +03:00
|
|
|
|
#include <sys/file.h>
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#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 *
|
2017-05-23 18:40:21 +03:00
|
|
|
|
typedef struct {
|
|
|
|
|
pthread_mutex_t mutex;
|
|
|
|
|
pthread_cond_t cond;
|
|
|
|
|
} mdbx_condmutex_t;
|
2017-05-23 21:54:06 +03:00
|
|
|
|
typedef pthread_mutex_t mdbx_fastmutex_t;
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif /* Platform */
|
|
|
|
|
|
|
|
|
|
#ifndef SSIZE_MAX
|
|
|
|
|
#define SSIZE_MAX INTPTR_MAX
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#endif
|
|
|
|
|
#ifdef HAVE_SYS_FILE_H
|
|
|
|
|
#include <sys/file.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-05-25 14:56:07 +03:00
|
|
|
|
#if !defined(UNALIGNED_OK)
|
|
|
|
|
#if defined(__i386) || defined(__x86_64__) || defined(_M_IX86) || \
|
|
|
|
|
defined(_M_X64) || defined(i386) || defined(_X86_) || defined(__i386__) || \
|
|
|
|
|
defined(_X86_64_)
|
|
|
|
|
#define UNALIGNED_OK 1
|
|
|
|
|
#else
|
2017-05-24 13:59:50 +03:00
|
|
|
|
#define UNALIGNED_OK 0
|
2017-05-25 14:56:07 +03:00
|
|
|
|
#endif
|
2017-05-24 13:59:50 +03:00
|
|
|
|
#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
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/* Compiler's includes for builtins/intrinsics */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
#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 <intrin.h>
|
|
|
|
|
|
|
|
|
|
#elif __GNUC_PREREQ(4, 4) || defined(__clang__)
|
|
|
|
|
#if defined(__i386__) || defined(__x86_64__)
|
|
|
|
|
#include <cpuid.h>
|
|
|
|
|
#include <x86intrin.h>
|
|
|
|
|
#endif
|
|
|
|
|
#elif defined(__INTEL_COMPILER)
|
|
|
|
|
#include <intrin.h>
|
|
|
|
|
#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 ? */
|
|
|
|
|
#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 <endian.h>
|
|
|
|
|
#elif defined(HAVE_SYS_PARAM_H)
|
|
|
|
|
#include <sys/param.h> /* for endianness */
|
|
|
|
|
#elif defined(HAVE_NETINET_IN_H) && defined(HAVE_RESOLV_H)
|
|
|
|
|
#include <netinet/in.h>
|
|
|
|
|
#include <resolv.h> /* defines BYTE_ORDER on HPUX and Solaris */
|
|
|
|
|
#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__) || \
|
2017-03-30 18:54:57 +03:00
|
|
|
|
defined(_X86_64_) || defined(_M_ARM) || defined(_M_ARM64) || \
|
|
|
|
|
defined(__e2k__)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#define __BYTE_ORDER__ __ORDER_LITTLE_ENDIAN__
|
|
|
|
|
#elif defined(__BIG_ENDIAN__) || defined(_BIG_ENDIAN) || defined(__ARMEB__) || \
|
|
|
|
|
defined(__THUMBEB__) || defined(__AARCH64EB__) || defined(__MIPSEB__) || \
|
2017-03-30 18:54:57 +03:00
|
|
|
|
defined(_MIPSEB) || defined(__MIPSEB) || defined(_M_IA64)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#define __BYTE_ORDER__ __ORDER_BIG_ENDIAN__
|
|
|
|
|
#else
|
|
|
|
|
#error __BYTE_ORDER__ should be defined.
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
2017-08-02 15:15:08 +03:00
|
|
|
|
#endif /* __BYTE_ORDER__ || __ORDER_LITTLE_ENDIAN__ || __ORDER_BIG_ENDIAN__ */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
#if __BYTE_ORDER__ != __ORDER_LITTLE_ENDIAN__ && \
|
|
|
|
|
__BYTE_ORDER__ != __ORDER_BIG_ENDIAN__
|
|
|
|
|
#error Unsupported byte order.
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/* Memory/Compiler barriers, cache coherence */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
/* Cache coherence and invalidation */
|
|
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#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 <asm/cachectl.h>
|
|
|
|
|
#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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/* libc compatibility stuff */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2017-05-15 12:08:04 +03:00
|
|
|
|
#ifndef mdbx_assert_fail
|
2017-05-24 21:43:29 +03:00
|
|
|
|
void mdbx_assert_fail(const MDBX_env *env, const char *msg, const char *func,
|
2017-05-15 12:08:04 +03:00
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
#ifdef _MSC_VER
|
2017-07-28 13:48:46 +03:00
|
|
|
|
|
2017-05-15 12:08:04 +03:00
|
|
|
|
#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 */
|
2017-07-28 13:48:46 +03:00
|
|
|
|
|
|
|
|
|
#ifdef _ASSERTE
|
|
|
|
|
#undef assert
|
|
|
|
|
#define assert _ASSERTE
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-05-15 12:08:04 +03:00
|
|
|
|
#endif /* _MSC_VER */
|
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/* OS abstraction layer stuff */
|
2017-05-15 12:08:04 +03:00
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
/* max bytes to write in one call */
|
2017-04-27 18:13:39 +03:00
|
|
|
|
#define MAX_WRITE UINT32_C(0x3fff0000)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
/* 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
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-15 12:08:04 +03:00
|
|
|
|
static __inline int mdbx_get_errno(void) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
DWORD rc = GetLastError();
|
|
|
|
|
#else
|
|
|
|
|
int rc = errno;
|
|
|
|
|
#endif
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
int mdbx_memalign_alloc(size_t alignment, size_t bytes, void **result);
|
|
|
|
|
void mdbx_memalign_free(void *ptr);
|
|
|
|
|
|
2017-05-23 18:40:21 +03:00
|
|
|
|
int mdbx_condmutex_init(mdbx_condmutex_t *condmutex);
|
|
|
|
|
int mdbx_condmutex_lock(mdbx_condmutex_t *condmutex);
|
|
|
|
|
int mdbx_condmutex_unlock(mdbx_condmutex_t *condmutex);
|
|
|
|
|
int mdbx_condmutex_signal(mdbx_condmutex_t *condmutex);
|
|
|
|
|
int mdbx_condmutex_wait(mdbx_condmutex_t *condmutex);
|
|
|
|
|
int mdbx_condmutex_destroy(mdbx_condmutex_t *condmutex);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2017-05-23 21:54:06 +03:00
|
|
|
|
int mdbx_fastmutex_init(mdbx_fastmutex_t *fastmutex);
|
|
|
|
|
int mdbx_fastmutex_acquire(mdbx_fastmutex_t *fastmutex);
|
|
|
|
|
int mdbx_fastmutex_release(mdbx_fastmutex_t *fastmutex);
|
|
|
|
|
int mdbx_fastmutex_destroy(mdbx_fastmutex_t *fastmutex);
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
int mdbx_pwritev(mdbx_filehandle_t fd, struct iovec *iov, int iovcnt,
|
2017-05-24 18:50:24 +03:00
|
|
|
|
uint64_t offset, size_t expected_written);
|
|
|
|
|
int mdbx_pread(mdbx_filehandle_t fd, void *buf, size_t count, uint64_t offset);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
int mdbx_pwrite(mdbx_filehandle_t fd, const void *buf, size_t count,
|
2017-05-24 18:50:24 +03:00
|
|
|
|
uint64_t offset);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
int mdbx_write(mdbx_filehandle_t fd, const void *buf, size_t count);
|
|
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
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);
|
|
|
|
|
|
2017-04-24 19:37:01 +03:00
|
|
|
|
int mdbx_filesync(mdbx_filehandle_t fd, bool fullsync);
|
2017-06-21 01:34:56 +03:00
|
|
|
|
int mdbx_filesize_sync(mdbx_filehandle_t fd);
|
2017-05-24 18:50:24 +03:00
|
|
|
|
int mdbx_ftruncate(mdbx_filehandle_t fd, uint64_t length);
|
|
|
|
|
int mdbx_filesize(mdbx_filehandle_t fd, uint64_t *length);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
int mdbx_openfile(const char *pathname, int flags, mode_t mode,
|
|
|
|
|
mdbx_filehandle_t *fd);
|
|
|
|
|
int mdbx_closefile(mdbx_filehandle_t fd);
|
|
|
|
|
|
2017-06-21 01:34:56 +03:00
|
|
|
|
typedef struct mdbx_mmap_param {
|
2017-07-11 14:10:24 +03:00
|
|
|
|
union {
|
|
|
|
|
void *address;
|
|
|
|
|
uint8_t *dxb;
|
|
|
|
|
struct MDBX_lockinfo *lck;
|
|
|
|
|
};
|
|
|
|
|
mdbx_filehandle_t fd;
|
2017-07-12 21:13:17 +03:00
|
|
|
|
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 */
|
|
|
|
|
#endif
|
2017-06-21 01:34:56 +03:00
|
|
|
|
#ifdef MDBX_OSAL_SECTION
|
|
|
|
|
MDBX_OSAL_SECTION section;
|
|
|
|
|
#endif
|
2017-07-11 14:10:24 +03:00
|
|
|
|
} mdbx_mmap_t;
|
|
|
|
|
|
2017-07-12 21:13:17 +03:00
|
|
|
|
int mdbx_mmap(int flags, mdbx_mmap_t *map, size_t must, size_t limit);
|
|
|
|
|
int mdbx_munmap(mdbx_mmap_t *map);
|
2017-07-11 14:10:24 +03:00
|
|
|
|
int mdbx_mresize(int flags, mdbx_mmap_t *map, size_t current, size_t wanna);
|
|
|
|
|
int mdbx_msync(mdbx_mmap_t *map, size_t offset, size_t length, int async);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
static __inline mdbx_pid_t mdbx_getpid(void) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
return GetCurrentProcessId();
|
|
|
|
|
#else
|
|
|
|
|
return getpid();
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-06 17:05:30 +03:00
|
|
|
|
static __inline mdbx_tid_t mdbx_thread_self(void) {
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
return GetCurrentThreadId();
|
|
|
|
|
#else
|
|
|
|
|
return pthread_self();
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2017-04-27 15:18:33 +03:00
|
|
|
|
void mdbx_osal_jitter(bool tiny);
|
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/* lck stuff */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
|
|
#if defined(_WIN32) || defined(_WIN64)
|
|
|
|
|
#undef MDBX_OSAL_LOCK
|
2017-05-30 16:22:42 +03:00
|
|
|
|
#define MDBX_OSAL_LOCK_SIGN UINT32_C(0xF10C)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#else
|
|
|
|
|
#define MDBX_OSAL_LOCK pthread_mutex_t
|
2017-05-30 16:22:42 +03:00
|
|
|
|
#define MDBX_OSAL_LOCK_SIGN UINT32_C(0x8017)
|
2017-03-16 18:09:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
|
2017-05-24 01:42:10 +03:00
|
|
|
|
int mdbx_lck_init(MDBX_env *env);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2017-05-24 01:42:10 +03:00
|
|
|
|
int mdbx_lck_seize(MDBX_env *env);
|
2017-07-19 09:12:46 +03:00
|
|
|
|
int mdbx_lck_downgrade(MDBX_env *env, bool complete);
|
2017-06-06 20:18:09 +03:00
|
|
|
|
int mdbx_lck_upgrade(MDBX_env *env);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
void mdbx_lck_destroy(MDBX_env *env);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2017-05-24 01:42:10 +03:00
|
|
|
|
int mdbx_rdt_lock(MDBX_env *env);
|
|
|
|
|
void mdbx_rdt_unlock(MDBX_env *env);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2017-10-26 21:47:41 +03:00
|
|
|
|
int mdbx_txn_lock(MDBX_env *env, bool dontwait);
|
2017-05-24 01:42:10 +03:00
|
|
|
|
void mdbx_txn_unlock(MDBX_env *env);
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
2017-05-24 01:42:10 +03:00
|
|
|
|
int mdbx_rpid_set(MDBX_env *env);
|
|
|
|
|
int mdbx_rpid_clear(MDBX_env *env);
|
2017-04-21 18:26:32 +03:00
|
|
|
|
|
|
|
|
|
/* 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. */
|
2017-05-24 01:42:10 +03:00
|
|
|
|
int mdbx_rpid_check(MDBX_env *env, mdbx_pid_t pid);
|
2017-04-04 15:17:24 +03:00
|
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/* Atomics */
|
2017-04-04 15:17:24 +03:00
|
|
|
|
|
2017-07-26 17:34:59 +03:00
|
|
|
|
#if !defined(__cplusplus) && (__STDC_VERSION__ >= 201112L) && \
|
|
|
|
|
!defined(__STDC_NO_ATOMICS__) && \
|
2017-04-27 18:33:05 +03:00
|
|
|
|
(__GNUC_PREREQ(4, 9) || __CLANG_PREREQ(3, 8) || \
|
|
|
|
|
!(defined(__GNUC__) || defined(__clang__)))
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#include <stdatomic.h>
|
|
|
|
|
#elif defined(__GNUC__) || defined(__clang__)
|
|
|
|
|
/* LY: nothing required */
|
|
|
|
|
#elif defined(_MSC_VER)
|
2017-05-10 19:26:56 +03:00
|
|
|
|
#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 */
|
2017-05-15 12:08:04 +03:00
|
|
|
|
#pragma warning(disable : 4267) /* 'function': conversion from 'size_t' to \
|
|
|
|
|
'long', possible loss of data */
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#pragma intrinsic(_InterlockedExchangeAdd, _InterlockedCompareExchange)
|
|
|
|
|
#pragma intrinsic(_InterlockedExchangeAdd64, _InterlockedCompareExchange64)
|
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
|
#include <libkern/OSAtomic.h>
|
|
|
|
|
#else
|
|
|
|
|
#error FIXME atomic-ops
|
|
|
|
|
#endif
|
|
|
|
|
|
2017-05-26 20:27:27 +03:00
|
|
|
|
static __inline uint32_t mdbx_atomic_add32(volatile uint32_t *p, uint32_t v) {
|
2017-07-26 17:34:59 +03:00
|
|
|
|
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
2017-06-24 19:33:34 +03:00
|
|
|
|
assert(atomic_is_lock_free(p));
|
|
|
|
|
return atomic_fetch_add((_Atomic uint32_t *)p, v);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#elif defined(__GNUC__) || defined(__clang__)
|
|
|
|
|
return __sync_fetch_and_add(p, v);
|
|
|
|
|
#else
|
|
|
|
|
#ifdef _MSC_VER
|
2017-05-26 20:27:27 +03:00
|
|
|
|
return _InterlockedExchangeAdd(p, v);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
|
|
|
|
#ifdef __APPLE__
|
2017-05-26 20:27:27 +03:00
|
|
|
|
return OSAtomicAdd32(v, (volatile int32_t *)p);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
2017-05-26 20:27:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static __inline uint64_t mdbx_atomic_add64(volatile uint64_t *p, uint64_t v) {
|
2017-07-26 17:34:59 +03:00
|
|
|
|
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
2017-06-24 19:33:34 +03:00
|
|
|
|
assert(atomic_is_lock_free(p));
|
|
|
|
|
return atomic_fetch_add((_Atomic uint64_t *)p, v);
|
2017-05-26 20:27:27 +03:00
|
|
|
|
#elif defined(__GNUC__) || defined(__clang__)
|
|
|
|
|
return __sync_fetch_and_add(p, v);
|
|
|
|
|
#else
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#ifdef _MSC_VER
|
2017-07-26 15:56:39 +03:00
|
|
|
|
return _InterlockedExchangeAdd64((volatile int64_t *)p, v);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
|
|
|
|
#ifdef __APPLE__
|
2017-05-26 20:27:27 +03:00
|
|
|
|
return OSAtomicAdd64(v, (volatile int64_t *)p);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-26 17:34:59 +03:00
|
|
|
|
#define mdbx_atomic_sub32(p, v) mdbx_atomic_add32(p, 0 - (v))
|
|
|
|
|
#define mdbx_atomic_sub64(p, v) mdbx_atomic_add64(p, 0 - (v))
|
2017-04-26 18:14:49 +03:00
|
|
|
|
|
2017-05-26 20:27:27 +03:00
|
|
|
|
static __inline bool mdbx_atomic_compare_and_swap32(volatile uint32_t *p,
|
|
|
|
|
uint32_t c, uint32_t v) {
|
2017-07-26 17:34:59 +03:00
|
|
|
|
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
2017-06-24 19:33:34 +03:00
|
|
|
|
assert(atomic_is_lock_free(p));
|
|
|
|
|
return atomic_compare_exchange_strong((_Atomic uint32_t *)p, &c, v);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#elif defined(__GNUC__) || defined(__clang__)
|
|
|
|
|
return __sync_bool_compare_and_swap(p, c, v);
|
|
|
|
|
#else
|
|
|
|
|
#ifdef _MSC_VER
|
2017-05-26 20:27:27 +03:00
|
|
|
|
return c == _InterlockedCompareExchange(p, v, c);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
|
|
|
|
#ifdef __APPLE__
|
2017-05-26 20:27:27 +03:00
|
|
|
|
return c == OSAtomicCompareAndSwap32Barrier(c, v, (volatile int32_t *)p);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
2017-05-26 20:27:27 +03:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static __inline bool mdbx_atomic_compare_and_swap64(volatile uint64_t *p,
|
|
|
|
|
uint64_t c, uint64_t v) {
|
2017-07-26 17:34:59 +03:00
|
|
|
|
#if !defined(__cplusplus) && defined(ATOMIC_VAR_INIT)
|
2017-06-24 19:33:34 +03:00
|
|
|
|
assert(atomic_is_lock_free(p));
|
|
|
|
|
return atomic_compare_exchange_strong((_Atomic uint64_t *)p, &c, v);
|
2017-05-26 20:27:27 +03:00
|
|
|
|
#elif defined(__GNUC__) || defined(__clang__)
|
|
|
|
|
return __sync_bool_compare_and_swap(p, c, v);
|
|
|
|
|
#else
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#ifdef _MSC_VER
|
2017-07-26 15:56:39 +03:00
|
|
|
|
return c == _InterlockedCompareExchange64((volatile int64_t *)p, v, c);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
|
|
|
|
#ifdef __APPLE__
|
2017-05-26 20:27:27 +03:00
|
|
|
|
return c == OSAtomicCompareAndSwap64Barrier(c, v, (volatile uint64_t *)p);
|
2017-04-26 18:14:49 +03:00
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
}
|
2017-05-10 19:26:56 +03:00
|
|
|
|
|
2017-05-24 13:59:50 +03:00
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
2017-07-26 18:32:46 +03:00
|
|
|
|
#if defined(_MSC_VER) && _MSC_VER >= 1900 && _MSC_VER < 1920
|
|
|
|
|
/* LY: MSVC 2015/2017 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 */
|
|
|
|
|
|
2017-05-10 19:26:56 +03:00
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
|
#pragma warning(pop)
|
|
|
|
|
#endif
|