summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--libs/libmdbx/libmdbx.vcxproj11
-rw-r--r--libs/libmdbx/src/CMakeLists.dist-minimal308
-rw-r--r--libs/libmdbx/src/CMakeLists.txt613
-rw-r--r--libs/libmdbx/src/GNUmakefile101
-rw-r--r--libs/libmdbx/src/README.md54
-rw-r--r--libs/libmdbx/src/appveyor.yml2
-rw-r--r--libs/libmdbx/src/cmake/compiler.cmake106
-rw-r--r--libs/libmdbx/src/cmake/utils.cmake47
-rw-r--r--libs/libmdbx/src/config.h (renamed from libs/libmdbx/src/src/elements/config.h)18
-rw-r--r--libs/libmdbx/src/mdbx.h131
-rw-r--r--libs/libmdbx/src/src/CMakeLists.txt263
-rw-r--r--libs/libmdbx/src/src/alloy.c14
-rw-r--r--libs/libmdbx/src/src/config.h.in (renamed from libs/libmdbx/src/src/elements/config.h.in)8
-rw-r--r--libs/libmdbx/src/src/core.c (renamed from libs/libmdbx/src/src/elements/core.c)385
-rw-r--r--libs/libmdbx/src/src/debug_begin.h (renamed from libs/libmdbx/src/src/elements/debug_begin.h)0
-rw-r--r--libs/libmdbx/src/src/debug_end.h (renamed from libs/libmdbx/src/src/elements/debug_end.h)0
-rw-r--r--libs/libmdbx/src/src/defs.h (renamed from libs/libmdbx/src/src/elements/defs.h)0
-rw-r--r--libs/libmdbx/src/src/internals.h (renamed from libs/libmdbx/src/src/elements/internals.h)47
-rw-r--r--libs/libmdbx/src/src/lck-posix.c (renamed from libs/libmdbx/src/src/elements/lck-posix.c)27
-rw-r--r--libs/libmdbx/src/src/lck-windows.c (renamed from libs/libmdbx/src/src/elements/lck-windows.c)4
-rw-r--r--libs/libmdbx/src/src/man1/mdbx_chk.12
-rw-r--r--libs/libmdbx/src/src/man1/mdbx_copy.12
-rw-r--r--libs/libmdbx/src/src/man1/mdbx_dump.12
-rw-r--r--libs/libmdbx/src/src/man1/mdbx_load.12
-rw-r--r--libs/libmdbx/src/src/man1/mdbx_stat.12
-rw-r--r--libs/libmdbx/src/src/mdbx_chk.c (renamed from libs/libmdbx/src/src/tools/mdbx_chk.c)8
-rw-r--r--libs/libmdbx/src/src/mdbx_copy.c (renamed from libs/libmdbx/src/src/tools/mdbx_copy.c)2
-rw-r--r--libs/libmdbx/src/src/mdbx_dump.c (renamed from libs/libmdbx/src/src/tools/mdbx_dump.c)2
-rw-r--r--libs/libmdbx/src/src/mdbx_load.c (renamed from libs/libmdbx/src/src/tools/mdbx_load.c)5
-rw-r--r--libs/libmdbx/src/src/mdbx_stat.c (renamed from libs/libmdbx/src/src/tools/mdbx_stat.c)8
-rw-r--r--libs/libmdbx/src/src/ntdll.def (renamed from libs/libmdbx/src/src/elements/ntdll.def)0
-rw-r--r--libs/libmdbx/src/src/options.h (renamed from libs/libmdbx/src/src/elements/options.h)6
-rw-r--r--libs/libmdbx/src/src/osal.c (renamed from libs/libmdbx/src/src/elements/osal.c)33
-rw-r--r--libs/libmdbx/src/src/osal.h (renamed from libs/libmdbx/src/src/elements/osal.h)27
-rw-r--r--libs/libmdbx/src/src/tools/CMakeLists.txt60
-rw-r--r--libs/libmdbx/src/src/version.c.in (renamed from libs/libmdbx/src/src/elements/version.c.in)0
-rw-r--r--libs/libmdbx/src/src/wingetopt.c (renamed from libs/libmdbx/src/src/tools/wingetopt.c)0
-rw-r--r--libs/libmdbx/src/src/wingetopt.h (renamed from libs/libmdbx/src/src/tools/wingetopt.h)0
-rw-r--r--libs/libmdbx/src/test/CMakeLists.txt23
-rw-r--r--libs/libmdbx/src/test/base.h6
-rw-r--r--libs/libmdbx/src/test/osal-unix.cc4
-rw-r--r--libs/libmdbx/src/test/osal-windows.cc12
-rw-r--r--libs/libmdbx/src/test/pcrf/CMakeLists.txt5
-rw-r--r--libs/libmdbx/src/version.c (renamed from libs/libmdbx/src/src/elements/version.c)12
-rw-r--r--tools/mdbx_chk/src/mdbx_chk.cc12
-rw-r--r--tools/mdbx_dump/src/mdbx_dump.cc6
-rw-r--r--tools/mdbx_load/src/mdbx_load.cc9
47 files changed, 1198 insertions, 1191 deletions
diff --git a/libs/libmdbx/libmdbx.vcxproj b/libs/libmdbx/libmdbx.vcxproj
index e7d489ca88..6f690abef4 100644
--- a/libs/libmdbx/libmdbx.vcxproj
+++ b/libs/libmdbx/libmdbx.vcxproj
@@ -26,23 +26,20 @@
<Import Project="$(ProjectDir)..\..\build\vc.common\lib.props" />
<ItemDefinitionGroup>
<ClCompile>
- <PreprocessorDefinitions>MDBX_CONFIG_MANUAL_TLS_CALLBACK=1;MDBX_DEBUG=1;LIBMDBX_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
+ <PreprocessorDefinitions>MDBX_CONFIG_MANUAL_TLS_CALLBACK=1;MDBX_DEBUG=1;LIBMDBX_EXPORTS;__STDC_FORMAT_MACROS=1;__STDC_LIMIT_MACROS=1;__STDC_CONSTANT_MACROS=1;_HAS_EXCEPTIONS=1;MDBX_CONFIG_H=1;%(PreprocessorDefinitions)</PreprocessorDefinitions>
</ClCompile>
<Link>
<AdditionalDependencies>ntdll.lib;%(AdditionalDependencies)</AdditionalDependencies>
</Link>
</ItemDefinitionGroup>
<ItemGroup>
- <ClCompile Include="src\src\elements\lck-windows.c">
+ <ClCompile Include="src\src\lck-windows.c">
<PrecompiledHeader>NotUsing</PrecompiledHeader>
</ClCompile>
- <ClCompile Include="src\src\elements\core.c">
+ <ClCompile Include="src\src\core.c">
<PrecompiledHeader>NotUsing</PrecompiledHeader>
</ClCompile>
- <ClCompile Include="src\src\elements\osal.c">
- <PrecompiledHeader>NotUsing</PrecompiledHeader>
- </ClCompile>
- <ClCompile Include="src\src\elements\version.c">
+ <ClCompile Include="src\src\osal.c">
<PrecompiledHeader>NotUsing</PrecompiledHeader>
</ClCompile>
</ItemGroup>
diff --git a/libs/libmdbx/src/CMakeLists.dist-minimal b/libs/libmdbx/src/CMakeLists.dist-minimal
deleted file mode 100644
index fdcd41d308..0000000000
--- a/libs/libmdbx/src/CMakeLists.dist-minimal
+++ /dev/null
@@ -1,308 +0,0 @@
-##
-## This is the minimal template for CMakeList.txt which could be used
-## to build libmdbx from the "amalgamated form" of libmdbx's source code.
-##
-## The amalgamated form is intended to embedding libmdbx in other projects
-## in cases when using as git-submodule is not acceptable or inconveniently.
-##
-## The amalgamated form could be generated from full git repository
-## on Linux just by `make dist`.
-##
-
-##
-## Copyright 2020 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>.
-##
-
-##
-## libmdbx = { Revised and extended descendant of Symas LMDB. }
-## Please see README.md at https://github.com/erthink/libmdbx
-##
-## Libmdbx is superior to LMDB in terms of features and reliability,
-## not inferior in performance. libmdbx works on Linux, FreeBSD, MacOS X
-## and other systems compliant with POSIX.1-2008, but also support Windows
-## as a complementary platform.
-##
-## The next version is under active non-public development and will be
-## released as MithrilDB and libmithrildb for libraries & packages.
-## Admittedly mythical Mithril is resembling silver but being stronger and
-## lighter than steel. Therefore MithrilDB is rightly relevant name.
-##
-## MithrilDB will be radically different from libmdbx by the new database
-## format and API based on C++17, as well as the Apache 2.0 License.
-## The goal of this revolution is to provide a clearer and robust API,
-## add more features and new valuable properties of database.
-##
-## The Future will (be) Positive. Всё будет хорошо.
-##
-
-cmake_minimum_required(VERSION 3.8.2)
-cmake_policy(PUSH)
-cmake_policy(VERSION 3.8.2)
-if(NOT CMAKE_VERSION VERSION_LESS 3.9)
- cmake_policy(SET CMP0069 NEW)
-endif()
-if(NOT CMAKE_VERSION VERSION_LESS 3.12)
- cmake_policy(SET CMP0075 NEW)
-endif()
-if(NOT CMAKE_VERSION VERSION_LESS 3.13)
- cmake_policy(SET CMP0077 NEW)
-endif()
-
-if(DEFINED PROJECT_NAME)
- set(SUBPROJECT ON)
- set(NOT_SUBPROJECT OFF)
-else()
- set(SUBPROJECT OFF)
- set(NOT_SUBPROJECT ON)
- project(libmdbx C CXX)
-endif()
-
-find_package(Threads REQUIRED)
-
-if(NOT CMAKE_BUILD_TYPE)
- set(CMAKE_BUILD_TYPE MinSizeRel CACHE STRING
- "Choose the type of build, options are: Debug Release RelWithDebInfo MinSizeRel."
- FORCE)
-endif()
-
-list(FIND CMAKE_C_COMPILE_FEATURES c_std_11 HAS_C11)
-if(NOT HAS_C11 LESS 0)
- set(MDBX_C_STANDARD 11)
-else()
- set(MDBX_C_STANDARD 99)
-endif()
-message(STATUS "Use C${MDBX_C_STANDARD} for libmdbx")
-
-# not supported by this (minimal) script
-add_definitions(-DMDBX_AVOID_CRT=0)
-
-# provide build timestamp
-string(TIMESTAMP MDBX_BUILD_TIMESTAMP UTC)
-add_definitions(-DMDBX_BUILD_TIMESTAMP="${MDBX_BUILD_TIMESTAMP}")
-
-# provide compiler info
-execute_process(COMMAND sh -c "${CMAKE_C_COMPILER} --version | head -1"
- OUTPUT_VARIABLE MDBX_BUILD_COMPILER
- OUTPUT_STRIP_TRAILING_WHITESPACE
- ERROR_QUIET
- RESULT_VARIABLE rc)
-if(rc OR NOT MDBX_BUILD_COMPILER)
- string(STRIP "${CMAKE_C_COMPILER_ID}-${CMAKE_C_COMPILER_VERSION}" MDBX_BUILD_COMPILER)
-endif()
-add_definitions(-DMDBX_BUILD_COMPILER="${MDBX_BUILD_COMPILER}")
-
-# provide cpu/arch-system pair
-if(CMAKE_C_COMPILER_TARGET)
- set(MDBX_BUILD_TARGET "${CMAKE_C_COMPILER_TARGET}")
-elseif(CMAKE_C_PLATFORM_ID AND NOT CMAKE_C_PLATFORM_ID STREQUAL CMAKE_SYSTEM_NAME)
- string(STRIP "${CMAKE_C_PLATFORM_ID}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-elseif(CMAKE_LIBRARY_ARCHITECTURE)
- string(STRIP "${CMAKE_LIBRARY_ARCHITECTURE}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-elseif(CMAKE_GENERATOR_PLATFORM AND NOT CMAKE_C_PLATFORM_ID STREQUAL CMAKE_SYSTEM_NAME)
- string(STRIP "${CMAKE_GENERATOR_PLATFORM}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-elseif(CMAKE_SYSTEM_ARCH)
- string(STRIP "${CMAKE_SYSTEM_ARCH}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-else()
- string(STRIP "${CMAKE_SYSTEM_PROCESSOR}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-endif()
-add_definitions(-DMDBX_BUILD_TARGET="${MDBX_BUILD_TARGET}")
-
-# provide build target-config
-if(CMAKE_CONFIGURATION_TYPES)
- add_definitions(-DMDBX_BUILD_CONFIG="$<CONFIG>")
-else()
- add_definitions(-DMDBX_BUILD_CONFIG="${CMAKE_BUILD_TYPE}")
-endif()
-
-# provide build cflags
-set(MDBX_BUILD_FLAGS "")
-list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_FLAGS})
-list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_DEFINES})
-if(CMAKE_CONFIGURATION_TYPES)
- add_definitions(-DMDBX_BUILD_FLAGS_CONFIG="$<$<CONFIG:Debug>:${CMAKE_C_FLAGS_DEBUG} ${CMAKE_C_DEFINES_DEBUG}>$<$<CONFIG:Release>:${CMAKE_C_FLAGS_RELEASE} ${CMAKE_C_DEFINES_RELEASE}>$<$<CONFIG:RelWithDebInfo>:${CMAKE_C_FLAGS_RELWITHDEBINFO} ${CMAKE_C_DEFINES_RELWITHDEBINFO}>$<$<CONFIG:MinSizeRel>:${CMAKE_C_FLAGS_MINSIZEREL} ${CMAKE_C_DEFINES_MINSIZEREL}>")
-else()
- string(TOUPPER ${CMAKE_BUILD_TYPE} CMAKE_BUILD_TYPE_UPPERCASE)
- list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_FLAGS_${CMAKE_BUILD_TYPE_UPPERCASE}})
- list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_DEFINES_${CMAKE_BUILD_TYPE_UPPERCASE}})
-endif()
-list(REMOVE_DUPLICATES MDBX_BUILD_FLAGS)
-string(REPLACE ";" " " MDBX_BUILD_FLAGS "${MDBX_BUILD_FLAGS}")
-add_definitions(-DMDBX_BUILD_FLAGS="${MDBX_BUILD_FLAGS}")
-
-# shared library
-if(NOT DEFINED MDBX_BUILD_SHARED_LIBRARY)
- if(DEFINED BUILD_SHARED_LIBS)
- option(MDBX_BUILD_SHARED_LIBRARY "Build libmdbx as shared library (DLL)" ${BUILD_SHARED_LIBS})
- else()
- option(MDBX_BUILD_SHARED_LIBRARY "Build libmdbx as shared library (DLL)" ON)
- endif()
-endif()
-if(MDBX_BUILD_SHARED_LIBRARY)
- add_library(mdbx SHARED mdbx.c mdbx.h)
- set_target_properties(mdbx PROPERTIES
- C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON
- PUBLIC_HEADER mdbx.h)
- target_compile_definitions(mdbx PRIVATE LIBMDBX_EXPORTS INTERFACE LIBMDBX_IMPORTS)
- if(DEFINED INTERPROCEDURAL_OPTIMIZATION)
- set_target_properties(mdbx PROPERTIES
- INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>)
- endif()
- target_link_libraries(mdbx PRIVATE ${CMAKE_THREAD_LIBS_INIT})
- if(WIN32)
- target_link_libraries(mdbx PRIVATE ntdll.lib)
- endif()
- if(${CMAKE_SYSTEM_NAME} STREQUAL "SunOS" OR ${CMAKE_SYSTEM_NAME} STREQUAL "Solaris")
- target_link_libraries(mdbx PRIVATE kstat)
- endif()
-endif()
-
-# static library used for tools, to avoid rpath/dll-path troubles
-add_library(mdbx-static STATIC mdbx.c mdbx.h)
-set_target_properties(mdbx-static PROPERTIES
- C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON
- PUBLIC_HEADER mdbx.h)
-target_link_libraries(mdbx-static INTERFACE ${CMAKE_THREAD_LIBS_INIT})
-if(DEFINED INTERPROCEDURAL_OPTIMIZATION)
- set_target_properties(mdbx-static PROPERTIES
- INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>)
-endif()
-if(WIN32)
- target_link_libraries(mdbx-static INTERFACE ntdll.lib)
-endif()
-if(${CMAKE_SYSTEM_NAME} STREQUAL "SunOS" OR ${CMAKE_SYSTEM_NAME} STREQUAL "Solaris")
- target_link_libraries(mdbx-static INTERFACE kstat)
-endif()
-
-# mdbx-tools
-foreach(TOOL mdbx_chk mdbx_copy mdbx_stat mdbx_dump mdbx_load)
- add_executable(${TOOL} ${TOOL}.c)
- set_target_properties(${TOOL} PROPERTIES
- C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON)
- if(DEFINED INTERPROCEDURAL_OPTIMIZATION)
- set_target_properties(${TOOL} PROPERTIES
- INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>)
- endif()
- target_link_libraries(${TOOL} mdbx-static)
-endforeach()
-
-# installation
-if(MDBX_BUILD_SHARED_LIBRARY)
- if(CMAKE_VERSION VERSION_LESS 3.12)
- install(TARGETS mdbx EXPORT libmdbx
- RUNTIME
- DESTINATION bin
- COMPONENT runtime
- LIBRARY
- DESTINATION lib
- COMPONENT runtime
- OBJECTS
- DESTINATION lib
- COMPONENT devel
- ARCHIVE
- DESTINATION lib
- COMPONENT devel
- PUBLIC_HEADER
- DESTINATION include
- COMPONENT devel
- INCLUDES
- DESTINATION include
- COMPONENT devel)
- else()
- install(TARGETS mdbx EXPORT libmdbx
- RUNTIME
- DESTINATION bin
- COMPONENT runtime
- LIBRARY
- DESTINATION lib
- COMPONENT runtime
- NAMELINK_COMPONENT devel
- OBJECTS
- DESTINATION lib
- COMPONENT devel
- ARCHIVE
- DESTINATION lib
- COMPONENT devel
- PUBLIC_HEADER
- DESTINATION include
- COMPONENT devel
- INCLUDES
- DESTINATION include
- COMPONENT devel)
- endif()
-endif()
-
-if(CMAKE_VERSION VERSION_LESS 3.12)
- install(TARGETS mdbx-static EXPORT libmdbx EXCLUDE_FROM_ALL
- RUNTIME
- DESTINATION bin
- COMPONENT runtime
- LIBRARY
- DESTINATION lib
- COMPONENT runtime
- OBJECTS
- DESTINATION lib
- COMPONENT devel
- ARCHIVE
- DESTINATION lib
- COMPONENT devel
- PUBLIC_HEADER
- DESTINATION include
- COMPONENT devel
- INCLUDES
- DESTINATION include
- COMPONENT devel)
-else()
- install(TARGETS mdbx-static EXPORT libmdbx EXCLUDE_FROM_ALL
- RUNTIME
- DESTINATION bin
- COMPONENT runtime
- LIBRARY
- DESTINATION lib
- COMPONENT runtime
- NAMELINK_COMPONENT devel
- OBJECTS
- DESTINATION lib
- COMPONENT devel
- ARCHIVE
- DESTINATION lib
- COMPONENT devel
- PUBLIC_HEADER
- DESTINATION include
- COMPONENT devel
- INCLUDES
- DESTINATION include
- COMPONENT devel)
-endif()
-
-install(
- TARGETS
- mdbx_chk
- mdbx_stat
- mdbx_copy
- mdbx_dump
- mdbx_load
- RUNTIME
- DESTINATION bin
- COMPONENT runtime)
-
-install(
- FILES
- man1/mdbx_chk.1
- man1/mdbx_stat.1
- man1/mdbx_copy.1
- man1/mdbx_dump.1
- man1/mdbx_load.1
- DESTINATION man/man1
- COMPONENT doc)
-
-cmake_policy(POP)
diff --git a/libs/libmdbx/src/CMakeLists.txt b/libs/libmdbx/src/CMakeLists.txt
index e67cdbe92a..a17bd35ab6 100644
--- a/libs/libmdbx/src/CMakeLists.txt
+++ b/libs/libmdbx/src/CMakeLists.txt
@@ -35,6 +35,7 @@
##
cmake_minimum_required(VERSION 3.8.2)
+
cmake_policy(PUSH)
cmake_policy(VERSION 3.8.2)
if(NOT CMAKE_VERSION VERSION_LESS 3.13)
@@ -51,17 +52,42 @@ else()
set(CMAKE_INTERPROCEDURAL_OPTIMIZATION_AVAILABLE FALSE)
endif()
+if(EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/.git" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/test/CMakeLists.txt" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/src/core.c" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/src/alloy.c" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/src/config.h.in" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/src/version.c.in" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/src/man1" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/src/mdbx_chk.c")
+ set(MDBX_AMALGAMATED_SOURCE FALSE)
+ find_program(GIT git)
+ if(NOT GIT)
+ message(SEND_ERROR "Git command-line tool not found")
+ endif()
+ set(MDBX_SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}/src")
+elseif(EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/VERSION" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/mdbx.c" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/config.h.in" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/man1" AND
+ EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/mdbx_chk.c")
+ set(MDBX_AMALGAMATED_SOURCE TRUE)
+ set(MDBX_SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}")
+else()
+ message(FATAL_ERROR "Please use libmdbx as a git-submodule or the amalgamated source code")
+endif()
+
if(DEFINED PROJECT_NAME)
set(SUBPROJECT ON)
set(NOT_SUBPROJECT OFF)
- if(NOT DEFINED BUILD_TESTING)
+ if(NOT MDBX_AMALGAMATED_SOURCE AND NOT DEFINED BUILD_TESTING)
set(BUILD_TESTING OFF)
endif()
else()
set(SUBPROJECT OFF)
set(NOT_SUBPROJECT ON)
project(libmdbx C CXX)
- if(NOT DEFINED BUILD_TESTING)
+ if(NOT MDBX_AMALGAMATED_SOURCE AND NOT DEFINED BUILD_TESTING)
set(BUILD_TESTING ON)
endif()
endif()
@@ -72,17 +98,19 @@ if(NOT CMAKE_BUILD_TYPE)
FORCE)
endif()
-macro(add_mdbx_option NAME DESCRIPTION DEFAULT)
- list(APPEND MDBX_BUILD_OPTIONS ${NAME})
- if(NOT ${DEFAULT} STREQUAL "AUTO")
- option(${NAME} "${DESCRIPTION}" ${DEFAULT})
- elseif(NOT DEFINED ${NAME})
- set(${NAME}_AUTO ON)
- endif()
-endmacro()
+if(NOT_SUBPROJECT AND (CMAKE_CROSSCOMPILING OR IOS))
+ set(CMAKE_TRY_COMPILE_TARGET_TYPE STATIC_LIBRARY)
+endif()
-# only for compatibility testing
-# set(CMAKE_CXX_STANDARD 14)
+if(IOS)
+ set(MDBX_BUILD_TOOLS_DEFAULT OFF)
+ if(NOT_SUBPROJECT)
+ cmake_policy(SET CMP0006 OLD)
+ set(CMAKE_XCODE_ATTRIBUTE_CODE_SIGNING_ALLOWED "NO")
+ endif()
+else()
+ set(MDBX_BUILD_TOOLS_DEFAULT ON)
+endif()
if(NOT "$ENV{TEAMCITY_PROCESS_FLOW_ID}" STREQUAL "")
set(CI TEAMCITY)
@@ -113,19 +141,8 @@ set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${MDBX_OUTPUT_DIR})
set(CMAKE_PDB_OUTPUT_DIRECTORY ${MDBX_OUTPUT_DIR})
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${MDBX_OUTPUT_DIR})
-include(CheckLibraryExists)
-include(CheckIncludeFiles)
-include(CheckCCompilerFlag)
-include(CheckSymbolExists)
-include(CheckCSourceRuns)
-include(CheckCXXSourceRuns)
-include(CheckCSourceCompiles)
-include(CheckCXXSourceCompiles)
-include(TestBigEndian)
include(CheckFunctionExists)
include(FindPackageMessage)
-include(CheckStructHasMember)
-include(CMakeDependentOption)
include(GNUInstallDirs)
if(CMAKE_C_COMPILER_ID STREQUAL "MSVC" AND MSVC_VERSION LESS 1900)
@@ -145,15 +162,6 @@ include(cmake/utils.cmake)
include(cmake/compiler.cmake)
include(cmake/profile.cmake)
-find_program(ECHO echo)
-find_program(CAT cat)
-find_program(GIT git)
-find_program(LD ld)
-
-# CHECK_INCLUDE_FILES(unistd.h HAVE_UNISTD_H)
-# CHECK_INCLUDE_FILES(sys/uio.h HAVE_SYS_UIO_H)
-# CHECK_INCLUDE_FILES(sys/stat.h HAVE_SYS_STAT_H)
-
CHECK_FUNCTION_EXISTS(pow NOT_NEED_LIBM)
if(NOT_NEED_LIBM)
set(LIB_MATH "")
@@ -222,57 +230,56 @@ else()
set(LTO_ENABLED FALSE)
endif()
- find_program(VALGRIND valgrind)
- if(VALGRIND)
- # LY: cmake is ugly and nasty.
- # - therefore memcheck-options should be defined before including ctest;
- # - otherwise ctest may ignore it.
- set(MEMORYCHECK_SUPPRESSIONS_FILE
- "${PROJECT_SOURCE_DIR}/test/valgrind_suppress.txt"
- CACHE FILEPATH "Suppressions file for Valgrind" FORCE)
- set(MEMORYCHECK_COMMAND_OPTIONS
- "--trace-children=yes --leak-check=full --track-origins=yes --error-exitcode=42 --error-markers=@ --errors-for-leak-kinds=definite --fair-sched=yes --suppressions=${MEMORYCHECK_SUPPRESSIONS_FILE}"
- CACHE STRING "Valgrind options" FORCE)
- set(VALGRIND_COMMAND_OPTIONS "${MEMORYCHECK_COMMAND_OPTIONS}" CACHE STRING "Valgrind options" FORCE)
- endif()
+ if(NOT MDBX_AMALGAMATED_SOURCE)
+ find_program(VALGRIND valgrind)
+ if(VALGRIND)
+ # LY: cmake is ugly and nasty.
+ # - therefore memcheck-options should be defined before including ctest;
+ # - otherwise ctest may ignore it.
+ set(MEMORYCHECK_SUPPRESSIONS_FILE
+ "${CMAKE_CURRENT_SOURCE_DIR}/test/valgrind_suppress.txt"
+ CACHE FILEPATH "Suppressions file for Valgrind" FORCE)
+ set(MEMORYCHECK_COMMAND_OPTIONS
+ "--trace-children=yes --leak-check=full --track-origins=yes --error-exitcode=42 --error-markers=@ --errors-for-leak-kinds=definite --fair-sched=yes --suppressions=${MEMORYCHECK_SUPPRESSIONS_FILE}"
+ CACHE STRING "Valgrind options" FORCE)
+ set(VALGRIND_COMMAND_OPTIONS "${MEMORYCHECK_COMMAND_OPTIONS}" CACHE STRING "Valgrind options" FORCE)
+ endif()
- #
- # Enable 'make tags' target.
- find_program(CTAGS ctags)
- if(CTAGS)
- add_custom_target(tags COMMAND ${CTAGS} -R -f tags
- WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
- add_custom_target(ctags DEPENDS tags)
- endif(CTAGS)
-
- #
- # Enable 'make reformat' target.
- find_program(CLANG_FORMAT
- NAMES clang-format-6.0 clang-format-5.0 clang-format-4.0
- clang-format-3.9 clang-format-3.8 clang-format-3.7 clang-format)
- if(CLANG_FORMAT AND UNIX)
- add_custom_target(reformat
- VERBATIM
- COMMAND
- git ls-files |
- grep -E \\.\(c|cxx|cc|cpp|h|hxx|hpp\)\(\\.in\)?\$ |
- xargs ${CLANG_FORMAT} -i --style=file
- WORKING_DIRECTORY ${CMAKE_SOURCE_DIR})
- endif()
+ # Enable 'make tags' target.
+ find_program(CTAGS ctags)
+ if(CTAGS)
+ add_custom_target(tags COMMAND ${CTAGS} -R -f tags
+ WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR})
+ add_custom_target(ctags DEPENDS tags)
+ endif(CTAGS)
+
+ # Enable 'make reformat' target.
+ find_program(CLANG_FORMAT
+ NAMES clang-format-11.0 clang-format-10.0 clang-format-9.0 clang-format-8.0 clang-format)
+ if(CLANG_FORMAT AND UNIX)
+ add_custom_target(reformat
+ VERBATIM
+ COMMAND
+ git ls-files |
+ grep -E \\.\(c|cxx|cc|cpp|h|hxx|hpp\)\(\\.in\)?\$ |
+ xargs ${CLANG_FORMAT} -i --style=file
+ WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR})
+ endif()
- if(NOT "${PROJECT_BINARY_DIR}" STREQUAL "${PROJECT_SOURCE_DIR}")
- add_custom_target(distclean)
- add_custom_command(TARGET distclean
- COMMAND ${CMAKE_COMMAND} -E remove_directory "${PROJECT_BINARY_DIR}"
- COMMENT "Removing the build directory and its content")
- elseif(IS_DIRECTORY .git AND GIT)
- add_custom_target(distclean)
- add_custom_command(TARGET distclean
- WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
- COMMAND ${GIT} submodule foreach --recursive git clean -f -X -d
- COMMAND ${GIT} clean -f -X -d
- COMMENT "Removing all build files from the source directory")
- endif()
+ if(NOT "${PROJECT_BINARY_DIR}" STREQUAL "${PROJECT_SOURCE_DIR}")
+ add_custom_target(distclean)
+ add_custom_command(TARGET distclean
+ COMMAND ${CMAKE_COMMAND} -E remove_directory "${PROJECT_BINARY_DIR}"
+ COMMENT "Removing the build directory and its content")
+ elseif(IS_DIRECTORY .git AND GIT)
+ add_custom_target(distclean)
+ add_custom_command(TARGET distclean
+ WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
+ COMMAND ${GIT} submodule foreach --recursive git clean -f -X -d
+ COMMAND ${GIT} clean -f -X -d
+ COMMENT "Removing all build files from the source directory")
+ endif()
+ endif(NOT MDBX_AMALGAMATED_SOURCE)
setup_compile_flags()
endif(SUBPROJECT)
@@ -285,6 +292,66 @@ else()
endif()
message(STATUS "Use C${MDBX_C_STANDARD} for libmdbx")
+if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows" AND EXISTS "${MDBX_SOURCE_DIR}/ntdll.def")
+ if(MSVC)
+ if(NOT MSVC_LIB_EXE)
+ # Find lib.exe
+ get_filename_component(CL_NAME ${CMAKE_C_COMPILER} NAME)
+ string(REPLACE cl.exe lib.exe MSVC_LIB_EXE ${CL_NAME})
+ find_program(MSVC_LIB_EXE ${MSVC_LIB_EXE})
+ endif()
+ if(MSVC_LIB_EXE)
+ message(STATUS "Found MSVC's lib tool: ${MSVC_LIB_EXE}")
+ set(MDBX_NTDLL_EXTRA_IMPLIB "${CMAKE_CURRENT_BINARY_DIR}/mdbx_ntdll_extra.lib")
+ add_custom_command(OUTPUT "${MDBX_NTDLL_EXTRA_IMPLIB}"
+ COMMENT "Create extra-import-library for ntdll.dll"
+ MAIN_DEPENDENCY "${MDBX_SOURCE_DIR}/ntdll.def"
+ COMMAND ${MSVC_LIB_EXE} /def:"${MDBX_SOURCE_DIR}/ntdll.def" /out:"${MDBX_NTDLL_EXTRA_IMPLIB}" ${INITIAL_CMAKE_STATIC_LINKER_FLAGS})
+ else()
+ message(WARNING "MSVC's lib tool not found")
+ endif()
+ elseif(MINGW OR MINGW64)
+ if(NOT DLLTOOL)
+ # Find dlltool
+ get_filename_component(GCC_NAME ${CMAKE_C_COMPILER} NAME)
+ string(REPLACE gcc dlltool DLLTOOL_NAME ${GCC_NAME})
+ find_program(DLLTOOL NAMES ${DLLTOOL_NAME})
+ endif()
+ if(DLLTOOL)
+ message(STATUS "Found dlltool: ${DLLTOOL}")
+ set(MDBX_NTDLL_EXTRA_IMPLIB "${CMAKE_CURRENT_BINARY_DIR}/mdbx_ntdll_extra.a")
+ add_custom_command(OUTPUT "${MDBX_NTDLL_EXTRA_IMPLIB}"
+ COMMENT "Create extra-import-library for ntdll.dll"
+ MAIN_DEPENDENCY "${MDBX_SOURCE_DIR}/ntdll.def"
+ COMMAND ${DLLTOOL} -d "${MDBX_SOURCE_DIR}/ntdll.def" -l "${MDBX_NTDLL_EXTRA_IMPLIB}")
+ else()
+ message(WARNING "dlltool not found")
+ endif()
+ endif()
+
+ if(MDBX_NTDLL_EXTRA_IMPLIB)
+ # LY: Sometimes CMake requires a nightmarish magic for simple things.
+ # 1) create a target out of the library compilation result
+ add_custom_target(ntdll_extra_target DEPENDS "${MDBX_NTDLL_EXTRA_IMPLIB}")
+ # 2) create an library target out of the library compilation result
+ add_library(ntdll_extra STATIC IMPORTED GLOBAL)
+ add_dependencies(ntdll_extra ntdll_extra_target)
+ # 3) specify where the library is (and where to find the headers)
+ set_target_properties(ntdll_extra
+ PROPERTIES
+ IMPORTED_LOCATION "${MDBX_NTDLL_EXTRA_IMPLIB}")
+ endif()
+endif()
+
+macro(add_mdbx_option NAME DESCRIPTION DEFAULT)
+ list(APPEND MDBX_BUILD_OPTIONS ${NAME})
+ if(NOT ${DEFAULT} STREQUAL "AUTO")
+ option(${NAME} "${DESCRIPTION}" ${DEFAULT})
+ elseif(NOT DEFINED ${NAME})
+ set(${NAME}_AUTO ON)
+ endif()
+endmacro()
+
################################################################################
################################################################################
#
@@ -297,42 +364,400 @@ message(STATUS "Use C${MDBX_C_STANDARD} for libmdbx")
#
set(MDBX_BUILD_OPTIONS ENABLE_ASAN MDBX_USE_VALGRIND ENABLE_GPROF ENABLE_GCOV)
+add_mdbx_option(MDBX_INSTALL_STATIC "Build and install libmdbx for static linking" OFF)
add_mdbx_option(MDBX_BUILD_SHARED_LIBRARY "Build libmdbx as shared library (DLL)" ${BUILD_SHARED_LIBS})
-add_mdbx_option(MDBX_ALLOY_BUILD "Build MDBX library as single object file" ON)
+add_mdbx_option(MDBX_BUILD_TOOLS "Build MDBX tools (mdbx_chk/stat/dump/load/copy)" ${MDBX_BUILD_TOOLS_DEFAULT})
add_mdbx_option(MDBX_TXN_CHECKOWNER "Checking transaction matches the calling thread inside libmdbx's API" ON)
add_mdbx_option(MDBX_TXN_CHECKPID "Paranoid checking PID inside libmdbx's API" AUTO)
mark_as_advanced(MDBX_TXN_CHECKPID)
if(${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
- add_mdbx_option(MDBX_DISABLE_GNU_SOURCE "Don't use nonstandard GNU/Linux extension functions" OFF)
+ add_mdbx_option(MDBX_DISABLE_GNU_SOURCE "Don't use GNU/Linux libc extensions" OFF)
mark_as_advanced(MDBX_DISABLE_GNU_SOURCE)
endif()
-if(${CMAKE_SYSTEM_NAME} STREQUAL "Darwin")
+if(${CMAKE_SYSTEM_NAME} STREQUAL "Darwin" OR IOS)
add_mdbx_option(MDBX_OSX_SPEED_INSTEADOF_DURABILITY "Disable use fcntl(F_FULLFSYNC) in favor of speed" OFF)
mark_as_advanced(MDBX_OSX_SPEED_INSTEADOF_DURABILITY)
endif()
if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
- add_mdbx_option(MDBX_AVOID_CRT "Avoid dependence from MSVC CRT" ${NOT_SUBPROJECT})
- if(NOT MDBX_BUILD_SHARED_LIBRARY)
- add_mdbx_option(MDBX_CONFIG_MANUAL_TLS_CALLBACK
- "Provide mdbx_dll_handler() for manual initialization" OFF)
- mark_as_advanced(MDBX_CONFIG_MANUAL_TLS_CALLBACK)
+ if(MDBX_NTDLL_EXTRA_IMPLIB)
+ add_mdbx_option(MDBX_AVOID_CRT "Avoid dependence from MSVC CRT and use ntdll.dll instead" ${NOT_SUBPROJECT})
endif()
+ add_mdbx_option(MDBX_CONFIG_MANUAL_TLS_CALLBACK
+ "Provide mdbx_dll_handler() for manual initialization" OFF)
+ mark_as_advanced(MDBX_CONFIG_MANUAL_TLS_CALLBACK)
else()
add_mdbx_option(MDBX_USE_OFDLOCKS "Use Open file description locks (aka OFD locks, non-POSIX)" AUTO)
mark_as_advanced(MDBX_USE_OFDLOCKS)
endif()
add_mdbx_option(MDBX_LOCKING "Locking method (Win32=-1, SysV=5, POSIX=1988, POSIX=2001, POSIX=2008, Futexes=1995)" AUTO)
mark_as_advanced(MDBX_LOCKING)
-add_mdbx_option(MDBX_TRUST_RTC "Does a system have battery-backed Real-Time Clock or just a fake." AUTO)
+add_mdbx_option(MDBX_TRUST_RTC "Does a system have battery-backed Real-Time Clock or just a fake" AUTO)
mark_as_advanced(MDBX_TRUST_RTC)
-option(MDBX_ENABLE_TESTS "Build MDBX tests." ${BUILD_TESTING})
-option(MDBX_FORCE_ASSERTIONS "Force enable assertion checking." OFF)
+option(MDBX_FORCE_ASSERTIONS "Force enable assertion checking" OFF)
+
+if(NOT MDBX_AMALGAMATED_SOURCE)
+ add_mdbx_option(MDBX_ALLOY_BUILD "Build MDBX library through single/alloyed object file" ON)
+ option(MDBX_ENABLE_TESTS "Build MDBX tests" ${BUILD_TESTING})
+endif(NOT MDBX_AMALGAMATED_SOURCE)
+
+if((MDBX_BUILD_TOOLS OR MDBX_ENABLE_TESTS) AND MDBX_BUILD_SHARED_LIBRARY)
+ add_mdbx_option(MDBX_LINK_TOOLS_NONSTATIC "Link MDBX tools with non-static libmdbx" OFF)
+else()
+ unset(MDBX_LINK_TOOLS_NONSTATIC CACHE)
+endif()
################################################################################
################################################################################
-add_subdirectory(src)
-if(MDBX_ENABLE_TESTS)
+# Get version
+fetch_version(MDBX "${CMAKE_CURRENT_SOURCE_DIR}/VERSION" FALSE)
+message(STATUS "libmdbx version is ${MDBX_VERSION}")
+
+# sources list
+set(LIBMDBX_SOURCES mdbx.h "${CMAKE_CURRENT_BINARY_DIR}/config.h")
+if(MDBX_AMALGAMATED_SOURCE)
+ list(APPEND LIBMDBX_SOURCES mdbx.c)
+else()
+ # generate version file
+ configure_file("${MDBX_SOURCE_DIR}/version.c.in"
+ "${CMAKE_CURRENT_BINARY_DIR}/version.c" ESCAPE_QUOTES)
+ file(SHA256 "${CMAKE_CURRENT_BINARY_DIR}/version.c" MDBX_SOURCERY_DIGEST)
+ string(MAKE_C_IDENTIFIER "${MDBX_GIT_DESCRIBE}" MDBX_SOURCERY_SUFFIX)
+ set(MDBX_BUILD_SOURCERY "${MDBX_SOURCERY_DIGEST}_${MDBX_SOURCERY_SUFFIX}")
+
+ if(MDBX_ALLOY_BUILD)
+ list(APPEND LIBMDBX_SOURCES ${MDBX_SOURCE_DIR}/alloy.c)
+ include_directories("${MDBX_SOURCE_DIR}" "${CMAKE_CURRENT_BINARY_DIR}")
+ else()
+ list(APPEND LIBMDBX_SOURCES
+ "${CMAKE_CURRENT_BINARY_DIR}/version.c"
+ "${MDBX_SOURCE_DIR}/options.h" "${MDBX_SOURCE_DIR}/defs.h"
+ "${MDBX_SOURCE_DIR}/internals.h" "${MDBX_SOURCE_DIR}/osal.h"
+ "${MDBX_SOURCE_DIR}/core.c" "${MDBX_SOURCE_DIR}/osal.c"
+ "${MDBX_SOURCE_DIR}/lck-posix.c" "${MDBX_SOURCE_DIR}/lck-windows.c")
+ include_directories("${MDBX_SOURCE_DIR}")
+ endif()
+endif(MDBX_AMALGAMATED_SOURCE)
+
+macro(target_setup_options TARGET)
+ if(DEFINED INTERPROCEDURAL_OPTIMIZATION)
+ set_target_properties(${TARGET} PROPERTIES
+ INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>)
+ endif()
+ if(CC_HAS_FASTMATH)
+ target_compile_options(${TARGET} PRIVATE "-ffast-math")
+ endif()
+ if(CC_HAS_VISIBILITY)
+ target_compile_options(${TARGET} PRIVATE "-fvisibility=hidden")
+ endif()
+ if(BUILD_FOR_NATIVE_CPU AND CC_HAS_ARCH_NATIVE)
+ target_compile_options(${TARGET} PUBLIC "-march=native")
+ endif()
+endmacro()
+
+macro(libmdbx_setup_libs TARGET MODE)
+ target_link_libraries(${TARGET} ${MODE} ${CMAKE_THREAD_LIBS_INIT})
+ if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
+ target_link_libraries(${TARGET} ${MODE} ntdll.lib)
+ if(MDBX_NTDLL_EXTRA_IMPLIB)
+ target_link_libraries(${TARGET} ${MODE} ntdll_extra)
+ endif()
+ elseif(${CMAKE_SYSTEM_NAME} STREQUAL "SunOS" OR ${CMAKE_SYSTEM_NAME} STREQUAL "Solaris")
+ target_link_libraries(${TARGET} ${MODE} kstat)
+ elseif(${CMAKE_SYSTEM_NAME} STREQUAL "Android")
+ target_link_libraries(${TARGET} ${MODE} log)
+ endif()
+endmacro()
+
+# build static library
+if(MDBX_INSTALL_STATIC)
+ add_library(mdbx-static STATIC ${LIBMDBX_SOURCES})
+else()
+ add_library(mdbx-static STATIC EXCLUDE_FROM_ALL ${LIBMDBX_SOURCES})
+endif()
+set_target_properties(mdbx-static PROPERTIES
+ C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON
+ PUBLIC_HEADER mdbx.h)
+target_compile_definitions(mdbx-static PRIVATE MDBX_BUILD_SHARED_LIBRARY=0)
+target_setup_options(mdbx-static)
+libmdbx_setup_libs(mdbx-static INTERFACE)
+if(MDBX_BUILD_SHARED_LIBRARY)
+ set_target_properties(mdbx-static PROPERTIES OUTPUT_NAME mdbx-static)
+else()
+ set_target_properties(mdbx-static PROPERTIES OUTPUT_NAME mdbx)
+endif()
+
+################################################################################
+
+# build shared library
+if(MDBX_BUILD_SHARED_LIBRARY)
+ add_library(mdbx SHARED ${LIBMDBX_SOURCES})
+ set_target_properties(mdbx PROPERTIES
+ C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON
+ PUBLIC_HEADER mdbx.h)
+ target_compile_definitions(mdbx PRIVATE LIBMDBX_EXPORTS MDBX_BUILD_SHARED_LIBRARY=1 INTERFACE LIBMDBX_IMPORTS)
+ target_setup_options(mdbx)
+ libmdbx_setup_libs(mdbx PRIVATE)
+ if(CC_HAS_VISIBILITY AND (LTO_ENABLED OR INTERPROCEDURAL_OPTIMIZATION))
+ set_target_properties(mdbx PROPERTIES LINK_FLAGS "-fvisibility=hidden")
+ endif()
+ list(APPEND MDBX_BUILD_FLAGS ${CMAKE_SHARED_LINKER_FLAGS})
+endif()
+
+if(MDBX_BUILD_SHARED_LIBRARY AND MDBX_LINK_TOOLS_NONSTATIC)
+ set(TOOL_MDBX_LIB mdbx)
+
+ # use, i.e. don't skip the full RPATH for the build tree
+ set(CMAKE_SKIP_BUILD_RPATH FALSE)
+
+ # when building, don't use the install RPATH already (but later on when installing)
+ set(CMAKE_BUILD_WITH_INSTALL_RPATH FALSE)
+
+ # add the automatically determined parts of the RPATH
+ # which point to directories outside the build tree to the install RPATH
+ set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE)
+
+ # the RPATH to be used when installing, but only if it's not a system directory
+ list(FIND CMAKE_PLATFORM_IMPLICIT_LINK_DIRECTORIES "${CMAKE_INSTALL_PREFIX}/lib" isSystemDir)
+ if(isSystemDir EQUAL -1)
+ if(${CMAKE_SYSTEM_NAME} STREQUAL "Darwin")
+ set(CMAKE_INSTALL_RPATH "@executable_path/../lib")
+ else()
+ set(CMAKE_INSTALL_RPATH "\$ORIGIN/../lib")
+ endif()
+ endif()
+else()
+ set(TOOL_MDBX_LIB mdbx-static)
+endif()
+
+# build mdbx-tools
+if(MDBX_BUILD_TOOLS)
+ if(NOT MDBX_AMALGAMATED_SOURCE AND ${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
+ set(WINGETOPT_SRC ${MDBX_SOURCE_DIR}/wingetopt.c ${MDBX_SOURCE_DIR}/wingetopt.h)
+ else()
+ set(WINGETOPT_SRC "")
+ endif()
+
+ foreach(TOOL mdbx_chk mdbx_copy mdbx_stat mdbx_dump mdbx_load)
+ add_executable(${TOOL} mdbx.h ${MDBX_SOURCE_DIR}/${TOOL}.c ${WINGETOPT_SRC})
+ set_target_properties(${TOOL} PROPERTIES
+ C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON)
+ target_setup_options(${TOOL})
+ target_link_libraries(${TOOL} ${TOOL_MDBX_LIB})
+ endforeach()
+ if(LIB_MATH)
+ target_link_libraries(mdbx_chk ${LIB_MATH})
+ target_link_libraries(mdbx_stat ${LIB_MATH})
+ endif()
+endif()
+
+################################################################################
+
+# mdbx-shared-lib installation
+if(MDBX_BUILD_SHARED_LIBRARY)
+ if(CMAKE_VERSION VERSION_LESS 3.12)
+ install(TARGETS mdbx EXPORT libmdbx
+ RUNTIME
+ DESTINATION bin
+ COMPONENT runtime
+ LIBRARY
+ DESTINATION lib
+ COMPONENT runtime
+ OBJECTS
+ DESTINATION lib
+ COMPONENT devel
+ ARCHIVE
+ DESTINATION lib
+ COMPONENT devel
+ PUBLIC_HEADER
+ DESTINATION include
+ COMPONENT devel
+ INCLUDES
+ DESTINATION include
+ COMPONENT devel)
+ else()
+ install(TARGETS mdbx EXPORT libmdbx
+ RUNTIME
+ DESTINATION bin
+ COMPONENT runtime
+ LIBRARY
+ DESTINATION lib
+ COMPONENT runtime
+ NAMELINK_COMPONENT devel
+ OBJECTS
+ DESTINATION lib
+ COMPONENT devel
+ ARCHIVE
+ DESTINATION lib
+ COMPONENT devel
+ PUBLIC_HEADER
+ DESTINATION include
+ COMPONENT devel
+ INCLUDES
+ DESTINATION include
+ COMPONENT devel)
+ endif()
+endif(MDBX_BUILD_SHARED_LIBRARY)
+
+# mdbx-tools installation
+if(MDBX_BUILD_TOOLS)
+ install(
+ TARGETS
+ mdbx_chk
+ mdbx_stat
+ mdbx_copy
+ mdbx_dump
+ mdbx_load
+ RUNTIME
+ DESTINATION bin
+ COMPONENT runtime)
+ install(
+ FILES
+ "${MDBX_SOURCE_DIR}/man1/mdbx_chk.1"
+ "${MDBX_SOURCE_DIR}/man1/mdbx_stat.1"
+ "${MDBX_SOURCE_DIR}/man1/mdbx_copy.1"
+ "${MDBX_SOURCE_DIR}/man1/mdbx_dump.1"
+ "${MDBX_SOURCE_DIR}/man1/mdbx_load.1"
+ DESTINATION man/man1
+ COMPONENT doc)
+endif(MDBX_BUILD_TOOLS)
+
+# mdbx-static-lib installation
+if(MDBX_INSTALL_STATIC)
+ if(CMAKE_VERSION VERSION_LESS 3.12)
+ install(TARGETS mdbx-static EXPORT libmdbx
+ RUNTIME
+ DESTINATION bin
+ COMPONENT runtime
+ LIBRARY
+ DESTINATION lib
+ COMPONENT runtime
+ OBJECTS
+ DESTINATION lib
+ COMPONENT devel
+ ARCHIVE
+ DESTINATION lib
+ COMPONENT devel
+ PUBLIC_HEADER
+ DESTINATION include
+ COMPONENT devel
+ INCLUDES
+ DESTINATION include
+ COMPONENT devel)
+ else()
+ install(TARGETS mdbx-static EXPORT libmdbx
+ RUNTIME
+ DESTINATION bin
+ COMPONENT runtime
+ LIBRARY
+ DESTINATION lib
+ COMPONENT runtime
+ NAMELINK_COMPONENT devel
+ OBJECTS
+ DESTINATION lib
+ COMPONENT devel
+ ARCHIVE
+ DESTINATION lib
+ COMPONENT devel
+ PUBLIC_HEADER
+ DESTINATION include
+ COMPONENT devel
+ INCLUDES
+ DESTINATION include
+ COMPONENT devel)
+ endif()
+endif(MDBX_INSTALL_STATIC)
+
+################################################################################
+
+# collect options & build info
+string(TIMESTAMP MDBX_BUILD_TIMESTAMP UTC)
+set(MDBX_BUILD_FLAGS ${CMAKE_C_FLAGS})
+
+# append cmake's build-type flags and defines
+if(NOT CMAKE_CONFIGURATION_TYPES)
+ list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_FLAGS_${CMAKE_BUILD_TYPE_UPPERCASE}})
+endif()
+
+# get definitions
+get_target_property(defs_list mdbx-static COMPILE_DEFINITIONS)
+if(defs_list)
+ list(APPEND MDBX_BUILD_FLAGS ${defs_list})
+endif()
+
+# get target compile options
+get_target_property(options_list mdbx-static COMPILE_OPTIONS)
+if(options_list)
+ list(APPEND MDBX_BUILD_FLAGS ${options_list})
+endif()
+
+list(REMOVE_DUPLICATES MDBX_BUILD_FLAGS)
+string(REPLACE ";" " " MDBX_BUILD_FLAGS "${MDBX_BUILD_FLAGS}")
+if(CMAKE_CONFIGURATION_TYPES)
+ # add dynamic part via per-configuration define
+ message(STATUS "MDBX Compile Flags: ${MDBX_BUILD_FLAGS} <AND CONFIGURATION DEPENDENT>")
+ add_definitions(-DMDBX_BUILD_FLAGS_CONFIG="$<$<CONFIG:Debug>:${CMAKE_C_FLAGS_DEBUG} ${CMAKE_C_DEFINES_DEBUG}>$<$<CONFIG:Release>:${CMAKE_C_FLAGS_RELEASE} ${CMAKE_C_DEFINES_RELEASE}>$<$<CONFIG:RelWithDebInfo>:${CMAKE_C_FLAGS_RELWITHDEBINFO} ${CMAKE_C_DEFINES_RELWITHDEBINFO}>$<$<CONFIG:MinSizeRel>:${CMAKE_C_FLAGS_MINSIZEREL} ${CMAKE_C_DEFINES_MINSIZEREL}>")
+else()
+ message(STATUS "MDBX Compile Flags: ${MDBX_BUILD_FLAGS}")
+endif()
+
+# get compiler info
+execute_process(COMMAND sh -c "${CMAKE_C_COMPILER} --version | head -1"
+ OUTPUT_VARIABLE MDBX_BUILD_COMPILER
+ OUTPUT_STRIP_TRAILING_WHITESPACE
+ ERROR_QUIET
+ RESULT_VARIABLE rc)
+if(rc OR NOT MDBX_BUILD_COMPILER)
+ string(STRIP "${CMAKE_C_COMPILER_ID}-${CMAKE_C_COMPILER_VERSION}" MDBX_BUILD_COMPILER)
+endif()
+
+# make a build-target triplet
+if(CMAKE_C_COMPILER_TARGET)
+ set(MDBX_BUILD_TARGET "${CMAKE_C_COMPILER_TARGET}")
+elseif(CMAKE_C_PLATFORM_ID AND NOT CMAKE_C_PLATFORM_ID STREQUAL CMAKE_SYSTEM_NAME)
+ string(STRIP "${CMAKE_C_PLATFORM_ID}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
+elseif(CMAKE_LIBRARY_ARCHITECTURE)
+ string(STRIP "${CMAKE_LIBRARY_ARCHITECTURE}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
+elseif(CMAKE_GENERATOR_PLATFORM AND NOT CMAKE_C_PLATFORM_ID STREQUAL CMAKE_SYSTEM_NAME)
+ string(STRIP "${CMAKE_GENERATOR_PLATFORM}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
+elseif(CMAKE_SYSTEM_ARCH)
+ string(STRIP "${CMAKE_SYSTEM_ARCH}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
+else()
+ string(STRIP "${CMAKE_SYSTEM_PROCESSOR}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
+endif()
+
+# provide build-type
+if(CMAKE_CONFIGURATION_TYPES)
+ add_definitions(-DMDBX_BUILD_TYPE="$<CONFIG>")
+else()
+ set(MDBX_BUILD_TYPE ${CMAKE_BUILD_TYPE})
+endif()
+
+# options
+set(options VERSION C_COMPILER CXX_COMPILER)
+foreach(item IN LISTS options)
+ if(DEFINED ${item})
+ set(value "${${item}}")
+ elseif(DEFINED MDBX_${item})
+ set(item MDBX_${item})
+ set(value "${${item}}")
+ elseif(DEFINED CMAKE_${item})
+ set(item CMAKE_${item})
+ set(value "${${item}}")
+ else()
+ set(value "undefined")
+ endif()
+ message(STATUS "${item}: ${value}")
+endforeach(item)
+
+# provide config.h for library build info
+configure_file("${MDBX_SOURCE_DIR}/config.h.in"
+ "${CMAKE_CURRENT_BINARY_DIR}/config.h" ESCAPE_QUOTES)
+add_definitions(-DMDBX_CONFIG_H="${CMAKE_CURRENT_BINARY_DIR}/config.h")
+
+################################################################################
+
+if(NOT MDBX_AMALGAMATED_SOURCE AND MDBX_ENABLE_TESTS)
add_subdirectory(test)
endif()
diff --git a/libs/libmdbx/src/GNUmakefile b/libs/libmdbx/src/GNUmakefile
index 5d9d25b9bf..6aeb1f92ee 100644
--- a/libs/libmdbx/src/GNUmakefile
+++ b/libs/libmdbx/src/GNUmakefile
@@ -9,7 +9,7 @@
SHELL := env bash
# install sandbox
-SANDBOX ?=
+DESTDIR ?=
# install prefixes (inside sandbox)
prefix ?= /usr/local
@@ -19,9 +19,10 @@ mandir ?= $(prefix)/man
suffix ?=
CC ?= gcc
+CFLAGS_EXTRA ?=
LD ?= ld
MDBX_OPTIONS ?= -DNDEBUG=1
-CFLAGS ?= -O2 -g -Wall -Werror -Wextra -Wpedantic -ffunction-sections -fPIC -fvisibility=hidden -std=gnu11 -pthread -Wno-error=attributes
+CFLAGS ?= -O2 -g -Wall -Werror -Wextra -Wpedantic -ffunction-sections -fPIC -fvisibility=hidden -std=gnu11 -pthread -Wno-error=attributes $(CFLAGS_EXTRA)
# -Wno-tautological-compare
# HINT: Try append '--no-as-needed,-lrt' for ability to built with modern glibc, but then run with the old.
@@ -61,7 +62,7 @@ strip: all
clean:
rm -rf $(TOOLS) mdbx_test @* *.[ao] *.[ls]o *~ tmp.db/* \
*.gcov *.log *.err src/*.o test/*.o mdbx_example dist \
- config.h src/elements/config.h src/elements/version.c *.tar*
+ config.h src/config.h src/version.c *.tar*
libmdbx.a: mdbx-static.o
$(AR) rs $@ $?
@@ -112,7 +113,8 @@ define uname2titer
esac
endef
-DIST_EXTRA := LICENSE README.md CMakeLists.txt GNUmakefile Makefile $(addprefix man1/, $(MANPAGES))
+DIST_EXTRA := LICENSE README.md CMakeLists.txt GNUmakefile Makefile VERSION config.h.in \
+ $(addprefix man1/, $(MANPAGES)) cmake/compiler.cmake cmake/profile.cmake cmake/utils.cmake
DIST_SRC := mdbx.h mdbx.c $(addsuffix .c, $(TOOLS))
TEST_DB ?= $(shell [ -d /dev/shm ] && echo /dev/shm || echo /tmp)/mdbx-test.db
@@ -125,14 +127,16 @@ TEST_OBJ := $(patsubst %.cc,%.o,$(TEST_SRC))
CXX ?= g++
CXXSTD ?= $(shell $(CXX) -std=c++27 -c test/test.cc -o /dev/null 2>/dev/null && echo -std=c++17 || echo -std=c++11)
CXXFLAGS := $(CXXSTD) $(filter-out -std=gnu11,$(CFLAGS))
+TAR ?= $(shell which gnu-tar || echo tar)
MAN_SRCDIR := src/man1/
-ALLOY_DEPS := $(wildcard src/elements/*)
+ALLOY_DEPS := $(wildcard src/*)
MDBX_VERSION_GIT = ${shell set -o pipefail; git describe --tags | sed -n 's|^v*\([0-9]\{1,\}\.[0-9]\{1,\}\.[0-9]\{1,\}\)\(.*\)|\1|p' || echo 'Please fetch tags and/or install latest git version'}
+MDBX_REVISION_GIT = $(shell git rev-list --count --no-merges HEAD || echo 'Please fetch tags and/or install latest git version')
MDBX_GIT_TIMESTAMP = $(shell git show --no-patch --format=%cI HEAD || echo 'Please install latest get version')
MDBX_GIT_DESCRIBE = $(shell git describe --tags --long --dirty=-dirty || echo 'Please fetch tags and/or install latest git version')
MDBX_VERSION_SUFFIX = $(shell set -o pipefail; echo -n '$(MDBX_GIT_DESCRIBE)' | tr -c -s '[a-zA-Z0-9]' _)
-MDBX_BUILD_SOURCERY = $(shell set -o pipefail; $(MAKE) -s src/elements/version.c && (openssl dgst -r -sha256 src/elements/version.c || sha256sum src/elements/version.c || shasum -a 256 src/elements/version.c) 2>/dev/null | cut -d ' ' -f 1 || echo 'Please install openssl or sha256sum or shasum')_$(MDBX_VERSION_SUFFIX)
+MDBX_BUILD_SOURCERY = $(shell set -o pipefail; $(MAKE) -s src/version.c && (openssl dgst -r -sha256 src/version.c || sha256sum src/version.c || shasum -a 256 src/version.c) 2>/dev/null | cut -d ' ' -f 1 || echo 'Please install openssl or sha256sum or shasum')_$(MDBX_VERSION_SUFFIX)
check: test mdbx_example dist
@@ -175,7 +179,7 @@ $(patsubst %.cc,%.o,$(1)): $(1) $(TEST_INC) mdbx.h $(lastword $(MAKEFILE_LIST))
endef
$(foreach file,$(TEST_SRC),$(eval $(call test-rule,$(file))))
-mdbx_%: src/tools/mdbx_%.c libmdbx.a
+mdbx_%: src/mdbx_%.c libmdbx.a
$(CC) $(CFLAGS) $(MDBX_OPTIONS) '-DMDBX_CONFIG_H="config.h"' $^ $(EXE_LDFLAGS) $(LIBS) -o $@
mdbx_test: $(TEST_OBJ) libmdbx.$(SO_SUFFIX)
@@ -183,7 +187,7 @@ mdbx_test: $(TEST_OBJ) libmdbx.$(SO_SUFFIX)
git_DIR := $(shell if [ -d .git ]; then echo .git; elif [ -s .git -a -f .git ]; then grep '^gitdir: ' .git | cut -d ':' -f 2; else echo "Please use libmdbx as a git-submodule or the amalgamated source code" >&2 && echo git_directory; fi)
-src/elements/version.c: src/elements/version.c.in $(lastword $(MAKEFILE_LIST)) $(git_DIR)/HEAD $(git_DIR)/index $(git_DIR)/refs/tags
+src/version.c: src/version.c.in $(lastword $(MAKEFILE_LIST)) $(git_DIR)/HEAD $(git_DIR)/index $(git_DIR)/refs/tags
sed \
-e "s|@MDBX_GIT_TIMESTAMP@|$(MDBX_GIT_TIMESTAMP)|" \
-e "s|@MDBX_GIT_TREE@|$(shell git show --no-patch --format=%T HEAD || echo 'Please install latest get version')|" \
@@ -192,10 +196,10 @@ src/elements/version.c: src/elements/version.c.in $(lastword $(MAKEFILE_LIST)) $
-e "s|\$${MDBX_VERSION_MAJOR}|$(shell echo '$(MDBX_VERSION_GIT)' | cut -d . -f 1)|" \
-e "s|\$${MDBX_VERSION_MINOR}|$(shell echo '$(MDBX_VERSION_GIT)' | cut -d . -f 2)|" \
-e "s|\$${MDBX_VERSION_RELEASE}|$(shell echo '$(MDBX_VERSION_GIT)' | cut -d . -f 3)|" \
- -e "s|\$${MDBX_VERSION_REVISION}|$(shell git rev-list --count --no-merges HEAD || echo 'Please fetch tags and/or install latest git version')|" \
- src/elements/version.c.in > $@
+ -e "s|\$${MDBX_VERSION_REVISION}|$(MDBX_REVISION_GIT)|" \
+ src/version.c.in > $@
-src/elements/config.h: src/elements/version.c $(lastword $(MAKEFILE_LIST))
+src/config.h: src/version.c $(lastword $(MAKEFILE_LIST))
(echo '#define MDBX_BUILD_TIMESTAMP "$(shell date +%Y-%m-%dT%H:%M:%S%z)"' \
&& echo '#define MDBX_BUILD_FLAGS "$(CFLAGS) $(LDFLAGS) $(LIBS)"' \
&& echo '#define MDBX_BUILD_COMPILER "$(shell set -o pipefail; $(CC) --version | head -1 || echo 'Please use GCC or CLANG compatible compiler')"' \
@@ -203,63 +207,65 @@ src/elements/config.h: src/elements/version.c $(lastword $(MAKEFILE_LIST))
&& echo '#define MDBX_BUILD_SOURCERY $(MDBX_BUILD_SOURCERY)' \
) > $@
-mdbx-dylib.o: src/elements/config.h src/elements/version.c src/alloy.c $(ALLOY_DEPS) $(lastword $(MAKEFILE_LIST))
+mdbx-dylib.o: src/config.h src/version.c src/alloy.c $(ALLOY_DEPS) $(lastword $(MAKEFILE_LIST))
$(CC) $(CFLAGS) $(MDBX_OPTIONS) '-DMDBX_CONFIG_H="config.h"' -DLIBMDBX_EXPORTS=1 -c src/alloy.c -o $@
-mdbx-static.o: src/elements/config.h src/elements/version.c src/alloy.c $(ALLOY_DEPS) $(lastword $(MAKEFILE_LIST))
+mdbx-static.o: src/config.h src/version.c src/alloy.c $(ALLOY_DEPS) $(lastword $(MAKEFILE_LIST))
$(CC) $(CFLAGS) $(MDBX_OPTIONS) '-DMDBX_CONFIG_H="config.h"' -ULIBMDBX_EXPORTS -c src/alloy.c -o $@
.PHONY: dist
dist: libmdbx-sources-$(MDBX_VERSION_SUFFIX).tar.gz $(lastword $(MAKEFILE_LIST))
libmdbx-sources-$(MDBX_VERSION_SUFFIX).tar.gz: $(addprefix dist/, $(DIST_SRC) $(DIST_EXTRA)) $(addprefix dist/man1/,$(MANPAGES))
- tar -c --owner=0 --group=0 -C dist $(DIST_SRC) $(DIST_EXTRA) -f - | gzip -c > $@ \
+ $(TAR) -c $(shell LC_ALL=C $(TAR) --help | grep -q -- '--owner' && echo '--owner=0 --group=0') -f - -C dist $(DIST_SRC) $(DIST_EXTRA) | gzip -c > $@ \
&& rm dist/@tmp-shared_internals.inc
-dist/mdbx.h: mdbx.h src/elements/version.c $(lastword $(MAKEFILE_LIST))
+dist/mdbx.h: mdbx.h src/version.c $(lastword $(MAKEFILE_LIST))
mkdir -p dist && cp $< $@
-dist/Makefile: Makefile
- mkdir -p dist && cp $< $@
-
-dist/GNUmakefile: GNUmakefile
- mkdir -p dist && sed -e '/^#> dist-cutoff-begin/,/^#< dist-cutoff-end/d' $< > $@
-
-dist/@tmp-shared_internals.inc: src/elements/version.c $(ALLOY_DEPS) $(lastword $(MAKEFILE_LIST))
+dist/@tmp-shared_internals.inc: src/version.c $(ALLOY_DEPS) $(lastword $(MAKEFILE_LIST))
mkdir -p dist && sed \
-e 's|#pragma once|#define MDBX_ALLOY 1\n#define MDBX_BUILD_SOURCERY $(MDBX_BUILD_SOURCERY)|' \
- -e 's|#include "../../mdbx.h"|@INCLUDE "mdbx.h"|' \
- -e '/#include "defs.h"/r src/elements/defs.h' \
- -e '/#include "osal.h"/r src/elements/osal.h' \
- -e '/#include "options.h"/r src/elements/options.h' \
- src/elements/internals.h > $@
+ -e 's|#include "../mdbx.h"|@INCLUDE "mdbx.h"|' \
+ -e '/#include "defs.h"/r src/defs.h' \
+ -e '/#include "osal.h"/r src/osal.h' \
+ -e '/#include "options.h"/r src/options.h' \
+ src/internals.h > $@
dist/mdbx.c: dist/@tmp-shared_internals.inc $(lastword $(MAKEFILE_LIST))
mkdir -p dist && (cat dist/@tmp-shared_internals.inc \
- && cat src/elements/core.c src/elements/osal.c src/elements/version.c src/elements/lck-windows.c src/elements/lck-posix.c \
+ && cat src/core.c src/osal.c src/version.c src/lck-windows.c src/lck-posix.c \
) | grep -v -e '#include "' -e '#pragma once' | sed 's|@INCLUDE|#include|' > $@
define dist-tool-rule
-dist/$(1).c: src/tools/$(1).c src/tools/wingetopt.h src/tools/wingetopt.c \
+dist/$(1).c: src/$(1).c src/wingetopt.h src/wingetopt.c \
dist/@tmp-shared_internals.inc $(lastword $(MAKEFILE_LIST))
mkdir -p dist && sed \
- -e '/#include "..\/elements\/internals.h"/r dist/@tmp-shared_internals.inc' \
- -e '/#include "wingetopt.h"/r src/tools/wingetopt.c' \
- src/tools/$(1).c \
+ -e '/#include "internals.h"/r dist/@tmp-shared_internals.inc' \
+ -e '/#include "wingetopt.h"/r src/wingetopt.c' \
+ src/$(1).c \
| grep -v -e '#include "' -e '#pragma once' -e '#define MDBX_ALLOY' \
| sed 's|@INCLUDE|#include|' > $$@
endef
$(foreach file,$(TOOLS),$(eval $(call dist-tool-rule,$(file))))
+define dist-extra-rule
+dist/$(1): $(1)
+ mkdir -p $$(dir $$@) && sed -e '/^#> dist-cutoff-begin/,/^#< dist-cutoff-end/d' $$< > $$@
+
+endef
+$(foreach file,$(filter-out man1/% VERSION %.in,$(DIST_EXTRA)),$(eval $(call dist-extra-rule,$(file))))
+
+dist/VERSION: src/version.c
+ mkdir -p dist/ && echo "$(MDBX_VERSION_GIT).$(MDBX_REVISION_GIT)" > $@
+
+dist/config.h.in: src/config.h.in
+ mkdir -p dist/cmake/ && cp $< $@
+
dist/man1/mdbx_%.1: src/man1/mdbx_%.1
mkdir -p dist/man1/ && cp $< $@
-dist/LICENSE: LICENSE
- mkdir -p dist/man1/ && cp $< $@
-dist/README.md: README.md
- mkdir -p dist/man1/ && cp $< $@
-dist/CMakeLists.txt: CMakeLists.dist-minimal
- mkdir -p dist/man1/ && cp $< $@
+
endif
################################################################################
@@ -302,14 +308,17 @@ cross-qemu:
#< dist-cutoff-end
install: $(LIBRARIES) $(TOOLS) $(HEADERS)
- mkdir -p $(SANDBOX)$(prefix)/bin$(suffix) \
- && cp -t $(SANDBOX)$(prefix)/bin$(suffix) $(TOOLS) && \
- mkdir -p $(SANDBOX)$(prefix)/lib$(suffix) \
- && cp -t $(SANDBOX)$(prefix)/lib$(suffix) $(LIBRARIES) && \
- mkdir -p $(SANDBOX)$(prefix)/include \
- && cp -t $(SANDBOX)$(prefix)/include $(HEADERS) && \
- mkdir -p $(SANDBOX)$(mandir)/man1 \
- && cp -t $(SANDBOX)$(mandir)/man1 $(addprefix $(MAN_SRCDIR), $(MANPAGES))
+ install -D -p -s -t $(DESTDIR)$(prefix)/bin$(suffix) $(TOOLS) && \
+ install -D -p -s -t $(DESTDIR)$(prefix)/lib$(suffix) $(filter-out libmdbx.a,$(LIBRARIES)) && \
+ install -D -p -t $(DESTDIR)$(prefix)/lib$(suffix) libmdbx.a && \
+ install -D -p -m 444 -t $(DESTDIR)$(prefix)/include $(HEADERS) && \
+ install -D -p -m 444 -t $(DESTDIR)$(mandir)/man1 $(addprefix $(MAN_SRCDIR), $(MANPAGES))
+
+uninstall:
+ rm -f $(addprefix $(DESTDIR)$(prefix)/bin$(suffix)/,$(TOOLS)) \
+ $(addprefix $(DESTDIR)$(prefix)/lib$(suffix)/,$(LIBRARIES)) \
+ $(addprefix $(DESTDIR)$(prefix)/include/,$(HEADERS)) \
+ $(addprefix $(DESTDIR)$(mandir)/man1/,$(MANPAGES))
################################################################################
# Benchmarking by ioarena
diff --git a/libs/libmdbx/src/README.md b/libs/libmdbx/src/README.md
index 4cbea049c2..9685648bb4 100644
--- a/libs/libmdbx/src/README.md
+++ b/libs/libmdbx/src/README.md
@@ -9,15 +9,25 @@ database, with [permissive license](LICENSE).
_MDBX_ has a specific set of properties and capabilities,
focused on creating unique lightweight solutions with extraordinary performance.
-1. Allows **swarm of multi-threaded processes to [ACID]((https://en.wikipedia.org/wiki/ACID))ly read and update** several key-value [maps](https://en.wikipedia.org/wiki/Associative_array) and [multimaps](https://en.wikipedia.org/wiki/Multimap) in a localy-shared database.
-
-2. Provides **extraordinary performance**, minimal overhead through [Memory-Mapping](https://en.wikipedia.org/wiki/Memory-mapped_file) and `Olog(N)` operations costs by virtue of [B+ tree](https://en.wikipedia.org/wiki/B%2B_tree).
-
-3. Requires **no maintenance and no crash recovery** since doesn't use [WAL](https://en.wikipedia.org/wiki/Write-ahead_logging), but that might be a caveat for write-intensive workloads.
-
-4. **Compact and friendly for fully embeddeding**. Only 25KLOC of `C11`, 64K x86 binary code,
-no internal threads neither processes, but implements a simplified variant of the
-[Berkeley DB](https://en.wikipedia.org/wiki/Berkeley_DB) and
+1. Allows **swarm of multi-threaded processes to
+[ACID]((https://en.wikipedia.org/wiki/ACID))ly read and update** several
+key-value [maps](https://en.wikipedia.org/wiki/Associative_array) and
+[multimaps](https://en.wikipedia.org/wiki/Multimap) in a localy-shared
+database.
+
+2. Provides **extraordinary performance**, minimal overhead through
+[Memory-Mapping](https://en.wikipedia.org/wiki/Memory-mapped_file) and
+`Olog(N)` operations costs by virtue of [B+
+tree](https://en.wikipedia.org/wiki/B%2B_tree).
+
+3. Requires **no maintenance and no crash recovery** since doesn't use
+[WAL](https://en.wikipedia.org/wiki/Write-ahead_logging), but that might
+be a caveat for write-intensive workloads with durability requirements.
+
+4. **Compact and friendly for fully embeddeding**. Only 25KLOC of `C11`,
+64K x86 binary code, no internal threads neither processes, but
+implements a simplified variant of the [Berkeley
+DB](https://en.wikipedia.org/wiki/Berkeley_DB) and
[dbm](https://en.wikipedia.org/wiki/DBM_(computing)) API.
5. Enforces [serializability](https://en.wikipedia.org/wiki/Serializability) for
@@ -30,7 +40,7 @@ for parallel readers without atomic/interlocked operations, while
6. **Guarantee data integrity** after crash unless this was explicitly
neglected in favour of write performance.
-7. Supports Linux, Windows, MacOS, FreeBSD, DragonFly, Solaris,
+7. Supports Linux, Windows, MacOS, Android, iOS, FreeBSD, DragonFly, Solaris,
OpenSolaris, OpenIndiana, NetBSD, OpenBSD and other systems compliant with
**POSIX.1-2008**.
@@ -49,6 +59,7 @@ _MithrilDB_ is rightly relevant name.
> revolution is to provide a clearer and robust API, add more features and
> new valuable properties of database.
+[![https://t.me/libmdbx](https://raw.githubusercontent.com/wiki/erthink/libmdbx/img/telegram.png)](https://t.me/libmdbx)
[![Build Status](https://travis-ci.org/erthink/libmdbx.svg?branch=master)](https://travis-ci.org/erthink/libmdbx)
[![Build status](https://ci.appveyor.com/api/projects/status/ue94mlopn50dqiqg/branch/master?svg=true)](https://ci.appveyor.com/project/erthink/libmdbx/branch/master)
[![Coverity Scan Status](https://scan.coverity.com/projects/12915/badge.svg)](https://scan.coverity.com/projects/reopen-libmdbx)
@@ -185,6 +196,9 @@ the user's point of view.
> _libmdbx_ manage the database size according to parameters specified
> by `mdbx_env_set_geometry()` function,
> ones include the growth step and the truncation threshold.
+ >
+ > Unfortunately, on-the-fly database size adjustment doesn't work under [Wine](https://en.wikipedia.org/wiki/Wine_(software))
+ > due to its internal limitations and unimplemented functions, i.e. the `MDBX_UNABLE_EXTEND_MAPSIZE` error will be returned.
4. Automatic continuous zero-overhead database compactification.
> During each commit _libmdbx_ merges suitable freeing pages into unallocated area
@@ -402,16 +416,34 @@ will need to install the current (not outdated) version of
recommend that you install [Homebrew](https://brew.sh/) and then execute
`brew install bash`.
+### Android
+We recommend using CMake to build _libmdbx_ for Android.
+Please refer to the [official guide](https://developer.android.com/studio/projects/add-native-code).
+
+### iOS
+To build _libmdbx_ for iOS, we recommend using CMake with the
+"[toolchain file](https://cmake.org/cmake/help/latest/variable/CMAKE_TOOLCHAIN_FILE.html)"
+from the [ios-cmake](https://github.com/leetal/ios-cmake) project.
+
+### Windows Subsystem for Linux
+_libmdbx_ could be using in [WSL2](https://en.wikipedia.org/wiki/Windows_Subsystem_for_Linux#WSL_2)
+but NOT in [WSL1](https://en.wikipedia.org/wiki/Windows_Subsystem_for_Linux#WSL_1) environment.
+This is a consequence of the fundamental shortcomings of _WSL1_ and cannot be fixed.
+To avoid data loss, _libmdbx_ returns the `ENOLCK` (37, "No record locks available")
+error when opening the database in a _WSL1_ environment.
+
## API description
For more information and API description see the [mdbx.h](mdbx.h) header.
+Please do not hesitate to point out errors in the documentation,
+including creating [PR](https://help.github.com/en/github/collaborating-with-issues-and-pull-requests/proposing-changes-to-your-work-with-pull-requests) with corrections and improvements.
## Bindings
| Runtime | GitHub | Author |
| -------- | ------ | ------ |
+ | Rust | [mdbx-rs](https://github.com/Kerollmops/mdbx-rs) | [@Kerollmops](https://github.com/Kerollmops) |
| Java | [mdbxjni](https://github.com/castortech/mdbxjni) | [Castor Technologies](https://castortech.com/) |
| .NET | [mdbx.NET](https://github.com/wangjia184/mdbx.NET) | [Jerry Wang](https://github.com/wangjia184) |
- | Rust | [mdbx-rs](https://github.com/Kerollmops/mdbx-rs) | [Clément Renault](https://github.com/Kerollmops) |
--------------------------------------------------------------------------------
diff --git a/libs/libmdbx/src/appveyor.yml b/libs/libmdbx/src/appveyor.yml
index c1b1085cd7..efdeae648d 100644
--- a/libs/libmdbx/src/appveyor.yml
+++ b/libs/libmdbx/src/appveyor.yml
@@ -1,4 +1,4 @@
-version: 0.6.0.{build}
+version: 0.7.0.{build}
environment:
matrix:
diff --git a/libs/libmdbx/src/cmake/compiler.cmake b/libs/libmdbx/src/cmake/compiler.cmake
index 93c2c9f18b..a3bca1ac29 100644
--- a/libs/libmdbx/src/cmake/compiler.cmake
+++ b/libs/libmdbx/src/cmake/compiler.cmake
@@ -17,17 +17,15 @@ cmake_minimum_required(VERSION 3.8.2)
cmake_policy(PUSH)
cmake_policy(VERSION 3.8.2)
-if (CMAKE_VERSION MATCHES ".*MSVC.*")
+if(CMAKE_VERSION MATCHES ".*MSVC.*" AND CMAKE_VERSION VERSION_LESS 3.16)
message(FATAL_ERROR "CMake from MSVC kit is unfit! "
- "Please use the original CMake from https://cmake.org/download/")
+ "Please use MSVC2019 with modern CMake the original CMake from https://cmake.org/download/")
endif()
-if (NOT (CMAKE_C_COMPILER_LOADED OR CMAKE_CXX_COMPILER_LOADED))
+if(NOT (CMAKE_C_COMPILER_LOADED OR CMAKE_CXX_COMPILER_LOADED))
message(FATAL_ERROR "This module required C or C++ to be enabled")
endif()
-include(CMakeDependentOption)
-
if(CMAKE_CXX_COMPILER_LOADED)
include(CheckCXXSourceRuns)
include(CheckCXXSourceCompiles)
@@ -38,6 +36,9 @@ if(CMAKE_C_COMPILER_LOADED)
include(CheckCSourceCompiles)
include(CheckCCompilerFlag)
endif()
+include(CMakeDependentOption)
+include(CheckLibraryExists)
+include(CheckIncludeFiles)
# Check if the same compile family is used for both C and CXX
if(CMAKE_C_COMPILER_LOADED AND CMAKE_CXX_COMPILER_LOADED AND
@@ -150,6 +151,16 @@ if(WIN32 AND CMAKE_COMPILER_IS_GNU${CMAKE_PRIMARY_LANG})
unset(__GCC_TARGET_MACHINE)
endif()
+if(NOT DEFINED IOS)
+ if(APPLE AND (CMAKE_SYSTEM_NAME STREQUAL "iOS"
+ OR DEFINED CMAKE_IOS_DEVELOPER_ROOT
+ OR DEFINED IOS_PLATFORM OR DEFINED IOS_ARCH))
+ set(IOS TRUE)
+ else()
+ set(IOS FALSE)
+ endif()
+endif()
+
if(CMAKE_COMPILER_IS_ELBRUSC OR CMAKE_SYSTEM_PROCESSOR MATCHES "e2k.*|E2K.*|elbrus.*|ELBRUS.*")
set(E2K TRUE)
set(CMAKE_SYSTEM_ARCH "Elbrus")
@@ -214,6 +225,15 @@ else()
check_compiler_flag("-Ominimal" CC_HAS_OMINIMAL)
check_compiler_flag("-ffunction-sections -fdata-sections" CC_HAS_SECTIONS)
check_compiler_flag("-ffast-math" CC_HAS_FASTMATH)
+ check_compiler_flag("-Wno-attributes" CC_HAS_WNO_ATTRIBUTES)
+
+ # Check for an omp support
+ set(CMAKE_REQUIRED_FLAGS "-fopenmp -Werror")
+ check_cxx_source_compiles("int main(void) {
+ #pragma omp parallel
+ return 0;
+ }" HAVE_OPENMP)
+ set(CMAKE_REQUIRED_FLAGS "")
endif()
# Check for LTO support by GCC
@@ -276,7 +296,7 @@ if(CMAKE_COMPILER_IS_GNU${CMAKE_PRIMARY_LANG})
endif()
# check for LTO by MSVC
-if(MSVC)
+if(MSVC AND NOT CMAKE_COMPILER_IS_CLANG)
if(NOT MSVC_VERSION LESS 1600)
set(MSVC_LTO_AVAILABLE TRUE)
message(STATUS "Link-Time Optimization by MSVC is available")
@@ -290,25 +310,37 @@ endif()
if(CMAKE_COMPILER_IS_CLANG)
if(NOT CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 3.5)
execute_process(COMMAND ${CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER} -print-search-dirs
- OUTPUT_VARIABLE clang_search_dirs)
+ OUTPUT_VARIABLE clang_search_dirs RESULT_VARIABLE clang_probe_result ERROR_QUIET)
unset(clang_bindir)
unset(clang_libdir)
- string(REGEX MATCH "^(.*programs: =)([^:]*:)*([^:]+/llvm[-.0-9]+/bin[^:]*)(:[^:]*)*(\n.+)$" clang_bindir_valid ${clang_search_dirs})
- if(clang_bindir_valid)
- string(REGEX REPLACE "^(.*programs: =)([^:]*:)*([^:]+/llvm[-.0-9]+/bin[^:]*)(:[^:]*)*(\n.+)$" "\\3" clang_bindir ${clang_search_dirs})
- get_filename_component(clang_libdir "${clang_bindir}/../lib" REALPATH)
- if(clang_libdir)
- message(STATUS "Found CLANG/LLVM directories: ${clang_bindir}, ${clang_libdir}")
+ if(clang_probe_result EQUAL 0)
+ string(REGEX MATCH "^(.*programs: =)([^:]*:)*([^:]+/llvm[-.0-9]+/bin[^:]*)(:[^:]*)*(\n.+)$" clang_bindir_valid ${clang_search_dirs})
+ if(clang_bindir_valid)
+ string(REGEX REPLACE "^(.*programs: =)([^:]*:)*([^:]+/llvm[-.0-9]+/bin[^:]*)(:[^:]*)*(\n.+)$" "\\3" clang_bindir ${clang_search_dirs})
+ if(CMAKE_SYSTEM_NAME STREQUAL "WINDOWS")
+ set(clang_libdir ${clang_bindir})
+ else()
+ get_filename_component(clang_libdir "${clang_bindir}/../lib" REALPATH)
+ endif()
+ endif()
+ else()
+ get_filename_component(clang_bindir ${CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER} DIRECTORY)
+ if(CMAKE_HOST_SYSTEM_NAME STREQUAL "Windows")
+ set(clang_libdir ${clang_bindir})
+ else()
+ get_filename_component(clang_libdir "${clang_bindir}/../lib" REALPATH)
endif()
endif()
- if(NOT (clang_bindir AND clang_libdir))
+ if(clang_bindir AND clang_libdir)
+ message(STATUS "Found CLANG/LLVM directories: ${clang_bindir}, ${clang_libdir}")
+ else()
message(STATUS "Could NOT find CLANG/LLVM directories (bin and/or lib).")
endif()
if(NOT CMAKE_CLANG_LD AND clang_bindir)
- find_program(CMAKE_CLANG_LD NAMES llvm-link link llvm-ld ld PATHS ${clang_bindir} NO_DEFAULT_PATH)
+ find_program(CMAKE_CLANG_LD NAMES lld-link ld.lld ld64.lld lld-link llvm-link link llvm-ld ld PATHS ${clang_bindir} NO_DEFAULT_PATH)
endif()
if(NOT CMAKE_CLANG_AR AND clang_bindir)
find_program(CMAKE_CLANG_AR NAMES llvm-ar ar PATHS ${clang_bindir} NO_DEFAULT_PATH)
@@ -346,19 +378,19 @@ if(CMAKE_COMPILER_IS_CLANG)
unset(clang_search_dirs)
endif()
- if((CLANG_LTO_PLUGIN AND CMAKE_LD_GOLD) AND
+ if((CLANG_LTO_PLUGIN AND CMAKE_LD_GOLD) OR
(CMAKE_CLANG_LD AND CMAKE_CLANG_AR AND CMAKE_CLANG_NM AND CMAKE_CLANG_RANLIB))
set(CLANG_LTO_AVAILABLE TRUE)
message(STATUS "Link-Time Optimization by CLANG/LLVM is available")
elseif(CMAKE_TOOLCHAIN_FILE AND NOT CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 7.0)
set(CLANG_LTO_AVAILABLE TRUE)
- if (NOT CMAKE_CLANG_AR)
+ if(NOT CMAKE_CLANG_AR)
set(CMAKE_CLANG_AR ${CMAKE_AR})
endif()
- if (NOT CMAKE_CLANG_NM)
+ if(NOT CMAKE_CLANG_NM)
set(CMAKE_CLANG_NM ${CMAKE_NM})
endif()
- if (NOT CMAKE_CLANG_RANLIB)
+ if(NOT CMAKE_CLANG_RANLIB)
set(CMAKE_CLANG_RANLIB ${CMAKE_RANLIB})
endif()
message(STATUS "Assume Link-Time Optimization by CLANG/LLVM is available via ${CMAKE_TOOLCHAIN_FILE}")
@@ -408,7 +440,7 @@ if(ENABLE_BACKTRACE)
endif()
macro(setup_compile_flags)
- # LY: save initial C/CXX flags
+ # save initial C/CXX flags
if(NOT INITIAL_CMAKE_FLAGS_SAVED)
if(MSVC)
string(REGEX REPLACE "^(.*)(/EHsc)( *)(.*)$" "\\1/EHs\\3\\4" CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS}")
@@ -422,7 +454,7 @@ macro(setup_compile_flags)
set(INITIAL_CMAKE_FLAGS_SAVED TRUE CACHE INTERNAL "State of initial CMake's flags" FORCE)
endif()
- # LY: reset C/CXX flags
+ # reset C/CXX flags
set(CXX_FLAGS ${INITIAL_CMAKE_CXX_FLAGS})
set(C_FLAGS ${INITIAL_CMAKE_C_FLAGS})
set(EXE_LINKER_FLAGS ${INITIAL_CMAKE_EXE_LINKER_FLAGS})
@@ -437,6 +469,12 @@ macro(setup_compile_flags)
add_compile_flags("CXX" "-fcxx-exceptions -frtti")
endif()
+ if(CC_HAS_WNO_ATTRIBUTES AND CMAKE_COMPILER_IS_GNU${CMAKE_PRIMARY_LANG}
+ AND CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 9)
+ # GCC < 9.x generates false-positive warnings for optimization attributes
+ add_compile_flags("C;CXX" "-Wno-attributes")
+ endif()
+
# In C a global variable without a storage specifier (static/extern) and
# without an initialiser is called a ’tentative definition’. The
# language permits multiple tentative definitions in the single
@@ -482,7 +520,7 @@ macro(setup_compile_flags)
endif()
if(MSVC)
- if (MSVC_VERSION LESS 1900)
+ if(MSVC_VERSION LESS 1900)
message(FATAL_ERROR "At least \"Microsoft C/C++ Compiler\" version 19.0.24234.1 (Visual Studio 2015 Update 3) is required.")
endif()
add_compile_flags("CXX" "/Zc:__cplusplus")
@@ -514,11 +552,11 @@ macro(setup_compile_flags)
if(MSVC)
add_compile_flags("C;CXX" "/WX")
elseif(CMAKE_COMPILER_IS_CLANG)
- if (NOT CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 6)
+ if(NOT CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 6)
add_compile_flags("C;CXX" "-Werror")
endif()
elseif(CMAKE_COMPILER_IS_GNUCC)
- if (NOT CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 6)
+ if(NOT CMAKE_${CMAKE_PRIMARY_LANG}_COMPILER_VERSION VERSION_LESS 6)
add_compile_flags("C;CXX" "-Werror")
endif()
else()
@@ -526,7 +564,7 @@ macro(setup_compile_flags)
endif()
endif()
- if (ENABLE_ASAN)
+ if(ENABLE_ASAN)
add_compile_flags("C;CXX" -fsanitize=address)
endif()
@@ -564,7 +602,7 @@ macro(setup_compile_flags)
endif()
endif()
- if(MSVC AND LTO_ENABLED)
+ if(MSVC AND NOT CMAKE_COMPILER_IS_CLANG AND LTO_ENABLED)
add_compile_flags("C;CXX" "/GL")
foreach(linkmode IN ITEMS EXE SHARED STATIC MODULE)
set(${linkmode}_LINKER_FLAGS "${${linkmode}_LINKER_FLAGS} /LTCG")
@@ -617,12 +655,14 @@ macro(setup_compile_flags)
set(CLANG_LTO_FLAG "-flto=thin")
endif()
add_compile_flags("C;CXX" ${CLANG_LTO_FLAG})
- set(EXE_LINKER_FLAGS "${EXE_LINKER_FLAGS} ${CLANG_LTO_FLAG} -fverbose-asm -fwhole-program")
- set(SHARED_LINKER_FLAGS "${SHARED_LINKER_FLAGS} ${CLANG_LTO_FLAG} -fverbose-asm")
- set(MODULE_LINKER_FLAGS "${MODULE_LINKER_FLAGS} ${CLANG_LTO_FLAG} -fverbose-asm")
+ if(NOT MSVC)
+ set(EXE_LINKER_FLAGS "${EXE_LINKER_FLAGS} ${CLANG_LTO_FLAG} -fverbose-asm -fwhole-program")
+ set(SHARED_LINKER_FLAGS "${SHARED_LINKER_FLAGS} ${CLANG_LTO_FLAG} -fverbose-asm")
+ set(MODULE_LINKER_FLAGS "${MODULE_LINKER_FLAGS} ${CLANG_LTO_FLAG} -fverbose-asm")
+ endif()
endif()
- # LY: push C/CXX flags into the cache
+ # push C/CXX flags into the cache
set(CMAKE_CXX_FLAGS ${CXX_FLAGS} CACHE STRING "Flags used by the C++ compiler during all build types" FORCE)
set(CMAKE_C_FLAGS ${C_FLAGS} CACHE STRING "Flags used by the C compiler during all build types" FORCE)
set(CMAKE_EXE_LINKER_FLAGS ${EXE_LINKER_FLAGS} CACHE STRING "Flags used by the linker" FORCE)
@@ -640,11 +680,11 @@ endmacro(setup_compile_flags)
# determine library for for std::filesystem
set(LIBCXX_FILESYSTEM "")
if(CMAKE_COMPILER_IS_GNUCXX)
- if (CMAKE_CXX_COMPILER_VERSION VERSION_LESS 9.0)
+ if(CMAKE_CXX_COMPILER_VERSION VERSION_LESS 9.0)
set(LIBCXX_FILESYSTEM "stdc++fs")
endif()
-elseif (CMAKE_COMPILER_IS_CLANG)
- if (CMAKE_CXX_COMPILER_VERSION VERSION_LESS 5.0)
+elseif(CMAKE_COMPILER_IS_CLANG)
+ if(CMAKE_CXX_COMPILER_VERSION VERSION_LESS 5.0)
set(LIBCXX_FILESYSTEM "c++experimental")
else()
set(LIBCXX_FILESYSTEM "stdc++fs")
diff --git a/libs/libmdbx/src/cmake/utils.cmake b/libs/libmdbx/src/cmake/utils.cmake
index 22e0d72f57..13add097dc 100644
--- a/libs/libmdbx/src/cmake/utils.cmake
+++ b/libs/libmdbx/src/cmake/utils.cmake
@@ -61,7 +61,7 @@ macro(set_source_files_compile_flags)
unset(_lang)
endmacro(set_source_files_compile_flags)
-macro(fetch_version name version_file)
+macro(fetch_version name version_file parent_scope)
set(${name}_VERSION "")
set(${name}_GIT_DESCRIBE "")
set(${name}_GIT_TIMESTAMP "")
@@ -69,7 +69,7 @@ macro(fetch_version name version_file)
set(${name}_GIT_COMMIT "")
set(${name}_GIT_REVISION 0)
set(${name}_GIT_VERSION "")
- if(GIT)
+ if(GIT AND EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/.git")
execute_process(COMMAND ${GIT} describe --tags --long --dirty=-dirty
OUTPUT_VARIABLE ${name}_GIT_DESCRIBE
OUTPUT_STRIP_TRAILING_WHITESPACE
@@ -137,7 +137,9 @@ macro(fetch_version name version_file)
endif()
if(NOT ${name}_GIT_VERSION OR NOT ${name}_GIT_TIMESTAMP OR NOT ${name}_GIT_REVISION)
- message(WARNING "Unable to retrive ${name} version from git.")
+ if(GIT AND EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/.git")
+ message(WARNING "Unable to retrive ${name} version from git.")
+ endif()
set(${name}_GIT_VERSION "0;0;0;0")
set(${name}_GIT_TIMESTAMP "")
set(${name}_GIT_REVISION 0)
@@ -166,18 +168,33 @@ macro(fetch_version name version_file)
list(GET ${name}_VERSION_LIST 2 "${name}_VERSION_RELEASE")
list(GET ${name}_VERSION_LIST 3 "${name}_VERSION_REVISION")
- set(${name}_VERSION_MAJOR ${${name}_VERSION_MAJOR} PARENT_SCOPE)
- set(${name}_VERSION_MINOR ${${name}_VERSION_MINOR} PARENT_SCOPE)
- set(${name}_VERSION_RELEASE ${${name}_VERSION_RELEASE} PARENT_SCOPE)
- set(${name}_VERSION_REVISION ${${name}_VERSION_REVISION} PARENT_SCOPE)
- set(${name}_VERSION ${${name}_VERSION} PARENT_SCOPE)
-
- set(${name}_GIT_DESCRIBE ${${name}_GIT_DESCRIBE} PARENT_SCOPE)
- set(${name}_GIT_TIMESTAMP ${${name}_GIT_TIMESTAMP} PARENT_SCOPE)
- set(${name}_GIT_TREE ${${name}_GIT_TREE} PARENT_SCOPE)
- set(${name}_GIT_COMMIT ${${name}_GIT_COMMIT} PARENT_SCOPE)
- set(${name}_GIT_REVISION ${${name}_GIT_REVISION} PARENT_SCOPE)
- set(${name}_GIT_VERSION ${${name}_GIT_VERSION} PARENT_SCOPE)
+ if(parent_scope)
+ set(${name}_VERSION_MAJOR ${${name}_VERSION_MAJOR} PARENT_SCOPE)
+ set(${name}_VERSION_MINOR ${${name}_VERSION_MINOR} PARENT_SCOPE)
+ set(${name}_VERSION_RELEASE ${${name}_VERSION_RELEASE} PARENT_SCOPE)
+ set(${name}_VERSION_REVISION ${${name}_VERSION_REVISION} PARENT_SCOPE)
+ set(${name}_VERSION ${${name}_VERSION} PARENT_SCOPE)
+
+ set(${name}_GIT_DESCRIBE ${${name}_GIT_DESCRIBE} PARENT_SCOPE)
+ set(${name}_GIT_TIMESTAMP ${${name}_GIT_TIMESTAMP} PARENT_SCOPE)
+ set(${name}_GIT_TREE ${${name}_GIT_TREE} PARENT_SCOPE)
+ set(${name}_GIT_COMMIT ${${name}_GIT_COMMIT} PARENT_SCOPE)
+ set(${name}_GIT_REVISION ${${name}_GIT_REVISION} PARENT_SCOPE)
+ set(${name}_GIT_VERSION ${${name}_GIT_VERSION} PARENT_SCOPE)
+ else()
+ set(${name}_VERSION_MAJOR ${${name}_VERSION_MAJOR})
+ set(${name}_VERSION_MINOR ${${name}_VERSION_MINOR})
+ set(${name}_VERSION_RELEASE ${${name}_VERSION_RELEASE})
+ set(${name}_VERSION_REVISION ${${name}_VERSION_REVISION})
+ set(${name}_VERSION ${${name}_VERSION})
+
+ set(${name}_GIT_DESCRIBE ${${name}_GIT_DESCRIBE})
+ set(${name}_GIT_TIMESTAMP ${${name}_GIT_TIMESTAMP})
+ set(${name}_GIT_TREE ${${name}_GIT_TREE})
+ set(${name}_GIT_COMMIT ${${name}_GIT_COMMIT})
+ set(${name}_GIT_REVISION ${${name}_GIT_REVISION})
+ set(${name}_GIT_VERSION ${${name}_GIT_VERSION})
+ endif()
endmacro(fetch_version)
cmake_policy(POP)
diff --git a/libs/libmdbx/src/src/elements/config.h b/libs/libmdbx/src/config.h
index f05ba6fc85..cc79caf8ed 100644
--- a/libs/libmdbx/src/src/elements/config.h
+++ b/libs/libmdbx/src/config.h
@@ -4,9 +4,6 @@
/* *INDENT-OFF* */
/* clang-format off */
-/* #undef HAVE_VALGRIND_MEMCHECK_H */
-/* #undef HAS_RELAXED_CONSTEXPR */
-
#define LTO_ENABLED
/* #undef MDBX_USE_VALGRIND */
/* #undef ENABLE_GPROF */
@@ -20,7 +17,6 @@
#ifndef MDBX_TXN_CHECKPID_AUTO
#define MDBX_TXN_CHECKPID 0
#endif
-#define MDBX_BUILD_SHARED_LIBRARY 1
#define MDBX_LOCKING_AUTO
#ifndef MDBX_LOCKING_AUTO
/* #undef MDBX_LOCKING */
@@ -31,10 +27,10 @@
#endif
/* Windows */
-#define MDBX_CONFIG_MANUAL_TLS_CALLBACK 0
+#define MDBX_CONFIG_MANUAL_TLS_CALLBACK 1
#define MDBX_AVOID_CRT 1
-/* MacOS */
+/* MacOS & iOS */
#define MDBX_OSX_SPEED_INSTEADOF_DURABILITY 0
/* POSIX */
@@ -45,12 +41,12 @@
#endif
/* Build Info */
-#define MDBX_BUILD_TIMESTAMP "2020-03-20T14:47:36Z"
+#define MDBX_BUILD_TIMESTAMP "2020-04-25T08:31:34Z"
#define MDBX_BUILD_TARGET "x86_64-Windows"
-/* #undef MDBX_BUILD_CONFIG */
-#define MDBX_BUILD_COMPILER "MSVC-19.25.28610.4"
-#define MDBX_BUILD_FLAGS "LIBMDBX_EXPORTS"
-#define MDBX_BUILD_SOURCERY b8a401ae08d0312d6e5365c29bb58adea2e44dbc7e3d808df4a0de2fefcc65ca_v0_6_0_45_gc20ba1e
+/* #undef MDBX_BUILD_TYPE */
+#define MDBX_BUILD_COMPILER "MSVC-19.25.28614.0"
+#define MDBX_BUILD_FLAGS "/DWIN32 /D_WINDOWS /W3 /Gy /W4 /utf-8 /WX /GL MDBX_BUILD_SHARED_LIBRARY=0"
+#define MDBX_BUILD_SOURCERY 51d296e8c57b5c01204315d0ea2160b23cf3eb40509644354cc038e3f67d61d4_v0_7_0_39_gca8fa31
/* *INDENT-ON* */
/* clang-format on */
diff --git a/libs/libmdbx/src/mdbx.h b/libs/libmdbx/src/mdbx.h
index 971b009402..c7a6ab3f43 100644
--- a/libs/libmdbx/src/mdbx.h
+++ b/libs/libmdbx/src/mdbx.h
@@ -555,6 +555,11 @@
#include <windows.h>
#include <winnt.h>
+
+#ifndef FSCTL_GET_EXTERNAL_BACKING
+#define FSCTL_GET_EXTERNAL_BACKING CTL_CODE(FILE_DEVICE_FILE_SYSTEM, 196, METHOD_BUFFERED, FILE_ANY_ACCESS)
+#endif
+
#ifndef __mode_t_defined
typedef unsigned short mode_t;
#endif
@@ -653,8 +658,9 @@ typedef pthread_t mdbx_tid_t;
/*----------------------------------------------------------------------------*/
+/* MDBX version 0.7.0, released 2020-03-18 */
#define MDBX_VERSION_MAJOR 0
-#define MDBX_VERSION_MINOR 6
+#define MDBX_VERSION_MINOR 7
#ifndef LIBMDBX_API
#if defined(LIBMDBX_EXPORTS)
@@ -1202,10 +1208,11 @@ LIBMDBX_API const char *mdbx_dump_val(const MDBX_val *key, char *const buf,
* In case single transaction after mdbx_env_sync, you may lose
* transaction itself, but not a whole database.
*
- * Nevertheless, MDBX_UTTERLY_NOSYNC provides ACID in case of a application
- * crash, and therefore may be very useful in scenarios where data
- * durability is not required over a system failure (e.g for short-lived
- * data), or if you can ignore such risk.
+ * Nevertheless, MDBX_UTTERLY_NOSYNC provides "weak" durability in case of
+ * an application crash (but no durability on system failure), and
+ * therefore may be very useful in scenarios where data durability is not
+ * required over a system failure (e.g for short-lived data), or if you can
+ * take such risk.
*
* MDBX_UTTERLY_NOSYNC flag may be changed at any time using
* mdbx_env_set_flags(), but don't has effect if passed to mdbx_txn_begin()
@@ -1447,7 +1454,8 @@ typedef enum MDBX_cursor_op {
/* Transaction has too many dirty pages, i.e transaction too big */
#define MDBX_TXN_FULL (-30788)
-/* Cursor stack too deep - internal error */
+/* Cursor stack too deep - this usually indicates corruption,
+ * i.e branch-pages loop */
#define MDBX_CURSOR_FULL (-30787)
/* Page has not enough space - internal error */
@@ -1462,6 +1470,13 @@ typedef enum MDBX_cursor_op {
* or explicit call of mdbx_env_set_geometry(). */
#define MDBX_UNABLE_EXTEND_MAPSIZE (-30785)
+/* MDBX_MAP_RESIZED is deprecated.
+ * Please review your code to use MDBX_UNABLE_EXTEND_MAPSIZE instead. */
+static __inline int __deprecated MDBX_MAP_RESIZED() {
+ return MDBX_UNABLE_EXTEND_MAPSIZE;
+}
+#define MDBX_MAP_RESIZED MDBX_MAP_RESIZED()
+
/* Environment or database is not compatible with the requested operation
* or the specified flags. This can mean:
* - The operation expects an MDBX_DUPSORT / MDBX_DUPFIXED database.
@@ -1904,14 +1919,17 @@ LIBMDBX_API int mdbx_env_set_syncperiod(MDBX_env *env,
*
* Returns A non-zero error value on failure and 0 on success.
* Some possible errors are:
- * - MDBX_BUSY = The write transaction is running by other thread, in such
- * case MDBX_env instance has NOT be destroyed not released!
- * NOTE: if any OTHER error code was returned then given
- * MDBX_env instance has been destroyed and released.
- * - MDBX_PANIC = If mdbx_env_close_ex() was called in the child process
- * after fork(). In this case MDBX_PANIC is a expecte,
- * i.e. MDBX_env instance was freed in proper manner.
- * - MDBX_EIO = an error occurred during synchronization. */
+ * - MDBX_BUSY = The write transaction is running by other thread, in such
+ * case MDBX_env instance has NOT be destroyed not released!
+ * NOTE: if any OTHER error code was returned then given
+ * MDBX_env instance has been destroyed and released.
+ * - MDBX_EBADSIGN = Environment handle already closed (i.e. mdbx_env_close()
+ * was already called or not valid (i.e. was not created
+ * by mdbx_env_create()).
+ * - MDBX_PANIC = If mdbx_env_close_ex() was called in the child process
+ * after fork(). In this case MDBX_PANIC is a expecte,
+ * i.e. MDBX_env instance was freed in proper manner.
+ * - MDBX_EIO = an error occurred during synchronization. */
LIBMDBX_API int mdbx_env_close_ex(MDBX_env *env, int dont_sync);
LIBMDBX_API int mdbx_env_close(MDBX_env *env);
@@ -1942,7 +1960,7 @@ LIBMDBX_API int mdbx_env_set_flags(MDBX_env *env, unsigned flags, int onoff);
* Returns A non-zero error value on failure and 0 on success, some
* possible errors are:
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_env_get_flags(MDBX_env *env, unsigned *flags);
+LIBMDBX_API int mdbx_env_get_flags(const MDBX_env *env, unsigned *flags);
/* Return the path that was used in mdbx_env_open().
*
@@ -1954,7 +1972,7 @@ LIBMDBX_API int mdbx_env_get_flags(MDBX_env *env, unsigned *flags);
* Returns A non-zero error value on failure and 0 on success, some
* possible errors are:
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_env_get_path(MDBX_env *env, const char **dest);
+LIBMDBX_API int mdbx_env_get_path(const MDBX_env *env, const char **dest);
/* Return the file descriptor for the given environment.
*
@@ -1967,7 +1985,7 @@ LIBMDBX_API int mdbx_env_get_path(MDBX_env *env, const char **dest);
* Returns A non-zero error value on failure and 0 on success, some
* possible errors are:
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_env_get_fd(MDBX_env *env, mdbx_filehandle_t *fd);
+LIBMDBX_API int mdbx_env_get_fd(const MDBX_env *env, mdbx_filehandle_t *fd);
/* Set all size-related parameters of environment, including page size and the
* min/max size of the memory map.
@@ -2220,7 +2238,7 @@ LIBMDBX_API int mdbx_env_set_maxreaders(MDBX_env *env, unsigned readers);
* Returns A non-zero error value on failure and 0 on success, some
* possible errors are:
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_env_get_maxreaders(MDBX_env *env, unsigned *readers);
+LIBMDBX_API int mdbx_env_get_maxreaders(const MDBX_env *env, unsigned *readers);
/* Set the maximum number of named databases for the environment.
*
@@ -2251,9 +2269,9 @@ LIBMDBX_API int mdbx_env_set_maxdbs(MDBX_env *env, MDBX_dbi dbs);
*
* Returns The maximum size of a key we can write,
* or -1 if something is wrong. */
-LIBMDBX_API int mdbx_env_get_maxkeysize_ex(MDBX_env *env, unsigned flags);
-LIBMDBX_API int mdbx_env_get_maxvalsize_ex(MDBX_env *env, unsigned flags);
-__deprecated LIBMDBX_API int mdbx_env_get_maxkeysize(MDBX_env *env);
+LIBMDBX_API int mdbx_env_get_maxkeysize_ex(const MDBX_env *env, unsigned flags);
+LIBMDBX_API int mdbx_env_get_maxvalsize_ex(const MDBX_env *env, unsigned flags);
+__deprecated LIBMDBX_API int mdbx_env_get_maxkeysize(const MDBX_env *env);
/* Set application information associated with the MDBX_env.
*
@@ -2267,7 +2285,7 @@ LIBMDBX_API int mdbx_env_set_userctx(MDBX_env *env, void *ctx);
*
* [in] env An environment handle returned by mdbx_env_create()
* Returns The pointer set by mdbx_env_set_userctx(). */
-LIBMDBX_API void *mdbx_env_get_userctx(MDBX_env *env);
+LIBMDBX_API void *mdbx_env_get_userctx(const MDBX_env *env);
/* Create a transaction for use with the environment.
*
@@ -2380,12 +2398,13 @@ typedef struct MDBX_txn_info {
* not needed (see description of MDBX_txn_info above).
*
* Returns A non-zero error value on failure and 0 on success. */
-LIBMDBX_API int mdbx_txn_info(MDBX_txn *txn, MDBX_txn_info *info, int scan_rlt);
+LIBMDBX_API int mdbx_txn_info(const MDBX_txn *txn, MDBX_txn_info *info,
+ int scan_rlt);
/* Returns the transaction's MDBX_env.
*
* [in] txn A transaction handle returned by mdbx_txn_begin() */
-LIBMDBX_API MDBX_env *mdbx_txn_env(MDBX_txn *txn);
+LIBMDBX_API MDBX_env *mdbx_txn_env(const MDBX_txn *txn);
/* Return the transaction's flags.
*
@@ -2395,7 +2414,7 @@ LIBMDBX_API MDBX_env *mdbx_txn_env(MDBX_txn *txn);
*
* Returns A transaction flags, valid if input is an valid transaction,
* otherwise -1. */
-LIBMDBX_API int mdbx_txn_flags(MDBX_txn *txn);
+LIBMDBX_API int mdbx_txn_flags(const MDBX_txn *txn);
/* Return the transaction's ID.
*
@@ -2407,7 +2426,7 @@ LIBMDBX_API int mdbx_txn_flags(MDBX_txn *txn);
*
* Returns A transaction ID, valid if input is an active transaction,
* otherwise 0. */
-LIBMDBX_API uint64_t mdbx_txn_id(MDBX_txn *txn);
+LIBMDBX_API uint64_t mdbx_txn_id(const MDBX_txn *txn);
/* Commit all the operations of a transaction into the database.
*
@@ -2514,7 +2533,7 @@ LIBMDBX_API int mdbx_canary_put(MDBX_txn *txn, const mdbx_canary *canary);
* will be copied.
*
* Returns A non-zero error value on failure and 0 on success. */
-LIBMDBX_API int mdbx_canary_get(MDBX_txn *txn, mdbx_canary *canary);
+LIBMDBX_API int mdbx_canary_get(const MDBX_txn *txn, mdbx_canary *canary);
/* A callback function used to compare two keys in a database */
typedef int(MDBX_cmp_func)(const MDBX_val *a, const MDBX_val *b);
@@ -2589,9 +2608,14 @@ typedef int(MDBX_cmp_func)(const MDBX_val *a, const MDBX_val *b);
* comparison functions for keys and values (for multimaps).
* However, I recommend not using custom comparison functions, but instead
* converting the keys to one of the forms that are suitable for built-in
- * comparators. The main reason for this is that you can't use mdbx_chk tools
- * with a custom comparators. For instance take look to the mdbx_key_from_xxx()
- * functions.
+ * comparators (for instance take look to the mdbx_key_from_xxx()
+ * functions). The reasons to not using custom comparators are:
+ * - The order of records could not be validated without your code.
+ * So mdbx_chk utility will reports "wrong order" errors
+ * and the '-i' option is required to ignore ones.
+ * - A records could not be ordered or sorted without your code.
+ * So mdbx_load utility should be used with '-a' option to preserve
+ * input data order.
*
* [in] keycmp Optional custom key comparison function for a database.
* [in] datacmp Optional custom data comparison function for a database, takes
@@ -2725,7 +2749,7 @@ LIBMDBX_API int mdbx_drop(MDBX_txn *txn, MDBX_dbi dbi, int del);
* possible errors are:
* - MDBX_NOTFOUND = the key was not in the database.
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_get(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
+LIBMDBX_API int mdbx_get(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
MDBX_val *data);
/* Get items from a database and optionaly number of data items for a given key.
@@ -2837,7 +2861,7 @@ LIBMDBX_API int mdbx_get_nearest(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
* - MDBX_TXN_FULL = the transaction has too many dirty pages.
* - MDBX_EACCES = an attempt was made to write in a read-only transaction.
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_put(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
+LIBMDBX_API int mdbx_put(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
MDBX_val *data, unsigned flags);
/* Replace items in a database.
@@ -2876,7 +2900,7 @@ LIBMDBX_API int mdbx_put(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
* multi-value/duplicates.
*
* Returns A non-zero error value on failure and 0 on success. */
-LIBMDBX_API int mdbx_replace(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
+LIBMDBX_API int mdbx_replace(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
MDBX_val *new_data, MDBX_val *old_data,
unsigned flags);
@@ -2900,8 +2924,8 @@ LIBMDBX_API int mdbx_replace(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
* possible errors are:
* - MDBX_EACCES = an attempt was made to write in a read-only transaction.
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_del(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
- MDBX_val *data);
+LIBMDBX_API int mdbx_del(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
+ const MDBX_val *data);
/* Create a cursor handle.
*
@@ -2965,12 +2989,12 @@ LIBMDBX_API int mdbx_cursor_renew(MDBX_txn *txn, MDBX_cursor *cursor);
/* Return the cursor's transaction handle.
*
* [in] cursor A cursor handle returned by mdbx_cursor_open(). */
-LIBMDBX_API MDBX_txn *mdbx_cursor_txn(MDBX_cursor *cursor);
+LIBMDBX_API MDBX_txn *mdbx_cursor_txn(const MDBX_cursor *cursor);
/* Return the cursor's database handle.
*
* [in] cursor A cursor handle returned by mdbx_cursor_open(). */
-LIBMDBX_API MDBX_dbi mdbx_cursor_dbi(MDBX_cursor *cursor);
+LIBMDBX_API MDBX_dbi mdbx_cursor_dbi(const MDBX_cursor *cursor);
/* Retrieve by cursor.
*
@@ -2997,11 +3021,11 @@ LIBMDBX_API int mdbx_cursor_get(MDBX_cursor *cursor, MDBX_val *key,
* This function stores key/data pairs into the database. The cursor is
* positioned at the new item, or on failure usually near it.
*
- * [in] cursor A cursor handle returned by mdbx_cursor_open().
- * [in] key The key operated on.
- * [in] data The data operated on.
- * [in] flags Options for this operation. This parameter
- * must be set to 0 or one of the values described here:
+ * [in] cursor A cursor handle returned by mdbx_cursor_open().
+ * [in] key The key operated on.
+ * [in,out] data The data operated on.
+ * [in] flags Options for this operation. This parameter
+ * must be set to 0 or one of the values described here:
*
* - MDBX_CURRENT
* Replace the item at the current cursor position. The key parameter
@@ -3058,7 +3082,7 @@ LIBMDBX_API int mdbx_cursor_get(MDBX_cursor *cursor, MDBX_val *key,
* - MDBX_TXN_FULL = the transaction has too many dirty pages.
* - MDBX_EACCES = an attempt was made to write in a read-only transaction.
* - MDBX_EINVAL = an invalid parameter was specified. */
-LIBMDBX_API int mdbx_cursor_put(MDBX_cursor *cursor, MDBX_val *key,
+LIBMDBX_API int mdbx_cursor_put(MDBX_cursor *cursor, const MDBX_val *key,
MDBX_val *data, unsigned flags);
/* Delete current key/data pair.
@@ -3094,7 +3118,7 @@ LIBMDBX_API int mdbx_cursor_del(MDBX_cursor *cursor, unsigned flags);
* possible errors are:
* - MDBX_EINVAL = cursor is not initialized, or an invalid parameter
* was specified. */
-LIBMDBX_API int mdbx_cursor_count(MDBX_cursor *cursor, size_t *countp);
+LIBMDBX_API int mdbx_cursor_count(const MDBX_cursor *cursor, size_t *countp);
/* Determines whether the cursor is pointed to a key-value pair or not,
* i.e. was not positioned or points to the end of data.
@@ -3105,7 +3129,7 @@ LIBMDBX_API int mdbx_cursor_count(MDBX_cursor *cursor, size_t *countp);
* - MDBX_RESULT_TRUE = no more data available or cursor not positioned;
* - MDBX_RESULT_FALSE = data available;
* - Otherwise the error code. */
-LIBMDBX_API int mdbx_cursor_eof(MDBX_cursor *mc);
+LIBMDBX_API int mdbx_cursor_eof(const MDBX_cursor *mc);
/* Determines whether the cursor is pointed to the first key-value pair or not.
*
@@ -3115,7 +3139,7 @@ LIBMDBX_API int mdbx_cursor_eof(MDBX_cursor *mc);
* - MDBX_RESULT_TRUE = cursor positioned to the first key-value pair.
* - MDBX_RESULT_FALSE = cursor NOT positioned to the first key-value pair.
* - Otherwise the error code. */
-LIBMDBX_API int mdbx_cursor_on_first(MDBX_cursor *mc);
+LIBMDBX_API int mdbx_cursor_on_first(const MDBX_cursor *mc);
/* Determines whether the cursor is pointed to the last key-value pair or not.
*
@@ -3125,7 +3149,7 @@ LIBMDBX_API int mdbx_cursor_on_first(MDBX_cursor *mc);
* - MDBX_RESULT_TRUE = cursor positioned to the last key-value pair.
* - MDBX_RESULT_FALSE = cursor NOT positioned to the last key-value pair.
* - Otherwise the error code. */
-LIBMDBX_API int mdbx_cursor_on_last(MDBX_cursor *mc);
+LIBMDBX_API int mdbx_cursor_on_last(const MDBX_cursor *mc);
/* Estimates the distance between cursors as a number of elements. The results
* of such estimation can be used to build and/or optimize query execution
@@ -3280,7 +3304,7 @@ LIBMDBX_API int mdbx_dbi_sequence(MDBX_txn *txn, MDBX_dbi dbi, uint64_t *result,
* [in] b The second item to compare.
*
* Returns < 0 if a < b, 0 if a == b, > 0 if a > b */
-LIBMDBX_API int mdbx_cmp(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
+LIBMDBX_API int mdbx_cmp(const MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
const MDBX_val *b);
/* Compare two data items according to a particular database.
@@ -3294,7 +3318,7 @@ LIBMDBX_API int mdbx_cmp(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
* [in] b The second item to compare.
*
* Returns < 0 if a < b, 0 if a == b, > 0 if a > b */
-LIBMDBX_API int mdbx_dcmp(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
+LIBMDBX_API int mdbx_dcmp(const MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
const MDBX_val *b);
/* A callback function used to enumerate the reader lock table.
@@ -3330,8 +3354,8 @@ typedef int(MDBX_reader_list_func)(void *ctx, int num, int slot, mdbx_pid_t pid,
*
* Returns A non-zero error value on failure and 0 on success,
* or MDBX_RESULT_TRUE (-1) if the reader lock table is empty. */
-LIBMDBX_API int mdbx_reader_list(MDBX_env *env, MDBX_reader_list_func *func,
- void *ctx);
+LIBMDBX_API int mdbx_reader_list(const MDBX_env *env,
+ MDBX_reader_list_func *func, void *ctx);
/* Check for stale entries in the reader lock table.
*
@@ -3353,7 +3377,8 @@ LIBMDBX_API int mdbx_reader_check(MDBX_env *env, int *dead);
*
* Returns Number of transactions committed after the given was started for
* read, or negative value on failure. */
-__deprecated LIBMDBX_API int mdbx_txn_straggler(MDBX_txn *txn, int *percent);
+__deprecated LIBMDBX_API int mdbx_txn_straggler(const MDBX_txn *txn,
+ int *percent);
/* A lack-of-space callback function to resolve issues with a laggard readers.
*
@@ -3426,7 +3451,7 @@ LIBMDBX_API int mdbx_env_set_oomfunc(MDBX_env *env, MDBX_oom_func *oom_func);
* [in] env An environment handle returned by mdbx_env_create().
*
* Returns A MDBX_oom_func function or NULL if disabled. */
-LIBMDBX_API MDBX_oom_func *mdbx_env_get_oomfunc(MDBX_env *env);
+LIBMDBX_API MDBX_oom_func *mdbx_env_get_oomfunc(const MDBX_env *env);
/**** B-tree Traversal *********************************************************
* This is internal API for mdbx_chk tool. You should avoid to use it, except
diff --git a/libs/libmdbx/src/src/CMakeLists.txt b/libs/libmdbx/src/src/CMakeLists.txt
deleted file mode 100644
index 8a33ede361..0000000000
--- a/libs/libmdbx/src/src/CMakeLists.txt
+++ /dev/null
@@ -1,263 +0,0 @@
-##
-## Copyright 2020 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>.
-##
-
-# Get version
-fetch_version(MDBX "${CMAKE_CURRENT_SOURCE_DIR}/../VERSION")
-message(STATUS "libmdbx version is ${MDBX_VERSION}")
-
-if(MDBX_ALLOY_MODE)
- set(LIBMDBX_SOURCES alloy.c)
-else()
- set(LIBMDBX_SOURCES
- elements/options.h elements/defs.h elements/internals.h elements/osal.h
- elements/core.c elements/osal.c elements/lck-posix.c elements/lck-windows.c)
-endif()
-list(APPEND LIBMDBX_SOURCES ../mdbx.h
- "${CMAKE_CURRENT_SOURCE_DIR}/elements/version.c"
- "${CMAKE_CURRENT_SOURCE_DIR}/elements/config.h")
-
-if(MDBX_BUILD_SHARED_LIBRARY)
- add_library(mdbx SHARED ${LIBMDBX_SOURCES})
- target_compile_definitions(mdbx PRIVATE LIBMDBX_EXPORTS INTERFACE LIBMDBX_IMPORTS)
- set(MDBX_LIBDEP_MODE PRIVATE)
-else()
- add_library(mdbx STATIC ${LIBMDBX_SOURCES})
- set(MDBX_LIBDEP_MODE PUBLIC)
-endif()
-
-if(CC_HAS_VISIBILITY AND (LTO_ENABLED OR INTERPROCEDURAL_OPTIMIZATION))
- set_target_properties(mdbx PROPERTIES LINK_FLAGS "-fvisibility=hidden")
-endif()
-
-if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
- if(MSVC)
- if(NOT MSVC_LIB_EXE)
- # Find lib.exe
- get_filename_component(CL_NAME ${CMAKE_C_COMPILER} NAME)
- string(REPLACE cl.exe lib.exe MSVC_LIB_EXE ${CL_NAME})
- find_program(MSVC_LIB_EXE ${MSVC_LIB_EXE})
- endif()
- if(MSVC_LIB_EXE)
- message(STATUS "Found MSVC's lib tool: ${MSVC_LIB_EXE}")
- set(MDBX_NTDLL_EXTRA_IMPLIB ${CMAKE_CURRENT_BINARY_DIR}/mdbx_ntdll_extra.lib)
- add_custom_command(OUTPUT ${MDBX_NTDLL_EXTRA_IMPLIB}
- COMMENT "Create extra-import-library for ntdll.dll"
- MAIN_DEPENDENCY "${CMAKE_CURRENT_SOURCE_DIR}/elements/ntdll.def"
- COMMAND ${MSVC_LIB_EXE} /def:"${CMAKE_CURRENT_SOURCE_DIR}/elements/ntdll.def" /out:"${MDBX_NTDLL_EXTRA_IMPLIB}" ${INITIAL_CMAKE_STATIC_LINKER_FLAGS})
- else()
- message(SEND_ERROR "MSVC's lib tool not found")
- endif()
- elseif(MINGW OR MINGW64)
- if(NOT DLLTOOL)
- # Find dlltool
- get_filename_component(GCC_NAME ${CMAKE_C_COMPILER} NAME)
- string(REPLACE gcc dlltool DLLTOOL_NAME ${GCC_NAME})
- find_program(DLLTOOL NAMES ${DLLTOOL_NAME})
- endif()
- if(DLLTOOL)
- message(STATUS "Found dlltool: ${DLLTOOL}")
- set(MDBX_NTDLL_EXTRA_IMPLIB "${CMAKE_CURRENT_BINARY_DIR}/mdbx_ntdll_extra.a")
- add_custom_command(OUTPUT ${MDBX_NTDLL_EXTRA_IMPLIB}
- COMMENT "Create extra-import-library for ntdll.dll"
- MAIN_DEPENDENCY "${CMAKE_CURRENT_SOURCE_DIR}/elements/ntdll.def"
- COMMAND ${DLLTOOL} -d "${CMAKE_CURRENT_SOURCE_DIR}/elements/ntdll.def" -l "${MDBX_NTDLL_EXTRA_IMPLIB}")
- else()
- message(SEND_ERROR "dlltool not found")
- endif()
- endif()
-endif()
-
-target_link_libraries(mdbx ${MDBX_LIBDEP_MODE} ${CMAKE_THREAD_LIBS_INIT})
-if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
- target_link_libraries(mdbx ${MDBX_LIBDEP_MODE} ntdll.lib)
- if(MDBX_NTDLL_EXTRA_IMPLIB)
- # LY: Sometimes Cmake requires a nightmarish magic for simple things.
- # 1) create a target out of the library compilation result
- add_custom_target(ntdll_extra_target DEPENDS ${MDBX_NTDLL_EXTRA_IMPLIB})
- # 2) create an library target out of the library compilation result
- add_library(ntdll_extra STATIC IMPORTED GLOBAL)
- add_dependencies(ntdll_extra ntdll_extra_target)
- # 3) specify where the library is (and where to find the headers)
- set_target_properties(ntdll_extra
- PROPERTIES
- IMPORTED_LOCATION ${MDBX_NTDLL_EXTRA_IMPLIB})
- target_link_libraries(mdbx ${MDBX_LIBDEP_MODE} ntdll_extra)
- endif()
-endif()
-
-if(${CMAKE_SYSTEM_NAME} STREQUAL "SunOS" OR ${CMAKE_SYSTEM_NAME} STREQUAL "Solaris")
- target_link_libraries(mdbx ${MDBX_LIBDEP_MODE} kstat)
-endif()
-
-set_target_properties(mdbx PROPERTIES
- INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>
- C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON
- PUBLIC_HEADER "../mdbx.h")
-
-if(CC_HAS_FASTMATH)
- target_compile_options(mdbx PRIVATE "-ffast-math")
-endif()
-if(BUILD_FOR_NATIVE_CPU AND CC_HAS_ARCH_NATIVE)
- target_compile_options(mdbx PUBLIC "-march=native")
-endif()
-if(CC_HAS_VISIBILITY)
- target_compile_options(mdbx PRIVATE "-fvisibility=hidden")
-endif()
-
-################################################################################
-#
-# library build info (used in library version output)
-#
-set(MDBX_BUILD_FLAGS "")
-
-# append cmake's build-type flags and defines
-if(NOT CMAKE_CONFIGURATION_TYPES)
- list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_FLAGS_${CMAKE_BUILD_TYPE_UPPERCASE}})
- list(APPEND MDBX_BUILD_FLAGS ${CMAKE_C_DEFINES_${CMAKE_BUILD_TYPE_UPPERCASE}})
-endif()
-
-# append linker dll's options
-if(LIBMDBX_TYPE STREQUAL "SHARED")
- list(APPEND MDBX_BUILD_FLAGS ${CMAKE_SHARED_LINKER_FLAGS})
-endif()
-
-# get definitions
-get_target_property(defs_list mdbx COMPILE_DEFINITIONS)
-if(defs_list)
- list(APPEND MDBX_BUILD_FLAGS ${defs_list})
-endif()
-
-# get target compile options
-get_target_property(options_list mdbx COMPILE_OPTIONS)
-if(options_list)
- list(APPEND MDBX_BUILD_FLAGS ${options_list})
-endif()
-
-list(REMOVE_DUPLICATES MDBX_BUILD_FLAGS)
-string(REPLACE ";" " " MDBX_BUILD_FLAGS "${MDBX_BUILD_FLAGS}")
-if(CMAKE_CONFIGURATION_TYPES)
- # add dynamic part via per-configuration define
- message(STATUS "MDBX Compile Flags: ${MDBX_BUILD_FLAGS} <AND CONFIGURATION DEPENDENT>")
- add_definitions(-DMDBX_BUILD_FLAGS_CONFIG="$<$<CONFIG:Debug>:${CMAKE_C_FLAGS_DEBUG} ${CMAKE_C_DEFINES_DEBUG}>$<$<CONFIG:Release>:${CMAKE_C_FLAGS_RELEASE} ${CMAKE_C_DEFINES_RELEASE}>$<$<CONFIG:RelWithDebInfo>:${CMAKE_C_FLAGS_RELWITHDEBINFO} ${CMAKE_C_DEFINES_RELWITHDEBINFO}>$<$<CONFIG:MinSizeRel>:${CMAKE_C_FLAGS_MINSIZEREL} ${CMAKE_C_DEFINES_MINSIZEREL}>")
-else()
- message(STATUS "MDBX Compile Flags: ${MDBX_BUILD_FLAGS}")
-endif()
-
-# get compiler info
-execute_process(COMMAND sh -c "${CMAKE_C_COMPILER} --version | head -1"
- OUTPUT_VARIABLE MDBX_BUILD_COMPILER
- OUTPUT_STRIP_TRAILING_WHITESPACE
- ERROR_QUIET
- RESULT_VARIABLE rc)
-if(rc OR NOT MDBX_BUILD_COMPILER)
- string(STRIP "${CMAKE_C_COMPILER_ID}-${CMAKE_C_COMPILER_VERSION}" MDBX_BUILD_COMPILER)
-endif()
-
-# make a build-target triplet
-if(CMAKE_C_COMPILER_TARGET)
- set(MDBX_BUILD_TARGET "${CMAKE_C_COMPILER_TARGET}")
-elseif(CMAKE_C_PLATFORM_ID AND NOT CMAKE_C_PLATFORM_ID STREQUAL CMAKE_SYSTEM_NAME)
- string(STRIP "${CMAKE_C_PLATFORM_ID}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-elseif(CMAKE_LIBRARY_ARCHITECTURE)
- string(STRIP "${CMAKE_LIBRARY_ARCHITECTURE}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-elseif(CMAKE_GENERATOR_PLATFORM AND NOT CMAKE_C_PLATFORM_ID STREQUAL CMAKE_SYSTEM_NAME)
- string(STRIP "${CMAKE_GENERATOR_PLATFORM}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-elseif(CMAKE_SYSTEM_ARCH)
- string(STRIP "${CMAKE_SYSTEM_ARCH}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-else()
- string(STRIP "${CMAKE_SYSTEM_PROCESSOR}-${CMAKE_SYSTEM_NAME}" MDBX_BUILD_TARGET)
-endif()
-if(CMAKE_CONFIGURATION_TYPES)
- add_definitions(-DMDBX_BUILD_CONFIG="$<CONFIG>")
-else()
- set(MDBX_BUILD_CONFIG ${CMAKE_BUILD_TYPE})
-endif()
-
-# options
-string(TIMESTAMP MDBX_BUILD_TIMESTAMP UTC)
-set(options VERSION C_COMPILER CXX_COMPILER)
-foreach(item IN LISTS options)
- if(DEFINED ${item})
- set(value "${${item}}")
- elseif(DEFINED MDBX_${item})
- set(item MDBX_${item})
- set(value "${${item}}")
- elseif(DEFINED CMAKE_${item})
- set(item CMAKE_${item})
- set(value "${${item}}")
- else()
- set(value "undefined")
- endif()
- message(STATUS "${item}: ${value}")
-endforeach(item)
-
-# generate version and config files
-configure_file("${CMAKE_CURRENT_SOURCE_DIR}/elements/version.c.in"
- "${CMAKE_CURRENT_SOURCE_DIR}/elements/version.c" ESCAPE_QUOTES)
-
-file(SHA256 "${CMAKE_CURRENT_SOURCE_DIR}/elements/version.c" MDBX_SOURCERY_DIGEST)
-string(MAKE_C_IDENTIFIER "${MDBX_GIT_DESCRIBE}" MDBX_SOURCERY_SUFFIX)
-set(MDBX_BUILD_SOURCERY "${MDBX_SOURCERY_DIGEST}_${MDBX_SOURCERY_SUFFIX}")
-
-configure_file("${CMAKE_CURRENT_SOURCE_DIR}/elements/config.h.in"
- "${CMAKE_CURRENT_SOURCE_DIR}/elements/config.h" ESCAPE_QUOTES)
-add_definitions(-DMDBX_CONFIG_H="config.h")
-
-# installation
-if(MDBX_BUILD_SHARED_LIBRARY)
- if(CMAKE_VERSION VERSION_LESS 3.12)
- install(TARGETS mdbx EXPORT libmdbx
- RUNTIME
- DESTINATION bin
- COMPONENT runtime
- LIBRARY
- DESTINATION lib
- COMPONENT runtime
- OBJECTS
- DESTINATION lib
- COMPONENT devel
- ARCHIVE
- DESTINATION lib
- COMPONENT devel
- PUBLIC_HEADER
- DESTINATION include
- COMPONENT devel
- INCLUDES
- DESTINATION include
- COMPONENT devel)
- else()
- install(TARGETS mdbx EXPORT libmdbx
- RUNTIME
- DESTINATION bin
- COMPONENT runtime
- LIBRARY
- DESTINATION lib
- COMPONENT runtime
- NAMELINK_COMPONENT devel
- OBJECTS
- DESTINATION lib
- COMPONENT devel
- ARCHIVE
- DESTINATION lib
- COMPONENT devel
- PUBLIC_HEADER
- DESTINATION include
- COMPONENT devel
- INCLUDES
- DESTINATION include
- COMPONENT devel)
- endif()
-endif()
-
-add_subdirectory(tools)
diff --git a/libs/libmdbx/src/src/alloy.c b/libs/libmdbx/src/src/alloy.c
index befdf8be45..c2b512005d 100644
--- a/libs/libmdbx/src/src/alloy.c
+++ b/libs/libmdbx/src/src/alloy.c
@@ -11,15 +11,15 @@
* top-level directory of the distribution or, alternatively, at
* <http://www.OpenLDAP.org/license.html>. */
-#define MDBX_ALLOY 1 /* amalgamated build */
-#include "elements/internals.h" /* must be included first */
+#define MDBX_ALLOY 1 /* alloyed build */
+#include "internals.h" /* must be included first */
-#include "elements/core.c"
-#include "elements/osal.c"
-#include "elements/version.c"
+#include "core.c"
+#include "osal.c"
+#include "version.c"
#if defined(_WIN32) || defined(_WIN64)
-#include "elements/lck-windows.c"
+#include "lck-windows.c"
#else
-#include "elements/lck-posix.c"
+#include "lck-posix.c"
#endif
diff --git a/libs/libmdbx/src/src/elements/config.h.in b/libs/libmdbx/src/src/config.h.in
index b8776d98a2..12f3dc635c 100644
--- a/libs/libmdbx/src/src/elements/config.h.in
+++ b/libs/libmdbx/src/src/config.h.in
@@ -4,9 +4,6 @@
/* *INDENT-OFF* */
/* clang-format off */
-#cmakedefine HAVE_VALGRIND_MEMCHECK_H
-#cmakedefine HAS_RELAXED_CONSTEXPR
-
#cmakedefine LTO_ENABLED
#cmakedefine MDBX_USE_VALGRIND
#cmakedefine ENABLE_GPROF
@@ -20,7 +17,6 @@
#ifndef MDBX_TXN_CHECKPID_AUTO
#cmakedefine01 MDBX_TXN_CHECKPID
#endif
-#cmakedefine01 MDBX_BUILD_SHARED_LIBRARY
#cmakedefine MDBX_LOCKING_AUTO
#ifndef MDBX_LOCKING_AUTO
#cmakedefine MDBX_LOCKING @MDBX_LOCKING@
@@ -34,7 +30,7 @@
#cmakedefine01 MDBX_CONFIG_MANUAL_TLS_CALLBACK
#cmakedefine01 MDBX_AVOID_CRT
-/* MacOS */
+/* MacOS & iOS */
#cmakedefine01 MDBX_OSX_SPEED_INSTEADOF_DURABILITY
/* POSIX */
@@ -47,7 +43,7 @@
/* Build Info */
#cmakedefine MDBX_BUILD_TIMESTAMP "@MDBX_BUILD_TIMESTAMP@"
#cmakedefine MDBX_BUILD_TARGET "@MDBX_BUILD_TARGET@"
-#cmakedefine MDBX_BUILD_CONFIG "@MDBX_BUILD_CONFIG@"
+#cmakedefine MDBX_BUILD_TYPE "@MDBX_BUILD_TYPE@"
#cmakedefine MDBX_BUILD_COMPILER "@MDBX_BUILD_COMPILER@"
#cmakedefine MDBX_BUILD_FLAGS "@MDBX_BUILD_FLAGS@"
#cmakedefine MDBX_BUILD_SOURCERY @MDBX_BUILD_SOURCERY@
diff --git a/libs/libmdbx/src/src/elements/core.c b/libs/libmdbx/src/src/core.c
index 6a5b079493..098da86f45 100644
--- a/libs/libmdbx/src/src/elements/core.c
+++ b/libs/libmdbx/src/src/core.c
@@ -328,17 +328,17 @@ node_largedata_pgno(const MDBX_node *const __restrict node) {
*/
#define PAGEROOM(pagesize) ((pagesize)-PAGEHDRSZ)
-#define EVEN_FLOOR(n) ((n) & ~1ul)
+#define EVEN_FLOOR(n) ((n) & ~(size_t)1)
#define BRANCH_NODEMAX(pagesize) \
(EVEN_FLOOR(PAGEROOM(pagesize) / (MDBX_MINKEYS * 2 - 1)) - sizeof(indx_t))
#define LEAF_NODEMAX(pagesize) (PAGEROOM(pagesize) - sizeof(indx_t))
#define MAX_GC1OVPAGE(pagesize) (PAGEROOM(pagesize) / sizeof(pgno_t) - 1)
-__cold int mdbx_env_get_maxkeysize(MDBX_env *env) {
+__cold int mdbx_env_get_maxkeysize(const MDBX_env *env) {
return mdbx_env_get_maxkeysize_ex(env, MDBX_DUPSORT);
}
-__cold int mdbx_env_get_maxkeysize_ex(MDBX_env *env, unsigned flags) {
+__cold int mdbx_env_get_maxkeysize_ex(const MDBX_env *env, unsigned flags) {
if (unlikely(!env || env->me_signature != MDBX_ME_SIGNATURE))
return -1;
@@ -370,7 +370,7 @@ __cold intptr_t mdbx_limits_keysize_max(intptr_t pagesize, unsigned flags) {
return BRANCH_NODEMAX(pagesize) - NODESIZE - sizeof(pgno_t);
}
-__cold int mdbx_env_get_maxvalsize_ex(MDBX_env *env, unsigned flags) {
+__cold int mdbx_env_get_maxvalsize_ex(const MDBX_env *env, unsigned flags) {
if (unlikely(!env || env->me_signature != MDBX_ME_SIGNATURE))
return -1;
@@ -489,7 +489,7 @@ static __pure_function __always_inline pgno_t bytes2pgno(const MDBX_env *env,
static __pure_function size_t pgno_align2os_bytes(const MDBX_env *env,
pgno_t pgno) {
- return roundup_powerof2(pgno2bytes(env, pgno), env->me_os_psize);
+ return ceil_powerof2(pgno2bytes(env, pgno), env->me_os_psize);
}
static __pure_function pgno_t pgno_align2os_pgno(const MDBX_env *env,
@@ -499,8 +499,7 @@ static __pure_function pgno_t pgno_align2os_pgno(const MDBX_env *env,
static __pure_function size_t bytes_align2os_bytes(const MDBX_env *env,
size_t bytes) {
- return roundup_powerof2(roundup_powerof2(bytes, env->me_psize),
- env->me_os_psize);
+ return ceil_powerof2(ceil_powerof2(bytes, env->me_psize), env->me_os_psize);
}
/* Address of first usable data byte in a page, after the header */
@@ -1012,13 +1011,13 @@ static __inline int thread_key_create(mdbx_thread_key_t *key) {
#else
rc = pthread_key_create(key, nullptr);
#endif
- mdbx_trace("&key = %p, value 0x%x, rc %d", __Wpedantic_format_voidptr(key),
- (unsigned)*key, rc);
+ mdbx_trace("&key = %p, value %" PRIuPTR ", rc %d",
+ __Wpedantic_format_voidptr(key), (uintptr_t)*key, rc);
return rc;
}
static __inline void thread_key_delete(mdbx_thread_key_t key) {
- mdbx_trace("key = 0x%x", (unsigned)key);
+ mdbx_trace("key = %" PRIuPTR, (uintptr_t)key);
#if defined(_WIN32) || defined(_WIN64)
mdbx_ensure(nullptr, TlsFree(key));
#else
@@ -1045,7 +1044,7 @@ static void thread_rthc_set(mdbx_thread_key_t key, const void *value) {
static __thread uint32_t thread_registration_state;
if (value && unlikely(thread_registration_state == MDBX_THREAD_RTHC_ZERO)) {
thread_registration_state = MDBX_THREAD_RTHC_REGISTERD;
- mdbx_trace("thread registered 0x%" PRIxPTR, (uintptr_t)mdbx_thread_self());
+ mdbx_trace("thread registered 0x%" PRIxPTR, mdbx_thread_self());
if (&__cxa_thread_atexit_impl == nullptr ||
__cxa_thread_atexit_impl(mdbx_rthc_thread_dtor,
&thread_registration_state,
@@ -1055,8 +1054,8 @@ static void thread_rthc_set(mdbx_thread_key_t key, const void *value) {
thread_registration_state = MDBX_THREAD_RTHC_COUNTED;
const unsigned count_before = atomic_add32(&rthc_pending, 1);
mdbx_ensure(nullptr, count_before < INT_MAX);
- mdbx_trace("fallback to pthreads' tsd, key 0x%x, count %u",
- (unsigned)rthc_key, count_before);
+ mdbx_trace("fallback to pthreads' tsd, key %" PRIuPTR ", count %u",
+ (uintptr_t)rthc_key, count_before);
(void)count_before;
}
}
@@ -1098,7 +1097,7 @@ __cold void mdbx_rthc_global_init(void) {
__cold void mdbx_rthc_thread_dtor(void *ptr) {
rthc_lock();
mdbx_trace(">> pid %d, thread 0x%" PRIxPTR ", rthc %p", mdbx_getpid(),
- (uintptr_t)mdbx_thread_self(), ptr);
+ mdbx_thread_self(), ptr);
const uint32_t self_pid = mdbx_getpid();
for (unsigned i = 0; i < rthc_count; ++i) {
@@ -1112,7 +1111,7 @@ __cold void mdbx_rthc_thread_dtor(void *ptr) {
if (pthread_setspecific(key, nullptr) != 0) {
mdbx_trace("== thread 0x%" PRIxPTR
", rthc %p: ignore race with tsd-key deletion",
- (uintptr_t)mdbx_thread_self(), ptr);
+ mdbx_thread_self(), ptr);
continue /* ignore race with tsd-key deletion by mdbx_env_close() */;
}
#endif
@@ -1120,39 +1119,35 @@ __cold void mdbx_rthc_thread_dtor(void *ptr) {
mdbx_trace("== thread 0x%" PRIxPTR
", rthc %p, [%i], %p ... %p (%+i), rtch-pid %i, "
"current-pid %i",
- (uintptr_t)mdbx_thread_self(), __Wpedantic_format_voidptr(rthc),
- i, __Wpedantic_format_voidptr(rthc_table[i].begin),
+ mdbx_thread_self(), __Wpedantic_format_voidptr(rthc), i,
+ __Wpedantic_format_voidptr(rthc_table[i].begin),
__Wpedantic_format_voidptr(rthc_table[i].end),
(int)(rthc - rthc_table[i].begin), rthc->mr_pid, self_pid);
if (rthc->mr_pid == self_pid) {
mdbx_trace("==== thread 0x%" PRIxPTR ", rthc %p, cleanup",
- (uintptr_t)mdbx_thread_self(),
- __Wpedantic_format_voidptr(rthc));
+ mdbx_thread_self(), __Wpedantic_format_voidptr(rthc));
rthc->mr_pid = 0;
}
}
#if defined(_WIN32) || defined(_WIN64)
- mdbx_trace("<< thread 0x%" PRIxPTR ", rthc %p", (uintptr_t)mdbx_thread_self(),
- ptr);
+ mdbx_trace("<< thread 0x%" PRIxPTR ", rthc %p", mdbx_thread_self(), ptr);
rthc_unlock();
#else
const char self_registration = *(char *)ptr;
*(char *)ptr = MDBX_THREAD_RTHC_ZERO;
mdbx_trace("== thread 0x%" PRIxPTR ", rthc %p, pid %d, self-status %d",
- (uintptr_t)mdbx_thread_self(), ptr, mdbx_getpid(),
- self_registration);
+ mdbx_thread_self(), ptr, mdbx_getpid(), self_registration);
if (self_registration == MDBX_THREAD_RTHC_COUNTED)
mdbx_ensure(nullptr, atomic_sub32(&rthc_pending, 1) > 0);
if (rthc_pending == 0) {
mdbx_trace("== thread 0x%" PRIxPTR ", rthc %p, pid %d, wake",
- (uintptr_t)mdbx_thread_self(), ptr, mdbx_getpid());
+ mdbx_thread_self(), ptr, mdbx_getpid());
mdbx_ensure(nullptr, pthread_cond_broadcast(&rthc_cond) == 0);
}
- mdbx_trace("<< thread 0x%" PRIxPTR ", rthc %p", (uintptr_t)mdbx_thread_self(),
- ptr);
+ mdbx_trace("<< thread 0x%" PRIxPTR ", rthc %p", mdbx_thread_self(), ptr);
/* Allow tail call optimization, i.e. gcc should generate the jmp instruction
* instead of a call for pthread_mutex_unlock() and therefore CPU could not
* return to current DSO's code section, which may be unloaded immediately
@@ -1168,7 +1163,7 @@ __cold void mdbx_rthc_global_dtor(void) {
#if !defined(_WIN32) && !defined(_WIN64)
char *rthc = (char *)pthread_getspecific(rthc_key);
mdbx_trace("== thread 0x%" PRIxPTR ", rthc %p, pid %d, self-status %d",
- (uintptr_t)mdbx_thread_self(), __Wpedantic_format_voidptr(rthc),
+ mdbx_thread_self(), __Wpedantic_format_voidptr(rthc),
mdbx_getpid(), rthc ? *rthc : -1);
if (rthc) {
const char self_registration = *(char *)rthc;
@@ -1205,9 +1200,9 @@ __cold void mdbx_rthc_global_dtor(void) {
thread_key_delete(key);
for (MDBX_reader *rthc = rthc_table[i].begin; rthc < rthc_table[i].end;
++rthc) {
- mdbx_trace("== [%i] = key %zu, %p ... %p, rthc %p (%+i), "
+ mdbx_trace("== [%i] = key %" PRIuPTR ", %p ... %p, rthc %p (%+i), "
"rthc-pid %i, current-pid %i",
- i, (size_t)key,
+ i, (uintptr_t)key,
__Wpedantic_format_voidptr(rthc_table[i].begin),
__Wpedantic_format_voidptr(rthc_table[i].end),
__Wpedantic_format_voidptr(rthc),
@@ -1251,8 +1246,8 @@ __cold int mdbx_rthc_alloc(mdbx_thread_key_t *key, MDBX_reader *begin,
rthc_lock();
const mdbx_thread_key_t new_key = key ? *key : 0;
- mdbx_trace(">> key %zu, rthc_count %u, rthc_limit %u", (size_t)new_key,
- rthc_count, rthc_limit);
+ mdbx_trace(">> key %" PRIuPTR ", rthc_count %u, rthc_limit %u",
+ (uintptr_t)new_key, rthc_count, rthc_limit);
if (rthc_count == rthc_limit) {
rthc_entry_t *new_table =
mdbx_realloc((rthc_table == rthc_table_static) ? nullptr : rthc_table,
@@ -1266,16 +1261,16 @@ __cold int mdbx_rthc_alloc(mdbx_thread_key_t *key, MDBX_reader *begin,
rthc_table = new_table;
rthc_limit *= 2;
}
- mdbx_trace("== [%i] = key %zu, %p ... %p", rthc_count, (size_t)new_key,
- __Wpedantic_format_voidptr(begin),
+ mdbx_trace("== [%i] = key %" PRIuPTR ", %p ... %p", rthc_count,
+ (uintptr_t)new_key, __Wpedantic_format_voidptr(begin),
__Wpedantic_format_voidptr(end));
rthc_table[rthc_count].key_valid = key ? true : false;
rthc_table[rthc_count].thr_tls_key = key ? new_key : 0;
rthc_table[rthc_count].begin = begin;
rthc_table[rthc_count].end = end;
++rthc_count;
- mdbx_trace("<< key %zu, rthc_count %u, rthc_limit %u", (size_t)new_key,
- rthc_count, rthc_limit);
+ mdbx_trace("<< key %" PRIuPTR ", rthc_count %u, rthc_limit %u",
+ (uintptr_t)new_key, rthc_count, rthc_limit);
rthc_unlock();
return MDBX_SUCCESS;
@@ -1324,7 +1319,7 @@ __cold void mdbx_rthc_remove(const mdbx_thread_key_t key) {
//------------------------------------------------------------------------------
-#define RTHC_ENVLIST_END ((MDBX_env *)((size_t)50459))
+#define RTHC_ENVLIST_END ((MDBX_env *)((uintptr_t)50459))
static MDBX_env *inprocess_lcklist_head = RTHC_ENVLIST_END;
static __inline void lcklist_lock(void) {
@@ -1378,8 +1373,8 @@ static int uniq_peek(const mdbx_mmap_t *pending, mdbx_mmap_t *scan) {
static int uniq_poke(const mdbx_mmap_t *pending, mdbx_mmap_t *scan,
uint64_t *abra) {
if (*abra == 0) {
- const size_t tid = mdbx_thread_self();
- size_t uit = 0;
+ const uintptr_t tid = mdbx_thread_self();
+ uintptr_t uit = 0;
memcpy(&uit, &tid, (sizeof(tid) < sizeof(uit)) ? sizeof(tid) : sizeof(uit));
*abra =
rrxmrrxmsx_0(mdbx_osal_monotime() + UINT64_C(5873865991930747) * uit);
@@ -2539,10 +2534,10 @@ static int lcklist_detach_locked(MDBX_env *env) {
static __always_inline size_t pnl2bytes(const size_t size) {
assert(size > 0 && size <= MDBX_PNL_MAX * 2);
- size_t bytes = roundup_powerof2(MDBX_ASSUME_MALLOC_OVERHEAD +
- sizeof(pgno_t) * (size + 2),
- MDBX_PNL_GRANULATE * sizeof(pgno_t)) -
- MDBX_ASSUME_MALLOC_OVERHEAD;
+ size_t bytes =
+ ceil_powerof2(MDBX_ASSUME_MALLOC_OVERHEAD + sizeof(pgno_t) * (size + 2),
+ MDBX_PNL_GRANULATE * sizeof(pgno_t)) -
+ MDBX_ASSUME_MALLOC_OVERHEAD;
return bytes;
}
@@ -2806,10 +2801,10 @@ static __hot unsigned mdbx_pnl_exist(MDBX_PNL pnl, pgno_t id) {
static __always_inline size_t txl2bytes(const size_t size) {
assert(size > 0 && size <= MDBX_TXL_MAX * 2);
- size_t bytes = roundup_powerof2(MDBX_ASSUME_MALLOC_OVERHEAD +
- sizeof(txnid_t) * (size + 2),
- MDBX_TXL_GRANULATE * sizeof(txnid_t)) -
- MDBX_ASSUME_MALLOC_OVERHEAD;
+ size_t bytes =
+ ceil_powerof2(MDBX_ASSUME_MALLOC_OVERHEAD + sizeof(txnid_t) * (size + 2),
+ MDBX_TXL_GRANULATE * sizeof(txnid_t)) -
+ MDBX_ASSUME_MALLOC_OVERHEAD;
return bytes;
}
@@ -3135,8 +3130,8 @@ static int __must_check_result mdbx_page_check(MDBX_env *env,
static int __must_check_result mdbx_cursor_check(MDBX_cursor *mc, bool pending);
static int __must_check_result mdbx_cursor_del0(MDBX_cursor *mc);
static int __must_check_result mdbx_del0(MDBX_txn *txn, MDBX_dbi dbi,
- MDBX_val *key, MDBX_val *data,
- unsigned flags);
+ const MDBX_val *key,
+ const MDBX_val *data, unsigned flags);
static int __must_check_result mdbx_cursor_sibling(MDBX_cursor *mc,
int move_right);
static int __must_check_result mdbx_cursor_next(MDBX_cursor *mc, MDBX_val *key,
@@ -3185,7 +3180,8 @@ static const char *__mdbx_strerr(int errnum) {
NULL /* MDBX_TLS_FULL (-30789): unused in MDBX */,
"MDBX_TXN_FULL: Transaction has too many dirty pages,"
" i.e transaction is too big",
- "MDBX_CURSOR_FULL: Internal error - Cursor stack limit reached",
+ "MDBX_CURSOR_FULL: Cursor stack limit reachedn - this usually indicates"
+ " corruption, i.e branch-pages loop",
"MDBX_PAGE_FULL: Internal error - Page has no more space",
"MDBX_UNABLE_EXTEND_MAPSIZE: Database engine was unable to extend"
" mapping, e.g. since address space is unavailable or busy,"
@@ -3522,13 +3518,13 @@ static __maybe_unused void mdbx_page_list(MDBX_page *mp) {
*tp = tracked->mc_next; \
} while (0)
-int mdbx_cmp(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
+int mdbx_cmp(const MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
const MDBX_val *b) {
mdbx_assert(NULL, txn->mt_signature == MDBX_MT_SIGNATURE);
return txn->mt_dbxs[dbi].md_cmp(a, b);
}
-int mdbx_dcmp(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
+int mdbx_dcmp(const MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *a,
const MDBX_val *b) {
mdbx_assert(NULL, txn->mt_signature == MDBX_MT_SIGNATURE);
return txn->mt_dbxs[dbi].md_dcmp(a, b);
@@ -4143,7 +4139,8 @@ mark_done:
* [in] data For a put operation, the data being stored.
*
* Returns 0 on success, non-zero on failure. */
-static int mdbx_page_spill(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data) {
+static int mdbx_page_spill(MDBX_cursor *mc, const MDBX_val *key,
+ const MDBX_val *data) {
if (mc->mc_flags & C_SUB)
return MDBX_SUCCESS;
@@ -4438,7 +4435,7 @@ static const char *mdbx_durable_str(const MDBX_meta *const meta) {
/*----------------------------------------------------------------------------*/
/* Find oldest txnid still referenced. */
-static txnid_t mdbx_find_oldest(MDBX_txn *txn) {
+static txnid_t mdbx_find_oldest(const MDBX_txn *txn) {
mdbx_tassert(txn, (txn->mt_flags & MDBX_RDONLY) == 0);
MDBX_env *env = txn->mt_env;
const txnid_t edge = mdbx_recent_steady_txnid(env);
@@ -4835,7 +4832,9 @@ __cold static int mdbx_wipe_steady(MDBX_env *env, const txnid_t last_steady) {
if (unlikely(err != MDBX_SUCCESS))
return err;
} else {
-#if (defined(__linux__) || defined(__gnu_linux__)) && !defined(MDBX_SAFE4QEMU)
+#if (defined(__linux__) || defined(__gnu_linux__)) && \
+ (!defined(__ANDROID_API__) || __ANDROID_API__ >= 26) && \
+ defined(_GNU_SOURCE) && !defined(MDBX_SAFE4QEMU)
if (sync_file_range(env->me_lazy_fd, 0, pgno2bytes(env, NUM_METAS),
SYNC_FILE_RANGE_WRITE | SYNC_FILE_RANGE_WAIT_AFTER))
err = errno;
@@ -4937,7 +4936,7 @@ skip_cache:
goto fail;
}
- MDBX_cursor recur;
+ MDBX_cursor_couple recur;
for (MDBX_cursor_op op = MDBX_FIRST;;
op = (flags & MDBX_LIFORECLAIM) ? MDBX_PREV : MDBX_NEXT) {
MDBX_val key, data;
@@ -4983,7 +4982,7 @@ skip_cache:
/* Prepare to fetch more and coalesce */
oldest = (flags & MDBX_LIFORECLAIM) ? mdbx_find_oldest(txn)
: *env->me_oldest;
- rc = mdbx_cursor_init(&recur, txn, FREE_DBI);
+ rc = mdbx_cursor_init(&recur.outer, txn, FREE_DBI);
if (unlikely(rc != MDBX_SUCCESS))
goto fail;
if (flags & MDBX_LIFORECLAIM) {
@@ -5011,7 +5010,7 @@ skip_cache:
}
}
- rc = mdbx_cursor_get(&recur, &key, NULL, op);
+ rc = mdbx_cursor_get(&recur.outer, &key, NULL, op);
if (rc == MDBX_NOTFOUND && (flags & MDBX_LIFORECLAIM)) {
if (op == MDBX_SET_RANGE)
continue;
@@ -5022,7 +5021,7 @@ skip_cache:
key.iov_base = &last;
key.iov_len = sizeof(last);
op = MDBX_SET_RANGE;
- rc = mdbx_cursor_get(&recur, &key, NULL, op);
+ rc = mdbx_cursor_get(&recur.outer, &key, NULL, op);
}
}
if (unlikely(rc)) {
@@ -5062,9 +5061,10 @@ skip_cache:
}
/* Reading next GC record */
- np = recur.mc_pg[recur.mc_top];
+ np = recur.outer.mc_pg[recur.outer.mc_top];
if (unlikely((rc = mdbx_node_read(
- &recur, page_node(np, recur.mc_ki[recur.mc_top]),
+ &recur.outer,
+ page_node(np, recur.outer.mc_ki[recur.outer.mc_top]),
&data)) != MDBX_SUCCESS))
goto fail;
@@ -5352,8 +5352,8 @@ __hot static void mdbx_page_copy(MDBX_page *dst, MDBX_page *src, size_t psize) {
/* If page isn't full, just copy the used portion. Adjust
* alignment so memcpy may copy words instead of bytes. */
if (unused >= MDBX_CACHELINE_SIZE * 2) {
- lower = roundup_powerof2(lower + PAGEHDRSZ, sizeof(void *));
- upper = (upper + PAGEHDRSZ) & ~(sizeof(void *) - 1);
+ lower = ceil_powerof2(lower + PAGEHDRSZ, sizeof(void *));
+ upper = floor_powerof2(upper + PAGEHDRSZ, sizeof(void *));
memcpy(dst, src, lower);
dst = (void *)((char *)dst + upper);
src = (void *)((char *)src + upper);
@@ -5528,20 +5528,8 @@ fail:
return rc;
}
-__cold int mdbx_env_sync_ex(MDBX_env *env, int force, int nonblock) {
- if (unlikely(!env))
- return MDBX_EINVAL;
-
- if (unlikely(env->me_signature != MDBX_ME_SIGNATURE))
- return MDBX_EBADSIGN;
-
-#if MDBX_TXN_CHECKPID
- if (unlikely(env->me_pid != mdbx_getpid())) {
- env->me_flags |= MDBX_FATAL_ERROR;
- return MDBX_PANIC;
- }
-#endif /* MDBX_TXN_CHECKPID */
-
+__cold static int mdbx_env_sync_internal(MDBX_env *env, int force,
+ int nonblock) {
unsigned flags = env->me_flags & ~MDBX_NOMETASYNC;
if (unlikely(flags & (MDBX_RDONLY | MDBX_FATAL_ERROR)))
return MDBX_EACCESS;
@@ -5636,6 +5624,23 @@ fastpath:
return rc;
}
+__cold int mdbx_env_sync_ex(MDBX_env *env, int force, int nonblock) {
+ if (unlikely(!env))
+ return MDBX_EINVAL;
+
+ if (unlikely(env->me_signature != MDBX_ME_SIGNATURE))
+ return MDBX_EBADSIGN;
+
+#if MDBX_TXN_CHECKPID
+ if (unlikely(env->me_pid != mdbx_getpid())) {
+ env->me_flags |= MDBX_FATAL_ERROR;
+ return MDBX_PANIC;
+ }
+#endif /* MDBX_TXN_CHECKPID */
+
+ return mdbx_env_sync_internal(env, force, nonblock);
+}
+
__cold int mdbx_env_sync(MDBX_env *env) {
return mdbx_env_sync_ex(env, true, false);
}
@@ -5775,6 +5780,9 @@ static void mdbx_txn_valgrind(MDBX_env *env, MDBX_txn *txn) {
/* inside write-txn */
MDBX_meta *head = mdbx_meta_head(env);
last = head->mm_geo.next;
+ } else if (env->me_flags & MDBX_RDONLY) {
+ /* read-only mode, no write-txn, no wlock mutex */
+ last = NUM_METAS;
} else if (mdbx_txn_lock(env, true) == MDBX_SUCCESS) {
/* no write-txn */
last = NUM_METAS;
@@ -5827,12 +5835,12 @@ static int mdbx_txn_renew0(MDBX_txn *txn, unsigned flags) {
mdbx_assert(env, (flags & ~(MDBX_TXN_BEGIN_FLAGS | MDBX_TXN_SPILLS |
MDBX_WRITEMAP)) == 0);
- const size_t tid = mdbx_thread_self();
+ const uintptr_t tid = mdbx_thread_self();
if (flags & MDBX_RDONLY) {
txn->mt_flags =
MDBX_RDONLY | (env->me_flags & (MDBX_NOTLS | MDBX_WRITEMAP));
MDBX_reader *r = txn->to.reader;
- STATIC_ASSERT(sizeof(size_t) == sizeof(r->mr_tid));
+ STATIC_ASSERT(sizeof(uintptr_t) == sizeof(r->mr_tid));
if (likely(env->me_flags & MDBX_ENV_TXKEY)) {
mdbx_assert(env, !(env->me_flags & MDBX_NOTLS));
r = thread_rthc_get(env->me_txkey);
@@ -6315,7 +6323,7 @@ int mdbx_txn_begin(MDBX_env *env, MDBX_txn *parent, unsigned flags,
return rc;
}
-int mdbx_txn_info(MDBX_txn *txn, MDBX_txn_info *info, int scan_rlt) {
+int mdbx_txn_info(const MDBX_txn *txn, MDBX_txn_info *info, int scan_rlt) {
int rc = check_txn(txn, MDBX_TXN_BLOCKED - MDBX_TXN_HAS_CHILD);
if (unlikely(rc != MDBX_SUCCESS))
return rc;
@@ -6429,20 +6437,20 @@ int mdbx_txn_info(MDBX_txn *txn, MDBX_txn_info *info, int scan_rlt) {
return MDBX_SUCCESS;
}
-MDBX_env *mdbx_txn_env(MDBX_txn *txn) {
+MDBX_env *mdbx_txn_env(const MDBX_txn *txn) {
if (unlikely(!txn || txn->mt_signature != MDBX_MT_SIGNATURE ||
txn->mt_env->me_signature != MDBX_ME_SIGNATURE))
return NULL;
return txn->mt_env;
}
-uint64_t mdbx_txn_id(MDBX_txn *txn) {
+uint64_t mdbx_txn_id(const MDBX_txn *txn) {
if (unlikely(!txn || txn->mt_signature != MDBX_MT_SIGNATURE))
return 0;
return txn->mt_txnid;
}
-int mdbx_txn_flags(MDBX_txn *txn) {
+int mdbx_txn_flags(const MDBX_txn *txn) {
if (unlikely(!txn || txn->mt_signature != MDBX_MT_SIGNATURE))
return -1;
return txn->mt_flags;
@@ -8145,7 +8153,7 @@ static int __cold mdbx_validate_meta(MDBX_env *env, MDBX_meta *const meta,
const uint64_t mapsize_max = meta->mm_geo.upper * (uint64_t)meta->mm_psize;
STATIC_ASSERT(MIN_MAPSIZE < MAX_MAPSIZE);
if (mapsize_max > MAX_MAPSIZE ||
- MAX_PAGENO < roundup_powerof2((size_t)mapsize_max, env->me_os_psize) /
+ MAX_PAGENO < ceil_powerof2((size_t)mapsize_max, env->me_os_psize) /
(size_t)meta->mm_psize) {
if (meta->mm_geo.next - 1 > MAX_PAGENO || used_bytes > MAX_MAPSIZE) {
mdbx_notice("meta[%u] has too large max-mapsize (%" PRIu64 "), skip it",
@@ -8407,27 +8415,26 @@ static int mdbx_sync_locked(MDBX_env *env, unsigned flags,
#endif /* MDBX_USE_VALGRIND || __SANITIZE_ADDRESS__ */
#if defined(MADV_DONTNEED)
const size_t largest_bytes = pgno2bytes(env, largest_pgno);
- const size_t madvise_gap = (largest_bytes < 65536 * 256)
- ? 65536
- : (largest_bytes > MEGABYTE * 4 * 256)
- ? MEGABYTE * 4
- : largest_bytes >> 8;
+ /* threshold to avoid unreasonable frequent madvise() calls */
+ const size_t madvise_treshold = (largest_bytes < 65536 * 256)
+ ? 65536
+ : (largest_bytes > MEGABYTE * 4 * 256)
+ ? MEGABYTE * 4
+ : largest_bytes >> 10;
const size_t discard_edge_bytes = bytes_align2os_bytes(
- env,
- (MDBX_RDONLY & (env->me_lck ? env->me_lck->mti_envmode : env->me_flags))
- ? largest_bytes
- : largest_bytes + madvise_gap);
+ env, ((MDBX_RDONLY &
+ (env->me_lck ? env->me_lck->mti_envmode : env->me_flags))
+ ? largest_bytes
+ : largest_bytes + madvise_treshold));
const pgno_t discard_edge_pgno = bytes2pgno(env, discard_edge_bytes);
const pgno_t prev_discarded_pgno = *env->me_discarded_tail;
- if (prev_discarded_pgno >
- discard_edge_pgno +
- /* threshold to avoid unreasonable frequent madvise() calls */
- bytes2pgno(env, madvise_gap)) {
+ if (prev_discarded_pgno >=
+ discard_edge_pgno + bytes2pgno(env, madvise_treshold)) {
mdbx_notice("open-MADV_%s %u..%u", "DONTNEED", *env->me_discarded_tail,
largest_pgno);
*env->me_discarded_tail = discard_edge_pgno;
const size_t prev_discarded_bytes =
- pgno2bytes(env, prev_discarded_pgno) & ~(env->me_os_psize - 1);
+ ceil_powerof2(pgno2bytes(env, prev_discarded_pgno), env->me_os_psize);
mdbx_ensure(env, prev_discarded_bytes > discard_edge_bytes);
int advise = MADV_DONTNEED;
#if defined(MADV_FREE) && \
@@ -8482,7 +8489,8 @@ static int mdbx_sync_locked(MDBX_env *env, unsigned flags,
mdbx_assert(env, ((flags ^ env->me_flags) & MDBX_WRITEMAP) == 0);
MDBX_meta *const recent_steady_meta = mdbx_meta_steady(env);
if (flags & MDBX_WRITEMAP) {
- const size_t begin = pgno2bytes(env, NUM_METAS) & ~(env->me_os_psize - 1);
+ const size_t begin =
+ floor_powerof2(pgno2bytes(env, NUM_METAS), env->me_os_psize);
const size_t end = pgno_align2os_bytes(env, pending->mm_geo.next);
if (end > begin) {
rc = mdbx_msync(&env->me_dxb_mmap, begin, end - begin,
@@ -8926,9 +8934,9 @@ mdbx_env_set_geometry(MDBX_env *env, intptr_t size_lower, intptr_t size_now,
const size_t unit =
(env->me_os_psize > (size_t)pagesize) ? env->me_os_psize : pagesize;
- size_lower = roundup_powerof2(size_lower, unit);
- size_upper = roundup_powerof2(size_upper, unit);
- size_now = roundup_powerof2(size_now, unit);
+ size_lower = ceil_powerof2(size_lower, unit);
+ size_upper = ceil_powerof2(size_upper, unit);
+ size_now = ceil_powerof2(size_now, unit);
/* LY: подбираем значение size_upper:
* - кратное размеру страницы
@@ -8963,13 +8971,13 @@ mdbx_env_set_geometry(MDBX_env *env, intptr_t size_lower, intptr_t size_now,
}
if (growth_step == 0 && shrink_threshold > 0)
growth_step = 1;
- growth_step = roundup_powerof2(growth_step, unit);
+ growth_step = ceil_powerof2(growth_step, unit);
if (bytes2pgno(env, growth_step) > UINT16_MAX)
growth_step = pgno2bytes(env, UINT16_MAX);
if (shrink_threshold < 0)
shrink_threshold = growth_step + growth_step;
- shrink_threshold = roundup_powerof2(shrink_threshold, unit);
+ shrink_threshold = ceil_powerof2(shrink_threshold, unit);
if (bytes2pgno(env, shrink_threshold) > UINT16_MAX)
shrink_threshold = pgno2bytes(env, UINT16_MAX);
@@ -9145,7 +9153,7 @@ int __cold mdbx_env_set_maxreaders(MDBX_env *env, unsigned readers) {
return MDBX_SUCCESS;
}
-int __cold mdbx_env_get_maxreaders(MDBX_env *env, unsigned *readers) {
+int __cold mdbx_env_get_maxreaders(const MDBX_env *env, unsigned *readers) {
if (!env || !readers)
return MDBX_EINVAL;
@@ -9209,7 +9217,7 @@ static int __cold mdbx_setup_dxb(MDBX_env *env, const int lck_rc) {
mdbx_setup_pagesize(env, meta.mm_psize);
const size_t used_bytes = pgno2bytes(env, meta.mm_geo.next);
const size_t used_aligned2os_bytes =
- roundup_powerof2(used_bytes, env->me_os_psize);
+ ceil_powerof2(used_bytes, env->me_os_psize);
if ((env->me_flags & MDBX_RDONLY) /* readonly */
|| lck_rc != MDBX_RESULT_TRUE /* not exclusive */) {
/* use present params from db */
@@ -9434,8 +9442,8 @@ static int __cold mdbx_setup_dxb(MDBX_env *env, const int lck_rc) {
head->mm_datasync_sign = MDBX_DATASIGN_WEAK;
head->mm_txnid_b.inconsistent = undo_txnid;
const size_t offset = (uint8_t *)data_page(head) - env->me_dxb_mmap.dxb;
- const size_t paged_offset = offset & ~(env->me_os_psize - 1);
- const size_t paged_length = roundup_powerof2(
+ const size_t paged_offset = floor_powerof2(offset, env->me_os_psize);
+ const size_t paged_length = ceil_powerof2(
env->me_psize + offset - paged_offset, env->me_os_psize);
err = mdbx_msync(&env->me_dxb_mmap, paged_offset, paged_length, false);
} else {
@@ -9665,9 +9673,9 @@ static int __cold mdbx_setup_lck(MDBX_env *env, char *lck_pathname,
goto bailout;
if (lck_seize_rc == MDBX_RESULT_TRUE) {
- size = roundup_powerof2(env->me_maxreaders * sizeof(MDBX_reader) +
- sizeof(MDBX_lockinfo),
- env->me_os_psize);
+ size = ceil_powerof2(env->me_maxreaders * sizeof(MDBX_reader) +
+ sizeof(MDBX_lockinfo),
+ env->me_os_psize);
mdbx_jitter4testing(false);
} else {
if (env->me_flags & MDBX_EXCLUSIVE) {
@@ -10149,12 +10157,13 @@ int __cold mdbx_env_open(MDBX_env *env, const char *pathname, unsigned flags,
#endif
bailout:
-#if defined(MDBX_USE_VALGRIND) || defined(__SANITIZE_ADDRESS__)
- mdbx_txn_valgrind(env, nullptr);
-#endif
- if (rc) {
+ if (rc != MDBX_SUCCESS) {
rc = mdbx_env_close0(env) ? MDBX_PANIC : rc;
env->me_flags = saved_me_flags | MDBX_FATAL_ERROR;
+ } else {
+#if defined(MDBX_USE_VALGRIND) || defined(__SANITIZE_ADDRESS__)
+ mdbx_txn_valgrind(env, nullptr);
+#endif
}
mdbx_free(lck_pathname);
return rc;
@@ -10167,6 +10176,7 @@ static int __cold mdbx_env_close0(MDBX_env *env) {
return MDBX_SUCCESS;
}
+ env->me_signature = 0;
env->me_flags &= ~MDBX_ENV_ACTIVE;
env->me_oldest = nullptr;
env->me_sync_timestamp = nullptr;
@@ -10251,37 +10261,34 @@ int __cold mdbx_env_close_ex(MDBX_env *env, int dont_sync) {
if ((env->me_flags & (MDBX_RDONLY | MDBX_FATAL_ERROR)) == 0 && env->me_txn0) {
if (env->me_txn0->mt_owner && env->me_txn0->mt_owner != mdbx_thread_self())
return MDBX_BUSY;
- if (!dont_sync) {
+ } else
+ dont_sync = true;
+
+ if (!atomic_cas32(&env->me_signature, MDBX_ME_SIGNATURE, 0))
+ return MDBX_EBADSIGN;
+
+ if (!dont_sync) {
#if defined(_WIN32) || defined(_WIN64)
- /* On windows, without blocking is impossible to determine whether another
- * process is running a writing transaction or not.
- * Because in the "owner died" condition kernel don't release
- * file lock immediately. */
- rc = mdbx_env_sync_ex(env, true, false);
- rc = (rc == MDBX_RESULT_TRUE) ? MDBX_SUCCESS : rc;
+ /* On windows, without blocking is impossible to determine whether another
+ * process is running a writing transaction or not.
+ * Because in the "owner died" condition kernel don't release
+ * file lock immediately. */
+ rc = mdbx_env_sync_internal(env, true, false);
+ rc = (rc == MDBX_RESULT_TRUE) ? MDBX_SUCCESS : rc;
#else
- struct stat st;
- if (unlikely(fstat(env->me_lazy_fd, &st)))
- rc = errno;
- else if (st.st_nlink > 0 /* don't sync deleted files */) {
- rc = mdbx_env_sync_ex(env, true, true);
- rc = (rc == MDBX_BUSY || rc == EAGAIN || rc == EACCES || rc == EBUSY ||
- rc == EWOULDBLOCK || rc == MDBX_RESULT_TRUE)
- ? MDBX_SUCCESS
- : rc;
- }
-#endif
+ struct stat st;
+ if (unlikely(fstat(env->me_lazy_fd, &st)))
+ rc = errno;
+ else if (st.st_nlink > 0 /* don't sync deleted files */) {
+ rc = mdbx_env_sync_internal(env, true, true);
+ rc = (rc == MDBX_BUSY || rc == EAGAIN || rc == EACCES || rc == EBUSY ||
+ rc == EWOULDBLOCK || rc == MDBX_RESULT_TRUE)
+ ? MDBX_SUCCESS
+ : rc;
}
+#endif
}
- while ((dp = env->me_dpages) != NULL) {
- ASAN_UNPOISON_MEMORY_REGION(&dp->mp_next, sizeof(dp->mp_next));
- VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
- env->me_dpages = dp->mp_next;
- mdbx_free(dp);
- }
- VALGRIND_DESTROY_MEMPOOL(env);
-
rc = mdbx_env_close0(env) ? MDBX_PANIC : rc;
mdbx_ensure(env, mdbx_fastmutex_destroy(&env->me_dbi_lock) == MDBX_SUCCESS);
#if defined(_WIN32) || defined(_WIN64)
@@ -10296,9 +10303,15 @@ int __cold mdbx_env_close_ex(MDBX_env *env, int dont_sync) {
mdbx_ensure(env, mdbx_ipclock_destroy(&env->me_lckless_stub.wlock) == 0);
#endif /* MDBX_LOCKING */
+ while ((dp = env->me_dpages) != NULL) {
+ ASAN_UNPOISON_MEMORY_REGION(&dp->mp_next, sizeof(dp->mp_next));
+ VALGRIND_MAKE_MEM_DEFINED(&dp->mp_next, sizeof(dp->mp_next));
+ env->me_dpages = dp->mp_next;
+ mdbx_free(dp);
+ }
+ VALGRIND_DESTROY_MEMPOOL(env);
mdbx_ensure(env, env->me_lcklist_next == nullptr);
env->me_pid = 0;
- env->me_signature = 0;
mdbx_free(env);
return rc;
@@ -10837,7 +10850,7 @@ static __always_inline int mdbx_node_read(MDBX_cursor *mc, MDBX_node *node,
return MDBX_SUCCESS;
}
-int mdbx_get(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data) {
+int mdbx_get(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key, MDBX_val *data) {
DKBUF;
mdbx_debug("===> get db %u key [%s]", dbi, DKEY(key));
@@ -10857,7 +10870,7 @@ int mdbx_get(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data) {
return rc;
int exact = 0;
- return mdbx_cursor_set(&cx.outer, key, data, MDBX_SET, &exact);
+ return mdbx_cursor_set(&cx.outer, (MDBX_val *)key, data, MDBX_SET, &exact);
}
int mdbx_get_nearest(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key,
@@ -11679,7 +11692,7 @@ static int mdbx_cursor_touch(MDBX_cursor *mc) {
return rc;
}
-int mdbx_cursor_put(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data,
+int mdbx_cursor_put(MDBX_cursor *mc, const MDBX_val *key, MDBX_val *data,
unsigned flags) {
MDBX_env *env;
MDBX_page *sub_root = NULL;
@@ -11829,7 +11842,7 @@ int mdbx_cursor_put(MDBX_cursor *mc, MDBX_val *key, MDBX_val *data,
}
}
} else {
- rc = mdbx_cursor_set(mc, key, &d2, MDBX_SET, &exact);
+ rc = mdbx_cursor_set(mc, (MDBX_val *)key, &d2, MDBX_SET, &exact);
}
if ((flags & MDBX_NOOVERWRITE) &&
(rc == MDBX_SUCCESS || rc == MDBX_EKEYMISMATCH)) {
@@ -12752,7 +12765,7 @@ static void mdbx_node_shrink(MDBX_page *mp, unsigned indx) {
/* Prepare to shift upward, set len = length(subpage part to shift) */
if (IS_LEAF2(sp)) {
- delta &= /* do not make the node uneven-sized */ ~1u;
+ delta &= /* do not make the node uneven-sized */ ~(size_t)1;
if (unlikely(delta) == 0)
return;
nsize = node_ds(node) - delta;
@@ -13003,7 +13016,7 @@ int mdbx_cursor_renew(MDBX_txn *txn, MDBX_cursor *mc) {
}
/* Return the count of duplicate data items for the current key */
-int mdbx_cursor_count(MDBX_cursor *mc, size_t *countp) {
+int mdbx_cursor_count(const MDBX_cursor *mc, size_t *countp) {
if (unlikely(mc == NULL))
return MDBX_EINVAL;
@@ -13067,7 +13080,7 @@ void mdbx_cursor_close(MDBX_cursor *mc) {
}
}
-MDBX_txn *mdbx_cursor_txn(MDBX_cursor *mc) {
+MDBX_txn *mdbx_cursor_txn(const MDBX_cursor *mc) {
if (unlikely(!mc || mc->mc_signature != MDBX_MC_SIGNATURE))
return NULL;
MDBX_txn *txn = mc->mc_txn;
@@ -13078,7 +13091,7 @@ MDBX_txn *mdbx_cursor_txn(MDBX_cursor *mc) {
return txn;
}
-MDBX_dbi mdbx_cursor_dbi(MDBX_cursor *mc) {
+MDBX_dbi mdbx_cursor_dbi(const MDBX_cursor *mc) {
if (unlikely(!mc || mc->mc_signature != MDBX_MC_SIGNATURE))
return UINT_MAX;
return mc->mc_dbi;
@@ -14304,7 +14317,8 @@ static int mdbx_cursor_del0(MDBX_cursor *mc) {
return rc;
}
-int mdbx_del(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data) {
+int mdbx_del(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
+ const MDBX_val *data) {
int rc = check_txn_rw(txn, MDBX_TXN_BLOCKED);
if (unlikely(rc != MDBX_SUCCESS))
return rc;
@@ -14321,8 +14335,8 @@ int mdbx_del(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data) {
return mdbx_del0(txn, dbi, key, data, 0);
}
-static int mdbx_del0(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data,
- unsigned flags) {
+static int mdbx_del0(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
+ const MDBX_val *data, unsigned flags) {
MDBX_cursor_couple cx;
MDBX_cursor_op op;
MDBX_val rdata;
@@ -14344,7 +14358,8 @@ static int mdbx_del0(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data,
op = MDBX_SET;
flags |= MDBX_NODUPDATA;
}
- rc = mdbx_cursor_set(&cx.outer, key, data, op, &exact);
+ rc =
+ mdbx_cursor_set(&cx.outer, (MDBX_val *)key, (MDBX_val *)data, op, &exact);
if (likely(rc == MDBX_SUCCESS)) {
/* let mdbx_page_split know about this cursor if needed:
* delete will trigger a rebalance; if it needs to move
@@ -14844,7 +14859,7 @@ done:
return rc;
}
-int mdbx_put(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data,
+int mdbx_put(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key, MDBX_val *data,
unsigned flags) {
int rc = check_txn_rw(txn, MDBX_TXN_BLOCKED);
if (unlikely(rc != MDBX_SUCCESS))
@@ -14872,7 +14887,7 @@ int mdbx_put(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *data,
/* LY: support for update (explicit overwrite) */
if (flags & MDBX_CURRENT) {
- rc = mdbx_cursor_get(&cx.outer, key, NULL, MDBX_SET);
+ rc = mdbx_cursor_get(&cx.outer, (MDBX_val *)key, NULL, MDBX_SET);
if (likely(rc == MDBX_SUCCESS) &&
(txn->mt_dbs[dbi].md_flags & MDBX_DUPSORT)) {
/* LY: allows update (explicit overwrite) only for unique keys */
@@ -15177,7 +15192,7 @@ static int __cold mdbx_env_compact(MDBX_env *env, MDBX_txn *read_txn,
const bool dest_is_pipe) {
const size_t meta_bytes = pgno2bytes(env, NUM_METAS);
uint8_t *const data_buffer =
- buffer + roundup_powerof2(meta_bytes, env->me_os_psize);
+ buffer + ceil_powerof2(meta_bytes, env->me_os_psize);
MDBX_meta *const meta = mdbx_init_metas(env, buffer);
/* copy canary sequenses if present */
if (read_txn->mt_canary.v) {
@@ -15343,10 +15358,12 @@ static int __cold mdbx_env_copy_asis(MDBX_env *env, MDBX_txn *read_txn,
rc = mdbx_write(fd, buffer, meta_bytes);
uint8_t *const data_buffer =
- buffer + roundup_powerof2(meta_bytes, env->me_os_psize);
+ buffer + ceil_powerof2(meta_bytes, env->me_os_psize);
for (size_t offset = meta_bytes; rc == MDBX_SUCCESS && offset < used_size;) {
if (dest_is_pipe) {
-#if defined(__linux__) || defined(__gnu_linux__) && !defined(MDBX_SAFE4QEMU)
+#if (defined(__linux__) || defined(__gnu_linux__)) && \
+ (!defined(__ANDROID_API__) || __ANDROID_API__ >= 21) && \
+ !defined(MDBX_SAFE4QEMU)
off_t in_offset = offset;
const intptr_t written =
sendfile(fd, env->me_lazy_fd, &in_offset, used_size - offset);
@@ -15420,8 +15437,8 @@ int __cold mdbx_env_copy2fd(MDBX_env *env, mdbx_filehandle_t fd,
const size_t buffer_size =
pgno_align2os_bytes(env, NUM_METAS) +
- roundup_powerof2(((flags & MDBX_CP_COMPACT) ? MDBX_WBUF * 2 : MDBX_WBUF),
- env->me_os_psize);
+ ceil_powerof2(((flags & MDBX_CP_COMPACT) ? MDBX_WBUF * 2 : MDBX_WBUF),
+ env->me_os_psize);
uint8_t *buffer = NULL;
int rc = mdbx_memalign_alloc(env->me_os_psize, buffer_size, (void **)&buffer);
@@ -15531,7 +15548,7 @@ int __cold mdbx_env_set_flags(MDBX_env *env, unsigned flags, int onoff) {
return MDBX_SUCCESS;
}
-int __cold mdbx_env_get_flags(MDBX_env *env, unsigned *arg) {
+int __cold mdbx_env_get_flags(const MDBX_env *env, unsigned *arg) {
if (unlikely(!env || !arg))
return MDBX_EINVAL;
@@ -15553,7 +15570,7 @@ int __cold mdbx_env_set_userctx(MDBX_env *env, void *ctx) {
return MDBX_SUCCESS;
}
-void *__cold mdbx_env_get_userctx(MDBX_env *env) {
+void *__cold mdbx_env_get_userctx(const MDBX_env *env) {
return env ? env->me_userctx : NULL;
}
@@ -15573,7 +15590,7 @@ int __cold mdbx_env_set_assert(MDBX_env *env, MDBX_assert_func *func) {
#endif
}
-int __cold mdbx_env_get_path(MDBX_env *env, const char **arg) {
+int __cold mdbx_env_get_path(const MDBX_env *env, const char **arg) {
if (unlikely(!env || !arg))
return MDBX_EINVAL;
@@ -15584,7 +15601,7 @@ int __cold mdbx_env_get_path(MDBX_env *env, const char **arg) {
return MDBX_SUCCESS;
}
-int __cold mdbx_env_get_fd(MDBX_env *env, mdbx_filehandle_t *arg) {
+int __cold mdbx_env_get_fd(const MDBX_env *env, mdbx_filehandle_t *arg) {
if (unlikely(!env || !arg))
return MDBX_EINVAL;
@@ -16310,7 +16327,7 @@ int mdbx_set_dupsort(MDBX_txn *txn, MDBX_dbi dbi, MDBX_cmp_func *cmp) {
return MDBX_SUCCESS;
}
-int __cold mdbx_reader_list(MDBX_env *env, MDBX_reader_list_func *func,
+int __cold mdbx_reader_list(const MDBX_env *env, MDBX_reader_list_func *func,
void *ctx) {
if (unlikely(!env || !func))
return MDBX_EINVAL;
@@ -16701,7 +16718,7 @@ int __cold mdbx_env_set_oomfunc(MDBX_env *env, MDBX_oom_func *oomfunc) {
return MDBX_SUCCESS;
}
-MDBX_oom_func *__cold mdbx_env_get_oomfunc(MDBX_env *env) {
+MDBX_oom_func *__cold mdbx_env_get_oomfunc(const MDBX_env *env) {
return likely(env && env->me_signature == MDBX_ME_SIGNATURE)
? env->me_oom_func
: NULL;
@@ -16711,7 +16728,7 @@ MDBX_oom_func *__cold mdbx_env_get_oomfunc(MDBX_env *env) {
/* LY: avoid tsan-trap by me_txn, mm_last_pg and mt_next_pgno */
__attribute__((__no_sanitize_thread__, __noinline__))
#endif
-int mdbx_txn_straggler(MDBX_txn *txn, int *percent)
+int mdbx_txn_straggler(const MDBX_txn *txn, int *percent)
{
int rc = check_txn(txn, MDBX_TXN_BLOCKED);
if (unlikely(rc != MDBX_SUCCESS))
@@ -17022,7 +17039,7 @@ int mdbx_canary_put(MDBX_txn *txn, const mdbx_canary *canary) {
return MDBX_SUCCESS;
}
-int mdbx_canary_get(MDBX_txn *txn, mdbx_canary *canary) {
+int mdbx_canary_get(const MDBX_txn *txn, mdbx_canary *canary) {
int rc = check_txn(txn, MDBX_TXN_BLOCKED);
if (unlikely(rc != MDBX_SUCCESS))
return rc;
@@ -17034,7 +17051,7 @@ int mdbx_canary_get(MDBX_txn *txn, mdbx_canary *canary) {
return MDBX_SUCCESS;
}
-int mdbx_cursor_on_first(MDBX_cursor *mc) {
+int mdbx_cursor_on_first(const MDBX_cursor *mc) {
if (unlikely(mc == NULL))
return MDBX_EINVAL;
@@ -17052,7 +17069,7 @@ int mdbx_cursor_on_first(MDBX_cursor *mc) {
return MDBX_RESULT_TRUE;
}
-int mdbx_cursor_on_last(MDBX_cursor *mc) {
+int mdbx_cursor_on_last(const MDBX_cursor *mc) {
if (unlikely(mc == NULL))
return MDBX_EINVAL;
@@ -17071,7 +17088,7 @@ int mdbx_cursor_on_last(MDBX_cursor *mc) {
return MDBX_RESULT_TRUE;
}
-int mdbx_cursor_eof(MDBX_cursor *mc) {
+int mdbx_cursor_eof(const MDBX_cursor *mc) {
if (unlikely(mc == NULL))
return MDBX_EINVAL;
@@ -17503,8 +17520,8 @@ int mdbx_estimate_range(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *begin_key,
* - внешняя аллокация курсоров, в том числе на стеке (без malloc).
* - получения статуса страницы по адресу (знать о P_DIRTY).
*/
-int mdbx_replace(MDBX_txn *txn, MDBX_dbi dbi, MDBX_val *key, MDBX_val *new_data,
- MDBX_val *old_data, unsigned flags) {
+int mdbx_replace(MDBX_txn *txn, MDBX_dbi dbi, const MDBX_val *key,
+ MDBX_val *new_data, MDBX_val *old_data, unsigned flags) {
int rc = check_txn_rw(txn, MDBX_TXN_BLOCKED);
if (unlikely(rc != MDBX_SUCCESS))
return rc;
@@ -18086,8 +18103,8 @@ __dll_export
#ifdef MDBX_BUILD_TARGET
MDBX_BUILD_TARGET
#else
- #if defined(__ANDROID__)
- "Android"
+ #if defined(__ANDROID_API__)
+ "Android" STRINGIFY(__ANDROID_API__)
#elif defined(__linux__) || defined(__gnu_linux__)
"Linux"
#elif defined(EMSCRIPTEN) || defined(__EMSCRIPTEN__)
@@ -18177,12 +18194,12 @@ __dll_export
#endif
#endif /* MDBX_BUILD_TARGET */
-#ifdef MDBX_BUILD_CONFIG
+#ifdef MDBX_BUILD_TYPE
# if defined(_MSC_VER)
-# pragma message("Configuration-depended MDBX_BUILD_CONFIG: " MDBX_BUILD_CONFIG)
+# pragma message("Configuration-depended MDBX_BUILD_TYPE: " MDBX_BUILD_TYPE)
# endif
- "-" MDBX_BUILD_CONFIG
-#endif /* MDBX_BUILD_CONFIG */
+ "-" MDBX_BUILD_TYPE
+#endif /* MDBX_BUILD_TYPE */
,
"MDBX_DEBUG=" STRINGIFY(MDBX_DEBUG)
#ifdef MDBX_LOGLEVEL_BUILD
@@ -18267,12 +18284,16 @@ __dll_export
#endif
#endif /* MDBX_BUILD_COMPILER */
,
-#ifdef MDBX_BUILD_FLAGS
- MDBX_BUILD_FLAGS
-#endif /* MDBX_BUILD_FLAGS */
#ifdef MDBX_BUILD_FLAGS_CONFIG
MDBX_BUILD_FLAGS_CONFIG
#endif /* MDBX_BUILD_FLAGS_CONFIG */
+#ifdef MDBX_BUILD_FLAGS
+ MDBX_BUILD_FLAGS
+#endif /* MDBX_BUILD_FLAGS */
+#if !(defined(MDBX_BUILD_FLAGS_CONFIG) || defined(MDBX_BUILD_FLAGS))
+ "undefined (please use correct build script)"
+#warning "Build flags undefined. Please use correct build script"
+#endif
};
#ifdef __SANITIZE_ADDRESS__
diff --git a/libs/libmdbx/src/src/elements/debug_begin.h b/libs/libmdbx/src/src/debug_begin.h
index f3306b5dc0..f3306b5dc0 100644
--- a/libs/libmdbx/src/src/elements/debug_begin.h
+++ b/libs/libmdbx/src/src/debug_begin.h
diff --git a/libs/libmdbx/src/src/elements/debug_end.h b/libs/libmdbx/src/src/debug_end.h
index e361119bc0..e361119bc0 100644
--- a/libs/libmdbx/src/src/elements/debug_end.h
+++ b/libs/libmdbx/src/src/debug_end.h
diff --git a/libs/libmdbx/src/src/elements/defs.h b/libs/libmdbx/src/src/defs.h
index b2710eede8..b2710eede8 100644
--- a/libs/libmdbx/src/src/elements/defs.h
+++ b/libs/libmdbx/src/src/defs.h
diff --git a/libs/libmdbx/src/src/elements/internals.h b/libs/libmdbx/src/src/internals.h
index 62552de95e..7ef818ca35 100644
--- a/libs/libmdbx/src/src/elements/internals.h
+++ b/libs/libmdbx/src/src/internals.h
@@ -13,7 +13,7 @@
#pragma once
#ifdef MDBX_CONFIG_H
-#include MDBX_CONFIG_H
+#include "../config.h"
#endif
/* *INDENT-OFF* */
@@ -90,7 +90,7 @@
#pragma warning(disable : 4200) /* nonstandard extension used: zero-sized array in struct/union */
#endif /* _MSC_VER (warnings) */
-#include "../../mdbx.h"
+#include "../mdbx.h"
#include "defs.h"
#if defined(__GNUC__) && !__GNUC_PREREQ(4,2)
@@ -862,15 +862,7 @@ typedef struct MDBX_cursor_couple {
/* The database environment. */
struct MDBX_env {
#define MDBX_ME_SIGNATURE UINT32_C(0x9A899641)
- size_t me_signature;
- mdbx_mmap_t me_dxb_mmap; /* The main data file */
-#define me_map me_dxb_mmap.dxb
-#define me_lazy_fd me_dxb_mmap.fd
- mdbx_filehandle_t me_dsync_fd;
- mdbx_mmap_t me_lck_mmap; /* The lock file */
-#define me_lfd me_lck_mmap.fd
-#define me_lck me_lck_mmap.lck
-
+ uint32_t me_signature;
/* Failed to update the meta page. Probably an I/O error. */
#define MDBX_FATAL_ERROR UINT32_C(0x80000000)
/* Additional flag for mdbx_sync_locked() */
@@ -879,7 +871,15 @@ struct MDBX_env {
#define MDBX_ENV_ACTIVE UINT32_C(0x20000000)
/* me_txkey is set */
#define MDBX_ENV_TXKEY UINT32_C(0x10000000)
- uint32_t me_flags; /* see mdbx_env */
+ uint32_t me_flags;
+ mdbx_mmap_t me_dxb_mmap; /* The main data file */
+#define me_map me_dxb_mmap.dxb
+#define me_lazy_fd me_dxb_mmap.fd
+ mdbx_filehandle_t me_dsync_fd;
+ mdbx_mmap_t me_lck_mmap; /* The lock file */
+#define me_lfd me_lck_mmap.fd
+#define me_lck me_lck_mmap.lck
+
unsigned me_psize; /* DB page size, inited from me_os_psize */
unsigned me_psize2log; /* log2 of DB page size */
unsigned me_os_psize; /* OS page size, from mdbx_syspagesize() */
@@ -991,8 +991,6 @@ MDBX_INTERNAL_FUNC void mdbx_debug_log(int type, const char *function, int line,
const char *fmt, ...)
__printf_args(4, 5);
-MDBX_INTERNAL_FUNC void mdbx_panic(const char *fmt, ...) __printf_args(1, 2);
-
#if MDBX_DEBUG
#define mdbx_assert_enabled() unlikely(mdbx_runtime_flags &MDBX_DBG_ASSERT)
@@ -1024,8 +1022,20 @@ MDBX_INTERNAL_FUNC void mdbx_panic(const char *fmt, ...) __printf_args(1, 2);
#endif /* MDBX_DEBUG */
+#if defined(__ANDROID_API__)
+#define mdbx_panic(fmt, ...) \
+ __android_log_assert("panic", "mdbx", fmt, __VA_ARGS__)
+#else
+MDBX_INTERNAL_FUNC void mdbx_panic(const char *fmt, ...) __printf_args(1, 2);
+#endif
+
+#if !MDBX_DEBUG && defined(__ANDROID_API__)
+#define mdbx_assert_fail(env, msg, func, line) \
+ __android_log_assert(msg, "mdbx", "%s:%u", func, line)
+#else
MDBX_INTERNAL_FUNC void mdbx_assert_fail(const MDBX_env *env, const char *msg,
const char *func, int line);
+#endif
#define mdbx_debug_extra(fmt, ...) \
do { \
@@ -1325,7 +1335,12 @@ is_powerof2(size_t x) {
}
static __pure_function __always_inline __maybe_unused size_t
-roundup_powerof2(size_t value, size_t granularity) {
+floor_powerof2(size_t value, size_t granularity) {
assert(is_powerof2(granularity));
- return (value + granularity - 1) & ~(granularity - 1);
+ return value & ~(granularity - 1);
+}
+
+static __pure_function __always_inline __maybe_unused size_t
+ceil_powerof2(size_t value, size_t granularity) {
+ return floor_powerof2(value + granularity - 1, granularity);
}
diff --git a/libs/libmdbx/src/src/elements/lck-posix.c b/libs/libmdbx/src/src/lck-posix.c
index 3754c25393..6b6127bd4b 100644
--- a/libs/libmdbx/src/src/elements/lck-posix.c
+++ b/libs/libmdbx/src/src/lck-posix.c
@@ -21,10 +21,20 @@
/* global constructor/destructor */
#if defined(__linux__) || defined(__gnu_linux__)
+
#include <sys/utsname.h>
+
#ifndef MDBX_ALLOY
uint32_t mdbx_linux_kernel_version;
+bool mdbx_RunningOnWSL;
#endif /* MDBX_ALLOY */
+
+static __cold bool probe_for_WSL(const char *tag) {
+ /* "Official" way of detecting WSL but not WSL2
+ * https://github.com/Microsoft/WSL/issues/423#issuecomment-221627364 */
+ return strstr(tag, "Microsoft") || strstr(tag, "WSL");
+}
+
#endif /* Linux */
static __cold __attribute__((__constructor__)) void
@@ -32,6 +42,9 @@ mdbx_global_constructor(void) {
#if defined(__linux__) || defined(__gnu_linux__)
struct utsname buffer;
if (uname(&buffer) == 0) {
+ mdbx_RunningOnWSL = probe_for_WSL(buffer.version) ||
+ probe_for_WSL(buffer.sysname) ||
+ probe_for_WSL(buffer.release);
int i = 0;
char *p = buffer.release;
while (*p && i < 4) {
@@ -123,7 +136,8 @@ static void __cold choice_fcntl() {
#if defined(__linux__) || defined(__gnu_linux__)
&& mdbx_linux_kernel_version >
0x030f0000 /* OFD locks are available since 3.15, but engages here
- only for 3.16 and larer kernels (LTS) for reliability reasons */
+ only for 3.16 and later kernels (i.e. LTS) because
+ of reliability reasons */
#endif /* linux */
) {
op_setlk = F_OFD_SETLK;
@@ -291,6 +305,17 @@ MDBX_INTERNAL_FUNC int __cold mdbx_lck_seize(MDBX_env *env) {
#endif /* MDBX_USE_OFDLOCKS */
int rc = MDBX_SUCCESS;
+#if defined(__linux__) || defined(__gnu_linux__)
+ if (unlikely(mdbx_RunningOnWSL)) {
+ rc = ENOLCK /* No record locks available */;
+ mdbx_error("%s, err %u",
+ "WSL (Windows Subsystem for Linux) is mad and trouble-full, "
+ "injecting failure to avoid data loss",
+ rc);
+ return rc;
+ }
+#endif /* Linux */
+
if (env->me_lfd == INVALID_HANDLE_VALUE) {
/* LY: without-lck mode (e.g. exclusive or on read-only filesystem) */
rc =
diff --git a/libs/libmdbx/src/src/elements/lck-windows.c b/libs/libmdbx/src/src/lck-windows.c
index 13178139ea..19a71cc337 100644
--- a/libs/libmdbx/src/src/elements/lck-windows.c
+++ b/libs/libmdbx/src/src/lck-windows.c
@@ -243,13 +243,13 @@ static int suspend_and_append(mdbx_handle_array_t **array,
MDBX_INTERNAL_FUNC int
mdbx_suspend_threads_before_remap(MDBX_env *env, mdbx_handle_array_t **array) {
- const size_t CurrentTid = GetCurrentThreadId();
+ const uintptr_t CurrentTid = GetCurrentThreadId();
int rc;
if (env->me_lck) {
/* Scan LCK for threads of the current process */
const MDBX_reader *const begin = env->me_lck->mti_readers;
const MDBX_reader *const end = begin + env->me_lck->mti_numreaders;
- const size_t WriteTxnOwner = env->me_txn0 ? env->me_txn0->mt_owner : 0;
+ const uintptr_t WriteTxnOwner = env->me_txn0 ? env->me_txn0->mt_owner : 0;
for (const MDBX_reader *reader = begin; reader < end; ++reader) {
if (reader->mr_pid != env->me_pid || !reader->mr_tid) {
skip_lck:
diff --git a/libs/libmdbx/src/src/man1/mdbx_chk.1 b/libs/libmdbx/src/src/man1/mdbx_chk.1
index 4cd0a3a838..d2d03684e2 100644
--- a/libs/libmdbx/src/src/man1/mdbx_chk.1
+++ b/libs/libmdbx/src/src/man1/mdbx_chk.1
@@ -1,6 +1,6 @@
.\" Copyright 2015-2020 Leonid Yuriev <leo@yuriev.ru>.
.\" Copying restrictions apply. See COPYRIGHT/LICENSE.
-.TH MDBX_CHK 1 "2020-01-20" "MDBX 0.6.x"
+.TH MDBX_CHK 1 "2020-03-18" "MDBX 0.7.x"
.SH NAME
mdbx_chk \- MDBX checking tool
.SH SYNOPSIS
diff --git a/libs/libmdbx/src/src/man1/mdbx_copy.1 b/libs/libmdbx/src/src/man1/mdbx_copy.1
index 863a58add5..a8427bb4df 100644
--- a/libs/libmdbx/src/src/man1/mdbx_copy.1
+++ b/libs/libmdbx/src/src/man1/mdbx_copy.1
@@ -2,7 +2,7 @@
.\" Copyright 2012-2015 Howard Chu, Symas Corp. All Rights Reserved.
.\" Copyright 2015,2016 Peter-Service R&D LLC <http://billing.ru/>.
.\" Copying restrictions apply. See COPYRIGHT/LICENSE.
-.TH MDBX_COPY 1 "2020-01-20" "MDBX 0.6.x"
+.TH MDBX_COPY 1 "2020-03-18" "MDBX 0.7.x"
.SH NAME
mdbx_copy \- MDBX environment copy tool
.SH SYNOPSIS
diff --git a/libs/libmdbx/src/src/man1/mdbx_dump.1 b/libs/libmdbx/src/src/man1/mdbx_dump.1
index 7cf708ab94..9985127f00 100644
--- a/libs/libmdbx/src/src/man1/mdbx_dump.1
+++ b/libs/libmdbx/src/src/man1/mdbx_dump.1
@@ -2,7 +2,7 @@
.\" Copyright 2014-2015 Howard Chu, Symas Corp. All Rights Reserved.
.\" Copyright 2015,2016 Peter-Service R&D LLC <http://billing.ru/>.
.\" Copying restrictions apply. See COPYRIGHT/LICENSE.
-.TH MDBX_DUMP 1 "2020-01-20" "MDBX 0.6.x"
+.TH MDBX_DUMP 1 "2020-03-18" "MDBX 0.7.x"
.SH NAME
mdbx_dump \- MDBX environment export tool
.SH SYNOPSIS
diff --git a/libs/libmdbx/src/src/man1/mdbx_load.1 b/libs/libmdbx/src/src/man1/mdbx_load.1
index 473f9303ef..b401b38715 100644
--- a/libs/libmdbx/src/src/man1/mdbx_load.1
+++ b/libs/libmdbx/src/src/man1/mdbx_load.1
@@ -2,7 +2,7 @@
.\" Copyright 2014-2015 Howard Chu, Symas Corp. All Rights Reserved.
.\" Copyright 2015,2016 Peter-Service R&D LLC <http://billing.ru/>.
.\" Copying restrictions apply. See COPYRIGHT/LICENSE.
-.TH MDBX_LOAD 1 "2020-01-20" "MDBX 0.6.x"
+.TH MDBX_LOAD 1 "2020-03-18" "MDBX 0.7.x"
.SH NAME
mdbx_load \- MDBX environment import tool
.SH SYNOPSIS
diff --git a/libs/libmdbx/src/src/man1/mdbx_stat.1 b/libs/libmdbx/src/src/man1/mdbx_stat.1
index 80969bb3f1..b7fbeb0833 100644
--- a/libs/libmdbx/src/src/man1/mdbx_stat.1
+++ b/libs/libmdbx/src/src/man1/mdbx_stat.1
@@ -2,7 +2,7 @@
.\" Copyright 2012-2015 Howard Chu, Symas Corp. All Rights Reserved.
.\" Copyright 2015,2016 Peter-Service R&D LLC <http://billing.ru/>.
.\" Copying restrictions apply. See COPYRIGHT/LICENSE.
-.TH MDBX_STAT 1 "2020-01-20" "MDBX 0.6.x"
+.TH MDBX_STAT 1 "2020-03-18" "MDBX 0.7.x"
.SH NAME
mdbx_stat \- MDBX environment status tool
.SH SYNOPSIS
diff --git a/libs/libmdbx/src/src/tools/mdbx_chk.c b/libs/libmdbx/src/src/mdbx_chk.c
index e5dd539af5..64c1d333bc 100644
--- a/libs/libmdbx/src/src/tools/mdbx_chk.c
+++ b/libs/libmdbx/src/src/mdbx_chk.c
@@ -21,7 +21,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../elements/internals.h"
+#include "internals.h"
typedef struct flagbit {
int bit;
@@ -559,6 +559,12 @@ static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler,
uint64_t record_count = 0, dups = 0;
uint64_t key_bytes = 0, data_bytes = 0;
+ if ((MDBX_TXN_FINISHED | MDBX_TXN_ERROR) & mdbx_txn_flags(txn)) {
+ print(" ! abort processing '%s' due to a previous error\n",
+ dbi_name ? dbi_name : "@MAIN");
+ return MDBX_BAD_TXN;
+ }
+
if (dbi_handle == ~0u) {
rc = mdbx_dbi_open(txn, dbi_name, 0, &dbi_handle);
if (rc) {
diff --git a/libs/libmdbx/src/src/tools/mdbx_copy.c b/libs/libmdbx/src/src/mdbx_copy.c
index 8545b38809..81d177ab99 100644
--- a/libs/libmdbx/src/src/tools/mdbx_copy.c
+++ b/libs/libmdbx/src/src/mdbx_copy.c
@@ -21,7 +21,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../elements/internals.h"
+#include "internals.h"
#if defined(_WIN32) || defined(_WIN64)
#include "wingetopt.h"
diff --git a/libs/libmdbx/src/src/tools/mdbx_dump.c b/libs/libmdbx/src/src/mdbx_dump.c
index 3afabc0b4a..8bb057b2df 100644
--- a/libs/libmdbx/src/src/tools/mdbx_dump.c
+++ b/libs/libmdbx/src/src/mdbx_dump.c
@@ -21,7 +21,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../elements/internals.h"
+#include "internals.h"
#include <ctype.h>
diff --git a/libs/libmdbx/src/src/tools/mdbx_load.c b/libs/libmdbx/src/src/mdbx_load.c
index 449848e7b8..eb39cc1174 100644
--- a/libs/libmdbx/src/src/tools/mdbx_load.c
+++ b/libs/libmdbx/src/src/mdbx_load.c
@@ -21,7 +21,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../elements/internals.h"
+#include "internals.h"
#include <ctype.h>
@@ -339,7 +339,8 @@ static void usage(void) {
"dbpath\n"
" -V\t\tprint version and exit\n"
" -q\t\tbe quiet\n"
- " -a\t\tappend records in input order\n"
+ " -a\t\tappend records in input order (required for custom "
+ "comparators)\n"
" -f file\tread from file instead of stdin\n"
" -s name\tload into named subDB\n"
" -N\t\tuse NOOVERWRITE on puts\n"
diff --git a/libs/libmdbx/src/src/tools/mdbx_stat.c b/libs/libmdbx/src/src/mdbx_stat.c
index 0867aedac3..d1a805f925 100644
--- a/libs/libmdbx/src/src/tools/mdbx_stat.c
+++ b/libs/libmdbx/src/src/mdbx_stat.c
@@ -21,7 +21,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../elements/internals.h"
+#include "internals.h"
#if defined(_WIN32) || defined(_WIN64)
#include "wingetopt.h"
@@ -77,8 +77,8 @@ static int reader_list_func(void *ctx, int num, int slot, mdbx_pid_t pid,
"pid", (int)sizeof(size_t) * 2, "thread", "txnid", "lag", "used",
"retained");
- printf(" %3d)\t[%d]\t%6" PRIdSIZE " %*" PRIxSIZE, num, slot, (size_t)pid,
- (int)sizeof(size_t) * 2, (size_t)thread);
+ printf(" %3d)\t[%d]\t%6" PRIdSIZE " %*" PRIxPTR, num, slot, (size_t)pid,
+ (int)sizeof(size_t) * 2, (uintptr_t)thread);
if (txnid)
printf(" %20" PRIu64 " %10" PRIu64 " %12.1fM %12.1fM\n", txnid, lag,
bytes_used / 1048576.0, bytes_retained / 1048576.0);
@@ -294,7 +294,7 @@ int main(int argc, char *argv[]) {
const pgno_t number = *iptr++;
pages += number;
- if (envinfo && mei.mi_latter_reader_txnid > *(size_t *)key.iov_base)
+ if (envinfo && mei.mi_latter_reader_txnid > *(txnid_t *)key.iov_base)
reclaimable += number;
if (freinfo > 1) {
diff --git a/libs/libmdbx/src/src/elements/ntdll.def b/libs/libmdbx/src/src/ntdll.def
index e3a6e33c94..e3a6e33c94 100644
--- a/libs/libmdbx/src/src/elements/ntdll.def
+++ b/libs/libmdbx/src/src/ntdll.def
diff --git a/libs/libmdbx/src/src/elements/options.h b/libs/libmdbx/src/src/options.h
index 2694e92011..715f883bfc 100644
--- a/libs/libmdbx/src/src/elements/options.h
+++ b/libs/libmdbx/src/src/options.h
@@ -79,8 +79,10 @@
/* Some platforms define the EOWNERDEAD error code even though they
* don't support Robust Mutexes. If doubt compile with -MDBX_LOCKING=2001. */
#if defined(EOWNERDEAD) && _POSIX_THREAD_PROCESS_SHARED >= 200809L && \
- (defined(_POSIX_THREAD_ROBUST_PRIO_INHERIT) || \
- defined(_POSIX_THREAD_ROBUST_PRIO_PROTECT) || \
+ ((defined(_POSIX_THREAD_ROBUST_PRIO_INHERIT) && \
+ _POSIX_THREAD_ROBUST_PRIO_INHERIT > 0) || \
+ (defined(_POSIX_THREAD_ROBUST_PRIO_PROTECT) && \
+ _POSIX_THREAD_ROBUST_PRIO_PROTECT > 0) || \
defined(PTHREAD_MUTEX_ROBUST) || defined(PTHREAD_MUTEX_ROBUST_NP)) && \
(!defined(__GLIBC__) || \
__GLIBC_PREREQ(2, 10) /* troubles with Robust mutexes before 2.10 */)
diff --git a/libs/libmdbx/src/src/elements/osal.c b/libs/libmdbx/src/src/osal.c
index 06a448cdbc..94902527b1 100644
--- a/libs/libmdbx/src/src/elements/osal.c
+++ b/libs/libmdbx/src/src/osal.c
@@ -197,6 +197,8 @@ __extern_C void __assert(const char *function, const char *file, int line,
#endif /* __assert_fail */
+#if !defined(__ANDROID_API__) || MDBX_DEBUG
+
MDBX_INTERNAL_FUNC void __cold mdbx_assert_fail(const MDBX_env *env,
const char *msg,
const char *func, int line) {
@@ -221,6 +223,8 @@ MDBX_INTERNAL_FUNC void __cold mdbx_assert_fail(const MDBX_env *env,
OutputDebugStringA(message);
if (IsDebuggerPresent())
DebugBreak();
+#elif defined(__ANDROID_API__)
+ __android_log_assert(msg, "mdbx", "%s:%u", func, line);
#else
__assert_fail(msg, "mdbx", line, func);
#endif
@@ -233,6 +237,10 @@ MDBX_INTERNAL_FUNC void __cold mdbx_assert_fail(const MDBX_env *env,
#endif
}
+#endif /* __ANDROID_API__ || MDBX_DEBUG */
+
+#if !defined(__ANDROID_API__)
+
MDBX_INTERNAL_FUNC __cold void mdbx_panic(const char *fmt, ...) {
va_list ap;
va_start(ap, fmt);
@@ -256,6 +264,8 @@ MDBX_INTERNAL_FUNC __cold void mdbx_panic(const char *fmt, ...) {
#endif
}
+#endif /* ! __ANDROID_API__ */
+
/*----------------------------------------------------------------------------*/
#ifndef mdbx_vasprintf
@@ -313,9 +323,10 @@ MDBX_INTERNAL_FUNC int mdbx_memalign_alloc(size_t alignment, size_t bytes,
*result = VirtualAlloc(NULL, bytes, MEM_COMMIT | MEM_RESERVE, PAGE_READWRITE);
return *result ? MDBX_SUCCESS : MDBX_ENOMEM /* ERROR_OUTOFMEMORY */;
#elif defined(_ISOC11_SOURCE)
- *result = aligned_alloc(alignment, roundup_powerof2(bytes, alignment));
+ *result = aligned_alloc(alignment, ceil_powerof2(bytes, alignment));
return *result ? MDBX_SUCCESS : errno;
-#elif _POSIX_VERSION >= 200112L
+#elif _POSIX_VERSION >= 200112L && \
+ (!defined(__ANDROID_API__) || __ANDROID_API__ >= 17)
*result = nullptr;
return posix_memalign(result, alignment, bytes);
#elif __GLIBC_PREREQ(2, 16) || __STDC_VERSION__ >= 201112L
@@ -753,7 +764,8 @@ MDBX_INTERNAL_FUNC int mdbx_write(mdbx_filehandle_t fd, const void *buf,
int mdbx_pwritev(mdbx_filehandle_t fd, struct iovec *iov, int iovcnt,
uint64_t offset, size_t expected_written) {
-#if defined(_WIN32) || defined(_WIN64) || defined(__APPLE__)
+#if defined(_WIN32) || defined(_WIN64) || defined(__APPLE__) || \
+ (defined(__ANDROID_API__) && __ANDROID_API__ < 24)
size_t written = 0;
for (int i = 0; i < iovcnt; ++i) {
int rc = mdbx_pwrite(fd, iov[i].iov_base, iov[i].iov_len, offset);
@@ -1098,6 +1110,7 @@ static int mdbx_check_fs_local(mdbx_filehandle_t handle, int flags) {
mdbx_free(PathBuffer);
return rc;
}
+
#else
struct statvfs statvfs_info;
@@ -1150,6 +1163,9 @@ static int mdbx_check_fs_local(mdbx_filehandle_t handle, int flags) {
defined(MFSTYPENAMELEN) || defined(VFS_NAMELEN)
const char *const name = statfs_info.f_fstypename;
const size_t name_len = sizeof(statfs_info.f_fstypename);
+#elif defined(__ANDROID_API__) && __ANDROID_API__ < 21
+ const char *const name = "";
+ const unsigned name_len = 0;
#else
const char *name = "";
@@ -1168,7 +1184,7 @@ static int mdbx_check_fs_local(mdbx_filehandle_t handle, int flags) {
mounted = setmntent("/etc/mtab", "r");
if (mounted) {
const struct mntent *ent;
-#if defined(_BSD_SOURCE) || defined(_SVID_SOURCE) || \
+#if defined(_BSD_SOURCE) || defined(_SVID_SOURCE) || defined(__BIONIC__) || \
(defined(_DEFAULT_SOURCE) && __GLIBC_PREREQ(2, 19))
struct mntent entbuf;
const bool should_copy = false;
@@ -1176,7 +1192,7 @@ static int mdbx_check_fs_local(mdbx_filehandle_t handle, int flags) {
(ent = getmntent_r(mounted, &entbuf, pathbuf, sizeof(pathbuf))))
#else
const bool should_copy = true;
- while (nullptr != (ent = getmntent(mounted))))
+ while (nullptr != (ent = getmntent(mounted)))
#endif
{
struct stat mnt;
@@ -1194,7 +1210,7 @@ static int mdbx_check_fs_local(mdbx_filehandle_t handle, int flags) {
}
endmntent(mounted);
}
-#endif /* !xBSD */
+#endif /* !xBSD && !Android/Bionic */
#endif
if (name_len) {
@@ -1376,7 +1392,10 @@ MDBX_INTERNAL_FUNC int mdbx_mmap(const int flags, mdbx_mmap_t *map,
MDBX_INTERNAL_FUNC int mdbx_munmap(mdbx_mmap_t *map) {
VALGRIND_MAKE_MEM_NOACCESS(map->address, map->current);
- ASAN_POISON_MEMORY_REGION(map->address, map->current);
+ /* Unpoisoning is required for ASAN to avoid false-positive diagnostic
+ * when this memory will re-used by malloc or another mmaping.
+ * See https://github.com/erthink/libmdbx/pull/93#issuecomment-613687203 */
+ ASAN_UNPOISON_MEMORY_REGION(map->address, map->limit);
#if defined(_WIN32) || defined(_WIN64)
if (map->section)
NtClose(map->section);
diff --git a/libs/libmdbx/src/src/elements/osal.h b/libs/libmdbx/src/src/osal.h
index 6e423e98dd..d9012ce768 100644
--- a/libs/libmdbx/src/src/elements/osal.h
+++ b/libs/libmdbx/src/src/osal.h
@@ -76,6 +76,10 @@ extern "C" {
/*----------------------------------------------------------------------------*/
/* Systems includes */
+#ifdef __APPLE__
+#include <TargetConditionals.h>
+#endif /* Apple OSX & iOS */
+
#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || \
defined(__BSD__) || defined(__bsdi__) || defined(__DragonFly__) || \
defined(__APPLE__) || defined(__MACH__)
@@ -90,7 +94,9 @@ extern "C" {
#else
#define SYSCTL_LEGACY_NONCONST_MIB
#endif
+#ifndef __MACH__
#include <sys/vmmeter.h>
+#endif
#else
#include <malloc.h>
#if !(defined(__sun) || defined(__SVR4) || defined(__svr4__) || \
@@ -238,6 +244,13 @@ typedef pthread_mutex_t mdbx_fastmutex_t;
#define malloc_usable_size(ptr) _msize(ptr)
#endif /* malloc_usable_size */
+#ifdef __ANDROID_API__
+#include <android/log.h>
+#if __ANDROID_API__ >= 21
+#include <sys/sendfile.h>
+#endif
+#endif /* Android */
+
/* *INDENT-OFF* */
/* clang-format off */
#if defined(HAVE_SYS_STAT_H) || __has_include(<sys/stat.h>)
@@ -338,7 +351,7 @@ typedef pthread_mutex_t mdbx_fastmutex_t;
/* *INDENT-OFF* */
/* clang-format off */
-#if defined(__GLIBC__) || defined(__GNU_LIBRARY__) || defined(__ANDROID__) || \
+#if defined(__GLIBC__) || defined(__GNU_LIBRARY__) || defined(__ANDROID_API__) || \
defined(HAVE_ENDIAN_H) || __has_include(<endian.h>)
#include <endian.h>
#elif defined(__APPLE__) || defined(__MACH__) || defined(__OpenBSD__) || \
@@ -495,6 +508,8 @@ MDBX_INTERNAL_FUNC int mdbx_vasprintf(char **strp, const char *fmt, va_list ap);
#if defined(__linux__) || defined(__gnu_linux__)
MDBX_INTERNAL_VAR uint32_t mdbx_linux_kernel_version;
+MDBX_INTERNAL_VAR bool
+ mdbx_RunningOnWSL /* Windows Subsystem for Linux is mad and trouble-full */;
#endif /* Linux */
/* Get the size of a memory page for the system.
@@ -636,13 +651,15 @@ static __maybe_unused __inline uint32_t mdbx_getpid(void) {
#endif
}
-static __maybe_unused __inline size_t mdbx_thread_self(void) {
- STATIC_ASSERT(sizeof(mdbx_tid_t) <= sizeof(size_t));
+static __maybe_unused __inline uintptr_t mdbx_thread_self(void) {
+ mdbx_tid_t thunk;
+ STATIC_ASSERT(sizeof(uintptr_t) >= sizeof(thunk));
#if defined(_WIN32) || defined(_WIN64)
- return GetCurrentThreadId();
+ thunk = GetCurrentThreadId();
#else
- return (size_t)pthread_self();
+ thunk = pthread_self();
#endif
+ return (uintptr_t)thunk;
}
MDBX_INTERNAL_FUNC void __maybe_unused mdbx_osal_jitter(bool tiny);
diff --git a/libs/libmdbx/src/src/tools/CMakeLists.txt b/libs/libmdbx/src/src/tools/CMakeLists.txt
deleted file mode 100644
index 99167fa86f..0000000000
--- a/libs/libmdbx/src/src/tools/CMakeLists.txt
+++ /dev/null
@@ -1,60 +0,0 @@
-set(MDBX_TOOLS mdbx_chk mdbx_copy mdbx_dump mdbx_load mdbx_stat)
-
-# use, i.e. don't skip the full RPATH for the build tree
-set(CMAKE_SKIP_BUILD_RPATH FALSE)
-
-# when building, don't use the install RPATH already (but later on when installing)
-set(CMAKE_BUILD_WITH_INSTALL_RPATH FALSE)
-
-# add the automatically determined parts of the RPATH
-# which point to directories outside the build tree to the install RPATH
-set(CMAKE_INSTALL_RPATH_USE_LINK_PATH TRUE)
-
-# the RPATH to be used when installing, but only if it's not a system directory
-list(FIND CMAKE_PLATFORM_IMPLICIT_LINK_DIRECTORIES "${CMAKE_INSTALL_PREFIX}/lib" isSystemDir)
-if(isSystemDir EQUAL -1)
- if(${CMAKE_SYSTEM_NAME} STREQUAL "Darwin")
- set(CMAKE_INSTALL_RPATH "@executable_path/../lib")
- else()
- set(CMAKE_INSTALL_RPATH "\$ORIGIN/../lib")
- endif()
-endif()
-
-foreach(TOOL ${MDBX_TOOLS})
- if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
- add_executable(${TOOL} ${TOOL}.c wingetopt.c wingetopt.h)
- else()
- add_executable(${TOOL} ${TOOL}.c)
- endif()
-
- target_link_libraries(${TOOL} mdbx ${CMAKE_THREAD_LIBS_INIT})
- set_target_properties(${TOOL} PROPERTIES
- C_STANDARD ${MDBX_C_STANDARD} C_STANDARD_REQUIRED ON
- INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>)
-endforeach()
-
-if(LIB_MATH)
- target_link_libraries(mdbx_chk ${LIB_MATH})
- target_link_libraries(mdbx_stat ${LIB_MATH})
-endif()
-
-install(
- TARGETS
- mdbx_chk
- mdbx_stat
- mdbx_copy
- mdbx_dump
- mdbx_load
- RUNTIME
- DESTINATION bin
- COMPONENT runtime)
-
-install(
- FILES
- ../man1/mdbx_chk.1
- ../man1/mdbx_stat.1
- ../man1/mdbx_copy.1
- ../man1/mdbx_dump.1
- ../man1/mdbx_load.1
- DESTINATION man/man1
- COMPONENT doc)
diff --git a/libs/libmdbx/src/src/elements/version.c.in b/libs/libmdbx/src/src/version.c.in
index 2854bd5d9d..2854bd5d9d 100644
--- a/libs/libmdbx/src/src/elements/version.c.in
+++ b/libs/libmdbx/src/src/version.c.in
diff --git a/libs/libmdbx/src/src/tools/wingetopt.c b/libs/libmdbx/src/src/wingetopt.c
index 7feb223fc1..7feb223fc1 100644
--- a/libs/libmdbx/src/src/tools/wingetopt.c
+++ b/libs/libmdbx/src/src/wingetopt.c
diff --git a/libs/libmdbx/src/src/tools/wingetopt.h b/libs/libmdbx/src/src/wingetopt.h
index d328e38ccc..d328e38ccc 100644
--- a/libs/libmdbx/src/src/tools/wingetopt.h
+++ b/libs/libmdbx/src/src/wingetopt.h
diff --git a/libs/libmdbx/src/test/CMakeLists.txt b/libs/libmdbx/src/test/CMakeLists.txt
index 0cc22182c9..2014cb57fa 100644
--- a/libs/libmdbx/src/test/CMakeLists.txt
+++ b/libs/libmdbx/src/test/CMakeLists.txt
@@ -1,9 +1,3 @@
-if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
- set(TEST_OSAL windows)
-else()
- set(TEST_OSAL unix)
-endif()
-
add_executable(mdbx_test
base.h
cases.cc
@@ -21,7 +15,8 @@ add_executable(mdbx_test
log.h
main.cc
osal.h
- osal-${TEST_OSAL}.cc
+ osal-unix.cc
+ osal-windows.cc
test.cc
test.h
try.cc
@@ -35,19 +30,15 @@ add_executable(mdbx_test
set_target_properties(mdbx_test PROPERTIES
INTERPROCEDURAL_OPTIMIZATION $<BOOL:${INTERPROCEDURAL_OPTIMIZATION}>
CXX_STANDARD 17 CXX_STANDARD_REQUIRED ON)
+target_setup_options(mdbx_test)
-if(CC_HAS_FASTMATH)
- target_compile_options(mdbx_test PRIVATE "-ffast-math")
-endif()
-if(CC_HAS_VISIBILITY AND (LTO_ENABLED OR INTERPROCEDURAL_OPTIMIZATION))
- set_target_properties(mdbx_test PROPERTIES LINK_FLAGS "-fvisibility=hidden")
-endif()
-
-target_link_libraries(mdbx_test mdbx ${LIB_MATH} ${CMAKE_THREAD_LIBS_INIT})
+target_link_libraries(mdbx_test ${TOOL_MDBX_LIB} ${LIB_MATH} ${CMAKE_THREAD_LIBS_INIT})
if(${CMAKE_SYSTEM_NAME} STREQUAL "Windows")
target_link_libraries(mdbx_test winmm.lib)
endif()
if(UNIX AND NOT SUBPROJECT)
- add_subdirectory(pcrf)
+ add_executable(pcrf_test pcrf/pcrf_test.c)
+ target_include_directories(pcrf_test PRIVATE "${PROJECT_SOURCE_DIR}")
+ target_link_libraries(pcrf_test ${TOOL_MDBX_LIB})
endif()
diff --git a/libs/libmdbx/src/test/base.h b/libs/libmdbx/src/test/base.h
index 01cf171455..7fc6cac68c 100644
--- a/libs/libmdbx/src/test/base.h
+++ b/libs/libmdbx/src/test/base.h
@@ -84,14 +84,14 @@
#define MDBX_INTERNAL_VAR extern
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
#include "../mdbx.h"
-#include "../src/elements/defs.h"
-#include "../src/elements/osal.h"
+#include "../src/defs.h"
+#include "../src/osal.h"
#if !defined(__thread) && (defined(_MSC_VER) || defined(__DMC__))
#define __thread __declspec(thread)
#endif /* __thread */
-#include "../src/elements/options.h"
+#include "../src/options.h"
#ifdef _MSC_VER
#pragma warning(pop)
diff --git a/libs/libmdbx/src/test/osal-unix.cc b/libs/libmdbx/src/test/osal-unix.cc
index edc3e91a48..dc0774063a 100644
--- a/libs/libmdbx/src/test/osal-unix.cc
+++ b/libs/libmdbx/src/test/osal-unix.cc
@@ -14,6 +14,8 @@
#include "test.h"
+#if !(defined(_WIN32) || defined(_WIN64))
+
#include <pthread.h>
#include <signal.h>
#include <sys/mman.h>
@@ -529,3 +531,5 @@ std::string osal_tempdir(void) {
int osal_removefile(const std::string &pathname) {
return unlink(pathname.c_str()) ? errno : MDBX_SUCCESS;
}
+
+#endif /* !Windows */
diff --git a/libs/libmdbx/src/test/osal-windows.cc b/libs/libmdbx/src/test/osal-windows.cc
index 6de674651b..9bde047a2c 100644
--- a/libs/libmdbx/src/test/osal-windows.cc
+++ b/libs/libmdbx/src/test/osal-windows.cc
@@ -1,5 +1,5 @@
/*
- * Copyright 2017-2019 Leonid Yuriev <leo@yuriev.ru>
+ * Copyright 2017-2020 Leonid Yuriev <leo@yuriev.ru>
* and other libmdbx authors: please see AUTHORS file.
* All rights reserved.
*
@@ -14,6 +14,8 @@
#include "test.h"
+#if defined(_WIN32) || defined(_WIN64)
+
static std::unordered_map<unsigned, HANDLE> events;
static HANDLE hBarrierSemaphore, hBarrierEvent;
static HANDLE hProgressActiveEvent, hProgressPassiveEvent;
@@ -422,16 +424,10 @@ void osal_udelay(unsigned us) {
static unsigned threshold_us;
if (threshold_us == 0) {
-#if 1
unsigned timeslice_ms = 1;
while (timeBeginPeriod(timeslice_ms) == TIMERR_NOCANDO)
++timeslice_ms;
threshold_us = timeslice_ms * 1500u;
-#else
- ULONGLONG InterruptTimePrecise_100ns;
- QueryInterruptTimePrecise(&InterruptTimePrecise_100ns);
- threshold_us = InterruptTimePrecise_100ns / 5;
-#endif
assert(threshold_us > 0);
}
@@ -459,3 +455,5 @@ std::string osal_tempdir(void) {
int osal_removefile(const std::string &pathname) {
return DeleteFileA(pathname.c_str()) ? MDBX_SUCCESS : GetLastError();
}
+
+#endif /* Windows */
diff --git a/libs/libmdbx/src/test/pcrf/CMakeLists.txt b/libs/libmdbx/src/test/pcrf/CMakeLists.txt
deleted file mode 100644
index 8bd3e3d859..0000000000
--- a/libs/libmdbx/src/test/pcrf/CMakeLists.txt
+++ /dev/null
@@ -1,5 +0,0 @@
-set(TARGET pcrf_test)
-add_executable(${TARGET} pcrf_test.c)
-target_include_directories(${TARGET} PRIVATE "${PROJECT_SOURCE_DIR}")
-target_link_libraries(${TARGET} mdbx)
-
diff --git a/libs/libmdbx/src/src/elements/version.c b/libs/libmdbx/src/version.c
index 1d964a93a1..d6b952761e 100644
--- a/libs/libmdbx/src/src/elements/version.c
+++ b/libs/libmdbx/src/version.c
@@ -1,10 +1,10 @@
/* This is CMake-template for libmdbx's version.c
******************************************************************************/
-#include "internals.h"
+#include "src/internals.h"
#if MDBX_VERSION_MAJOR != 0 || \
- MDBX_VERSION_MINOR != 6
+ MDBX_VERSION_MINOR != 7
#error "API version mismatch! Had `git fetch --tags` done?"
#endif
@@ -24,11 +24,11 @@ __dll_export
#endif
const mdbx_version_info mdbx_version = {
0,
- 6,
+ 7,
0,
- 1935,
- {"2020-03-20T17:45:36+03:00", "e01a07c814dffd0b1433ad536da16ab95344615a", "c20ba1e84f51ca9dd10603fe461b1bcc171aa075",
- "v0.6.0-45-gc20ba1e"},
+ 1981,
+ {"2020-04-25T11:12:23+03:00", "5c78012e38f306d9601e1f43109c8aecbacb2e14", "ca8fa31c3fe1b4d92278d7a54364f6fb73dbae04",
+ "v0.7.0-39-gca8fa31"},
sourcery};
__dll_export
diff --git a/tools/mdbx_chk/src/mdbx_chk.cc b/tools/mdbx_chk/src/mdbx_chk.cc
index 5b41402f85..f2725bd37f 100644
--- a/tools/mdbx_chk/src/mdbx_chk.cc
+++ b/tools/mdbx_chk/src/mdbx_chk.cc
@@ -23,7 +23,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../../libs/libmdbx/src/src/elements/internals.h"
+#include "../../libs/libmdbx/src/src/internals.h"
typedef struct flagbit {
int bit;
@@ -39,8 +39,8 @@ const flagbit dbflags[] = {{MDBX_DUPSORT, "dupsort"},
{0, NULL}};
#if defined(_WIN32) || defined(_WIN64)
-#include "../../libs/libmdbx/src/src/tools/wingetopt.h"
-#include "../../libs/libmdbx/src/src/tools/wingetopt.c"
+#include "../../libs/libmdbx/src/src/wingetopt.h"
+#include "../../libs/libmdbx/src/src/wingetopt.c"
static volatile BOOL user_break;
static BOOL WINAPI ConsoleBreakHandlerRoutine(DWORD dwCtrlType) {
@@ -562,6 +562,12 @@ static int process_db(MDBX_dbi dbi_handle, char *dbi_name, visitor *handler,
uint64_t record_count = 0, dups = 0;
uint64_t key_bytes = 0, data_bytes = 0;
+ if ((MDBX_TXN_FINISHED | MDBX_TXN_ERROR) & mdbx_txn_flags(txn)) {
+ print(" ! abort processing '%s' due to a previous error\n",
+ dbi_name ? dbi_name : "@MAIN");
+ return MDBX_BAD_TXN;
+ }
+
if (dbi_handle == ~0u) {
rc = mdbx_dbi_open(txn, dbi_name, 0, &dbi_handle);
if (rc) {
diff --git a/tools/mdbx_dump/src/mdbx_dump.cc b/tools/mdbx_dump/src/mdbx_dump.cc
index f51903df4c..138650d8e7 100644
--- a/tools/mdbx_dump/src/mdbx_dump.cc
+++ b/tools/mdbx_dump/src/mdbx_dump.cc
@@ -23,7 +23,7 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../../libs/libmdbx/src/src/elements/internals.h"
+#include "../../libs/libmdbx/src/src/internals.h"
#include <ctype.h>
@@ -44,8 +44,8 @@ flagbit dbflags[] = {{MDBX_REVERSEKEY, "reversekey"},
{0, NULL}};
#if defined(_WIN32) || defined(_WIN64)
-#include "../../libs/libmdbx/src/src/tools/wingetopt.h"
-#include "../../libs/libmdbx/src/src/tools/wingetopt.c"
+#include "../../libs/libmdbx/src/src/wingetopt.h"
+#include "../../libs/libmdbx/src/src/wingetopt.c"
static volatile BOOL user_break;
static BOOL WINAPI ConsoleBreakHandlerRoutine(DWORD dwCtrlType) {
diff --git a/tools/mdbx_load/src/mdbx_load.cc b/tools/mdbx_load/src/mdbx_load.cc
index 567ad5b93a..d9f7f872bd 100644
--- a/tools/mdbx_load/src/mdbx_load.cc
+++ b/tools/mdbx_load/src/mdbx_load.cc
@@ -23,13 +23,13 @@
#endif /* _MSC_VER (warnings) */
#define MDBX_TOOLS /* Avoid using internal mdbx_assert() */
-#include "../../libs/libmdbx/src/src/elements/internals.h"
+#include "../../libs/libmdbx/src/src/internals.h"
#include <ctype.h>
#if defined(_WIN32) || defined(_WIN64)
-#include "../../libs/libmdbx/src/src/tools/wingetopt.h"
-#include "../../libs/libmdbx/src/src/tools/wingetopt.c"
+#include "../../libs/libmdbx/src/src/wingetopt.h"
+#include "../../libs/libmdbx/src/src/wingetopt.c"
static volatile BOOL user_break;
static BOOL WINAPI ConsoleBreakHandlerRoutine(DWORD dwCtrlType) {
@@ -342,7 +342,8 @@ static void usage(void) {
"dbpath\n"
" -V\t\tprint version and exit\n"
" -q\t\tbe quiet\n"
- " -a\t\tappend records in input order\n"
+ " -a\t\tappend records in input order (required for custom "
+ "comparators)\n"
" -f file\tread from file instead of stdin\n"
" -s name\tload into named subDB\n"
" -N\t\tuse NOOVERWRITE on puts\n"