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
|
|
|
|
/* *INDENT-OFF* */
|
|
|
|
/* clang-format off */
|
|
|
|
|
|
|
|
#ifndef __GNUC_PREREQ
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) && defined(__GNUC_MINOR__)
|
|
|
|
# define __GNUC_PREREQ(maj, min) \
|
|
|
|
((__GNUC__ << 16) + __GNUC_MINOR__ >= ((maj) << 16) + (min))
|
|
|
|
# else
|
|
|
|
# define __GNUC_PREREQ(maj, min) (0)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __GNUC_PREREQ */
|
|
|
|
|
|
|
|
#ifndef __CLANG_PREREQ
|
2017-05-24 13:37:06 +03:00
|
|
|
# ifdef __clang__
|
|
|
|
# define __CLANG_PREREQ(maj,min) \
|
|
|
|
((__clang_major__ << 16) + __clang_minor__ >= ((maj) << 16) + (min))
|
|
|
|
# else
|
|
|
|
# define __CLANG_PREREQ(maj,min) (0)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __CLANG_PREREQ */
|
|
|
|
|
|
|
|
#ifndef __GLIBC_PREREQ
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GLIBC__) && defined(__GLIBC_MINOR__)
|
|
|
|
# define __GLIBC_PREREQ(maj, min) \
|
|
|
|
((__GLIBC__ << 16) + __GLIBC_MINOR__ >= ((maj) << 16) + (min))
|
|
|
|
# else
|
|
|
|
# define __GLIBC_PREREQ(maj, min) (0)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __GLIBC_PREREQ */
|
|
|
|
|
|
|
|
#ifndef __has_attribute
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __has_attribute(x) (0)
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __has_feature
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __has_feature(x) (0)
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __has_extension
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __has_extension(x) (0)
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef __has_builtin
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __has_builtin(x) (0)
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if __has_feature(thread_sanitizer)
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __SANITIZE_THREAD__ 1
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if __has_feature(address_sanitizer)
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __SANITIZE_ADDRESS__ 1
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#ifndef __extern_C
|
2017-05-24 13:37:06 +03:00
|
|
|
# ifdef __cplusplus
|
|
|
|
# define __extern_C extern "C"
|
|
|
|
# else
|
|
|
|
# define __extern_C
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __extern_C */
|
|
|
|
|
|
|
|
#ifndef __cplusplus
|
2017-05-24 13:37:06 +03:00
|
|
|
# ifndef bool
|
|
|
|
# define bool _Bool
|
|
|
|
# endif
|
|
|
|
# ifndef true
|
|
|
|
# define true (1)
|
|
|
|
# endif
|
|
|
|
# ifndef false
|
|
|
|
# define false (0)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if !defined(nullptr) && !defined(__cplusplus) || (__cplusplus < 201103L && !defined(_MSC_VER))
|
2017-05-24 13:37:06 +03:00
|
|
|
# define nullptr NULL
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#if !defined(__thread) && (defined(_MSC_VER) || defined(__DMC__))
|
2017-05-24 13:37:06 +03:00
|
|
|
# define __thread __declspec(thread)
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __thread */
|
|
|
|
|
|
|
|
#ifndef __alwaysinline
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(always_inline)
|
|
|
|
# define __alwaysinline __inline __attribute__((always_inline))
|
|
|
|
# elif defined(_MSC_VER)
|
|
|
|
# define __alwaysinline __forceinline
|
|
|
|
# else
|
|
|
|
# define __alwaysinline
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __alwaysinline */
|
|
|
|
|
|
|
|
#ifndef __noinline
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(noinline)
|
|
|
|
# define __noinline __attribute__((noinline))
|
|
|
|
# elif defined(_MSC_VER)
|
|
|
|
# define __noinline __declspec(noinline)
|
|
|
|
# elif defined(__SUNPRO_C) || defined(__sun) || defined(sun)
|
|
|
|
# define __noinline inline
|
|
|
|
# elif !defined(__INTEL_COMPILER)
|
|
|
|
# define __noinline /* FIXME ? */
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __noinline */
|
|
|
|
|
|
|
|
#ifndef __must_check_result
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(warn_unused_result)
|
|
|
|
# define __must_check_result __attribute__((warn_unused_result))
|
|
|
|
# else
|
|
|
|
# define __must_check_result
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __must_check_result */
|
|
|
|
|
|
|
|
#ifndef __deprecated
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(deprecated)
|
|
|
|
# define __deprecated __attribute__((deprecated))
|
|
|
|
# elif defined(_MSC_VER)
|
|
|
|
# define __deprecated __declspec(deprecated)
|
|
|
|
# else
|
|
|
|
# define __deprecated
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __deprecated */
|
|
|
|
|
|
|
|
#ifndef __packed
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(packed)
|
|
|
|
# define __packed __attribute__((packed))
|
|
|
|
# else
|
|
|
|
# define __packed
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __packed */
|
|
|
|
|
|
|
|
#ifndef __aligned
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(aligned)
|
|
|
|
# define __aligned(N) __attribute__((aligned(N)))
|
|
|
|
# elif defined(_MSC_VER)
|
|
|
|
# define __aligned(N) __declspec(align(N))
|
|
|
|
# else
|
|
|
|
# define __aligned(N)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __aligned */
|
|
|
|
|
|
|
|
#ifndef __noreturn
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(noreturn)
|
|
|
|
# define __noreturn __attribute__((noreturn))
|
|
|
|
# elif defined(_MSC_VER)
|
|
|
|
# define __noreturn __declspec(noreturn)
|
|
|
|
# else
|
|
|
|
# define __noreturn
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __noreturn */
|
|
|
|
|
|
|
|
#ifndef __nothrow
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(nothrow)
|
|
|
|
# define __nothrow __attribute__((nothrow))
|
|
|
|
# elif defined(_MSC_VER) && defined(__cplusplus)
|
|
|
|
# define __nothrow __declspec(nothrow)
|
|
|
|
# else
|
|
|
|
# define __nothrow
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __nothrow */
|
|
|
|
|
|
|
|
#ifndef __pure_function
|
2017-05-24 13:37:06 +03:00
|
|
|
/* Many functions have no effects except the return value and their
|
|
|
|
* return value depends only on the parameters and/or global variables.
|
|
|
|
* Such a function can be subject to common subexpression elimination
|
|
|
|
* and loop optimization just as an arithmetic operator would be.
|
|
|
|
* These functions should be declared with the attribute pure. */
|
|
|
|
# if defined(__GNUC__) || __has_attribute(pure)
|
|
|
|
# define __pure_function __attribute__((pure))
|
|
|
|
# else
|
|
|
|
# define __pure_function
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __pure_function */
|
|
|
|
|
|
|
|
#ifndef __const_function
|
2017-05-24 13:37:06 +03:00
|
|
|
/* Many functions do not examine any values except their arguments,
|
|
|
|
* and have no effects except the return value. Basically this is just
|
|
|
|
* slightly more strict class than the PURE attribute, since function
|
|
|
|
* is not allowed to read global memory.
|
|
|
|
*
|
|
|
|
* Note that a function that has pointer arguments and examines the
|
|
|
|
* data pointed to must not be declared const. Likewise, a function
|
|
|
|
* that calls a non-const function usually must not be const.
|
|
|
|
* It does not make sense for a const function to return void. */
|
|
|
|
# if defined(__GNUC__) || __has_attribute(const)
|
|
|
|
# define __const_function __attribute__((const))
|
|
|
|
# else
|
|
|
|
# define __const_function
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __const_function */
|
|
|
|
|
|
|
|
#ifndef __dll_hidden
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || __has_attribute(visibility)
|
|
|
|
# define __hidden __attribute__((visibility("hidden")))
|
|
|
|
# else
|
|
|
|
# define __hidden
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __dll_hidden */
|
|
|
|
|
|
|
|
#ifndef __optimize
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__OPTIMIZE__)
|
|
|
|
# if defined(__clang__) && !__has_attribute(optimize)
|
|
|
|
# define __optimize(ops)
|
|
|
|
# elif defined(__GNUC__) || __has_attribute(optimize)
|
|
|
|
# define __optimize(ops) __attribute__((optimize(ops)))
|
|
|
|
# else
|
|
|
|
# define __optimize(ops)
|
|
|
|
# endif
|
|
|
|
# else
|
|
|
|
# define __optimize(ops)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __optimize */
|
|
|
|
|
|
|
|
#ifndef __hot
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__OPTIMIZE__)
|
|
|
|
# if defined(__clang__) && !__has_attribute(hot)
|
|
|
|
/* just put frequently used functions in separate section */
|
|
|
|
# define __hot __attribute__((section("text.hot"))) __optimize("O3")
|
|
|
|
# elif defined(__GNUC__) || __has_attribute(hot)
|
|
|
|
# define __hot __attribute__((hot)) __optimize("O3")
|
|
|
|
# else
|
|
|
|
# define __hot __optimize("O3")
|
|
|
|
# endif
|
|
|
|
# else
|
|
|
|
# define __hot
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __hot */
|
|
|
|
|
|
|
|
#ifndef __cold
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__OPTIMIZE__)
|
|
|
|
# if defined(__clang__) && !__has_attribute(cold)
|
|
|
|
/* just put infrequently used functions in separate section */
|
|
|
|
# define __cold __attribute__((section("text.unlikely"))) __optimize("Os")
|
|
|
|
# elif defined(__GNUC__) || __has_attribute(cold)
|
|
|
|
# define __cold __attribute__((cold)) __optimize("Os")
|
|
|
|
# else
|
|
|
|
# define __cold __optimize("Os")
|
|
|
|
# endif
|
|
|
|
# else
|
|
|
|
# define __cold
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __cold */
|
|
|
|
|
|
|
|
#ifndef __flatten
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__OPTIMIZE__) && (defined(__GNUC__) || __has_attribute(flatten))
|
|
|
|
# define __flatten __attribute__((flatten))
|
|
|
|
# else
|
|
|
|
# define __flatten
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* __flatten */
|
|
|
|
|
|
|
|
#ifndef likely
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || defined(__clang__)
|
|
|
|
# define likely(cond) __builtin_expect(!!(cond), 1)
|
|
|
|
# else
|
|
|
|
# define likely(x) (x)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* likely */
|
|
|
|
|
|
|
|
#ifndef unlikely
|
2017-05-24 13:37:06 +03:00
|
|
|
# if defined(__GNUC__) || defined(__clang__)
|
|
|
|
# define unlikely(cond) __builtin_expect(!!(cond), 0)
|
|
|
|
# else
|
|
|
|
# define unlikely(x) (x)
|
|
|
|
# endif
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif /* unlikely */
|
|
|
|
|
2017-03-30 18:54:57 +03:00
|
|
|
#if !defined(__noop) && !defined(_MSC_VER)
|
2017-05-24 13:37:06 +03:00
|
|
|
static __inline int __do_noop(void* crutch, ...) {
|
|
|
|
(void) crutch; return 0;
|
|
|
|
}
|
|
|
|
# define __noop(...) __do_noop(0, __VA_ARGS__)
|
2017-03-30 18:54:57 +03:00
|
|
|
#endif /* __noop */
|
|
|
|
|
2017-03-16 18:09:27 +03:00
|
|
|
/* Wrapper around __func__, which is a C99 feature */
|
|
|
|
#if defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L
|
2017-05-24 13:37:06 +03:00
|
|
|
# define mdbx_func_ __func__
|
2017-03-16 18:09:27 +03:00
|
|
|
#elif (defined(__GNUC__) && __GNUC__ >= 2) || defined(__clang__) || defined(_MSC_VER)
|
2017-05-24 13:37:06 +03:00
|
|
|
# define mdbx_func_ __FUNCTION__
|
2017-03-16 18:09:27 +03:00
|
|
|
#else
|
2017-05-24 13:37:06 +03:00
|
|
|
# define mdbx_func_ "<mdbx_unknown>"
|
2017-03-16 18:09:27 +03:00
|
|
|
#endif
|
|
|
|
|
2017-05-24 13:59:50 +03:00
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#if defined(USE_VALGRIND)
|
|
|
|
# include <valgrind/memcheck.h>
|
|
|
|
# ifndef VALGRIND_DISABLE_ADDR_ERROR_REPORTING_IN_RANGE
|
|
|
|
/* LY: available since Valgrind 3.10 */
|
|
|
|
# define VALGRIND_DISABLE_ADDR_ERROR_REPORTING_IN_RANGE(a,s)
|
|
|
|
# define VALGRIND_ENABLE_ADDR_ERROR_REPORTING_IN_RANGE(a,s)
|
|
|
|
# endif
|
|
|
|
#else
|
|
|
|
# define VALGRIND_CREATE_MEMPOOL(h,r,z)
|
|
|
|
# define VALGRIND_DESTROY_MEMPOOL(h)
|
|
|
|
# define VALGRIND_MEMPOOL_TRIM(h,a,s)
|
|
|
|
# define VALGRIND_MEMPOOL_ALLOC(h,a,s)
|
|
|
|
# define VALGRIND_MEMPOOL_FREE(h,a)
|
|
|
|
# define VALGRIND_MEMPOOL_CHANGE(h,a,b,s)
|
|
|
|
# define VALGRIND_MAKE_MEM_NOACCESS(a,s)
|
|
|
|
# define VALGRIND_MAKE_MEM_DEFINED(a,s)
|
|
|
|
# define VALGRIND_MAKE_MEM_UNDEFINED(a,s)
|
|
|
|
# define VALGRIND_DISABLE_ADDR_ERROR_REPORTING_IN_RANGE(a,s)
|
|
|
|
# define VALGRIND_ENABLE_ADDR_ERROR_REPORTING_IN_RANGE(a,s)
|
|
|
|
# define VALGRIND_CHECK_MEM_IS_ADDRESSABLE(a,s) (0)
|
|
|
|
# define VALGRIND_CHECK_MEM_IS_DEFINED(a,s) (0)
|
|
|
|
#endif /* USE_VALGRIND */
|
|
|
|
|
|
|
|
#ifdef __SANITIZE_ADDRESS__
|
|
|
|
# include <sanitizer/asan_interface.h>
|
|
|
|
#else
|
|
|
|
# define ASAN_POISON_MEMORY_REGION(addr, size) \
|
|
|
|
((void)(addr), (void)(size))
|
|
|
|
# define ASAN_UNPOISON_MEMORY_REGION(addr, size) \
|
|
|
|
((void)(addr), (void)(size))
|
|
|
|
#endif /* __SANITIZE_ADDRESS__ */
|
|
|
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
#ifndef ARRAY_LENGTH
|
|
|
|
# ifdef __cplusplus
|
|
|
|
template <typename T, size_t N>
|
|
|
|
char (&__ArraySizeHelper(T (&array)[N]))[N];
|
|
|
|
# define ARRAY_LENGTH(array) (sizeof(::__ArraySizeHelper(array)))
|
|
|
|
# else
|
|
|
|
# define ARRAY_LENGTH(array) (sizeof(array) / sizeof(array[0]))
|
|
|
|
# endif
|
|
|
|
#endif /* ARRAY_LENGTH */
|
|
|
|
|
|
|
|
#ifndef ARRAY_END
|
|
|
|
# define ARRAY_END(array) (&array[ARRAY_LENGTH(array)])
|
|
|
|
#endif /* ARRAY_END */
|
|
|
|
|
|
|
|
#ifndef STRINGIFY
|
|
|
|
# define STRINGIFY_HELPER(x) #x
|
|
|
|
# define STRINGIFY(x) STRINGIFY_HELPER(x)
|
|
|
|
#endif /* STRINGIFY */
|
|
|
|
|
|
|
|
#ifndef offsetof
|
|
|
|
# define offsetof(type, member) __builtin_offsetof(type, member)
|
|
|
|
#endif /* offsetof */
|
|
|
|
|
|
|
|
#ifndef container_of
|
|
|
|
# define container_of(ptr, type, member) \
|
|
|
|
((type *)((char *)(ptr) - offsetof(type, member)))
|
|
|
|
#endif /* container_of */
|
2017-03-16 18:09:27 +03:00
|
|
|
|
|
|
|
#define MDBX_TETRAD(a, b, c, d) \
|
2017-05-24 13:37:06 +03:00
|
|
|
((uint32_t)(a) << 24 | (uint32_t)(b) << 16 | (uint32_t)(c) << 8 | (d))
|
2017-05-24 13:59:50 +03:00
|
|
|
|
|
|
|
#define FIXME "FIXME: " __FILE__ ", " STRINGIFY(__LINE__)
|
|
|
|
|
2017-05-26 17:11:48 +03:00
|
|
|
#ifndef STATIC_ASSERT_MSG
|
|
|
|
# if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L) \
|
|
|
|
|| __has_feature(c_static_assert)
|
|
|
|
# define STATIC_ASSERT_MSG(expr, msg) _Static_assert(expr, msg)
|
2017-05-24 18:42:13 +03:00
|
|
|
# elif defined(static_assert)
|
2017-05-26 17:11:48 +03:00
|
|
|
# define STATIC_ASSERT_MSG(expr, msg) static_assert(expr, msg)
|
2017-05-24 18:42:13 +03:00
|
|
|
# else
|
2017-05-26 17:11:48 +03:00
|
|
|
# define STATIC_ASSERT_MSG(expr, msg) switch (0) {case 0:case (expr):;}
|
2017-05-24 18:42:13 +03:00
|
|
|
# endif
|
|
|
|
#endif /* STATIC_ASSERT */
|
|
|
|
|
2017-05-26 17:11:48 +03:00
|
|
|
#ifndef STATIC_ASSERT
|
|
|
|
# define STATIC_ASSERT(expr) STATIC_ASSERT_MSG(expr, #expr)
|
|
|
|
#endif
|
|
|
|
|
2017-05-24 13:59:50 +03:00
|
|
|
/* *INDENT-ON* */
|
|
|
|
/* clang-format on */
|