diff options
Diffstat (limited to 'libs/libmdbx/src/mdbx.h++')
-rw-r--r-- | libs/libmdbx/src/mdbx.h++ | 4707 |
1 files changed, 0 insertions, 4707 deletions
diff --git a/libs/libmdbx/src/mdbx.h++ b/libs/libmdbx/src/mdbx.h++ deleted file mode 100644 index 5ae3838aeb..0000000000 --- a/libs/libmdbx/src/mdbx.h++ +++ /dev/null @@ -1,4707 +0,0 @@ -/// \file mdbx.h++ -/// \brief The libmdbx C++ API header file (preliminary). -/// -/// \author Copyright (c) 2020-2021, Leonid Yuriev <leo@yuriev.ru>. -/// \copyright SPDX-License-Identifier: Apache-2.0 -/// -/// Tested with: -/// - LCC >= 1.23 (http://www.mcst.ru/lcc), -/// - GNU C++ >= 4.8, -/// - clang >= 4.0, -/// - MSVC >= 19.0 (Visual Studio 2015), -/// but 19.2x could hang due optimizer bug. -/// - -#pragma once - -#if (!defined(__cplusplus) || __cplusplus < 201103L) && \ - !(defined( \ - _MSC_VER) /* MSVC is mad and don't define __cplusplus properly */ \ - && _MSC_VER == 1900) -#error "C++11 or better is required" -#endif - -#if (defined(_WIN32) || defined(_WIN64)) && MDBX_AVOID_CRT -#error "CRT is required for C++ API, the MDBX_AVOID_CRT option must be disabled" -#endif /* Windows */ - -#ifndef __has_include -#define __has_include(header) (0) -#endif /* __has_include */ - -#if __has_include(<version>) -#include <version> -#endif /* <version> */ - -/* Disable min/max macros from C' headers */ -#ifndef NOMINMAX -#define NOMINMAX -#endif - -#include <algorithm> // for std::min/max -#include <cassert> // for assert() -#include <cstring> // for std::strlen, str:memcmp -#include <exception> // for std::exception_ptr -#include <ostream> // for std::ostream -#include <sstream> // for std::ostringstream -#include <stdexcept> // for std::invalid_argument -#include <string> // for std::string -#include <type_traits> // for std::is_pod<>, etc. -#include <vector> // for std::vector<> as template args - -// Unused for now -// #if defined(__cpp_lib_bit_cast) && __cpp_lib_bit_cast >= 201806L -// #include <bit> -// #endif - -#if defined(__cpp_lib_memory_resource) && __cpp_lib_memory_resource >= 201603L -#include <memory_resource> -#endif - -#if defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L -#include <string_view> -#endif - -#if defined(__cpp_lib_filesystem) && __cpp_lib_filesystem >= 201703L -#include <filesystem> -#endif - -#include "mdbx.h" - -#if defined(DOXYGEN) || \ - defined(__cpp_constexpr) && __cpp_constexpr >= 201603L && \ - ((defined(_MSC_VER) && _MSC_VER >= 1915) || \ - (defined(__clang__) && __clang_major__ > 5) || \ - (defined(__GNUC__) && __GNUC__ > 7) || \ - (!defined(__GNUC__) && !defined(__clang__) && !defined(_MSC_VER))) -#define MDBX_CXX17_CONSTEXPR constexpr -#else -#define MDBX_CXX17_CONSTEXPR inline -#endif /* MDBX_CXX17_CONSTEXPR */ - -#if defined(DOXYGEN) || defined(__cpp_lib_is_constant_evaluated) && \ - __cpp_lib_is_constant_evaluated >= 201811L && \ - defined(__cpp_lib_constexpr_string) && \ - __cpp_lib_constexpr_string >= 201907L -#define MDBX_CXX20_CONSTEXPR constexpr -#else -#define MDBX_CXX20_CONSTEXPR inline -#endif /* MDBX_CXX20_CONSTEXPR */ - -#if defined(CONSTEXPR_ASSERT) -#define MDBX_CONSTEXPR_ASSERT(expr) CONSTEXPR_ASSERT(expr) -#elif defined NDEBUG -#define MDBX_CONSTEXPR_ASSERT(expr) void(0) -#else -#define MDBX_CONSTEXPR_ASSERT(expr) \ - ((expr) ? void(0) : [] { assert(!#expr); }()) -#endif /* MDBX_CONSTEXPR_ASSERT */ - -#ifndef MDBX_LIKELY -#if defined(DOXYGEN) || \ - (defined(__GNUC__) || __has_builtin(__builtin_expect)) && \ - !defined(__COVERITY__) -#define MDBX_LIKELY(cond) __builtin_expect(!!(cond), 1) -#else -#define MDBX_LIKELY(x) (x) -#endif -#endif /* MDBX_LIKELY */ - -#ifndef MDBX_UNLIKELY -#if defined(DOXYGEN) || \ - (defined(__GNUC__) || __has_builtin(__builtin_expect)) && \ - !defined(__COVERITY__) -#define MDBX_UNLIKELY(cond) __builtin_expect(!!(cond), 0) -#else -#define MDBX_UNLIKELY(x) (x) -#endif -#endif /* MDBX_UNLIKELY */ - -#if defined(DOXYGEN) || \ - (__has_cpp_attribute(fallthrough) && \ - (!defined(__clang__) || __clang__ > 4)) || \ - __cplusplus >= 201703L -#define MDBX_CXX17_FALLTHROUGH [[fallthrough]] -#else -#define MDBX_CXX17_FALLTHROUGH -#endif /* MDBX_CXX17_FALLTHROUGH */ - -#if defined(DOXYGEN) || __has_cpp_attribute(likely) -#define MDBX_CXX20_LIKELY [[likely]] -#else -#define MDBX_CXX20_LIKELY -#endif /* MDBX_CXX20_LIKELY */ - -#ifndef MDBX_CXX20_UNLIKELY -#if defined(DOXYGEN) || __has_cpp_attribute(unlikely) -#define MDBX_CXX20_UNLIKELY [[unlikely]] -#else -#define MDBX_CXX20_UNLIKELY -#endif -#endif /* MDBX_CXX20_UNLIKELY */ - -#ifdef _MSC_VER -#pragma warning(push, 4) -#pragma warning(disable : 4251) /* 'std::FOO' needs to have dll-interface to \ - be used by clients of 'mdbx::BAR' */ -#pragma warning(disable : 4275) /* non dll-interface 'std::FOO' used as \ - base for dll-interface 'mdbx::BAR' */ -/* MSVC is mad and can generate this warning for its own intermediate - * automatically generated code, which becomes unreachable after some kinds of - * optimization (copy elision, etc). */ -#pragma warning(disable : 4702) /* unreachable code */ -#endif /* _MSC_VER (warnings) */ - -//------------------------------------------------------------------------------ -/// \defgroup cxx_api C++ API -/// @{ - -namespace mdbx { - -// Functions whose signature depends on the `mdbx::byte` type -// must be strictly defined as inline! -#if defined(DOXYGEN) || (defined(__cpp_char8_t) && __cpp_char8_t >= 201811) -// Wanna using a non-aliasing type to release more power of an optimizer. -using byte = char8_t; -#else -// Wanna not using std::byte since it doesn't add features, -// but add inconvenient restrictions. -using byte = unsigned char; -#endif /* __cpp_char8_t >= 201811*/ - -/// \copydoc MDBX_version_info -using version_info = ::MDBX_version_info; -/// \brief Returns libmdbx version information. -MDBX_CXX11_CONSTEXPR const version_info &get_version() noexcept; -/// \copydoc MDBX_build_info -using build_info = ::MDBX_build_info; -/// \brief Returns libmdbx build information. -MDBX_CXX11_CONSTEXPR const build_info &get_build() noexcept; - -/// \brief constexpr-compatible strlen(). -static MDBX_CXX17_CONSTEXPR size_t strlen(const char *c_str) noexcept; - -struct slice; -class env; -class env_managed; -class txn; -class txn_managed; -class cursor; -class cursor_managed; - -/// \brief Legacy default allocator -/// but it is recommended to use \ref polymorphic_allocator. -using legacy_allocator = ::std::string::allocator_type; - -#if defined(DOXYGEN) || \ - defined(__cpp_lib_memory_resource) && __cpp_lib_memory_resource >= 201603L -/// \brief Default polymorphic allocator for modern code. -using polymorphic_allocator = ::std::pmr::string::allocator_type; -#endif /* __cpp_lib_memory_resource >= 201603L */ - -/// \brief Default singe-byte string. -template <class ALLOCATOR = legacy_allocator> -using string = ::std::basic_string<char, ::std::char_traits<char>, ALLOCATOR>; - -using filehandle = ::mdbx_filehandle_t; -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_filesystem) && __cpp_lib_filesystem >= 201703L && \ - (!defined(__MAC_OS_X_VERSION_MIN_REQUIRED) || \ - __MAC_OS_X_VERSION_MIN_REQUIRED >= 101500)) -#define MDBX_STD_FILESYSTEM_PATH -using path = ::std::filesystem::path; -#elif defined(_WIN32) || defined(_WIN64) -using path = ::std::wstring; -#else -using path = ::std::string; -#endif - -/// \brief Transfers C++ exceptions thru C callbacks. -/// \details Implements saving exceptions before returning -/// from an C++'s environment to the intermediate C code and re-throwing after -/// returning from C to the C++'s environment. -class LIBMDBX_API_TYPE exception_thunk { - ::std::exception_ptr captured_; - -public: - exception_thunk() noexcept = default; - exception_thunk(const exception_thunk &) = delete; - exception_thunk(exception_thunk &&) = delete; - exception_thunk &operator=(const exception_thunk &) = delete; - exception_thunk &operator=(exception_thunk &&) = delete; - inline bool is_clean() const noexcept; - inline void capture() noexcept; - inline void rethrow_captured() const; -}; - -/// \brief Implements error information and throwing corresponding exceptions. -class LIBMDBX_API_TYPE error { - MDBX_error_t code_; - inline error &operator=(MDBX_error_t error_code) noexcept; - -public: - MDBX_CXX11_CONSTEXPR error(MDBX_error_t error_code) noexcept; - error(const error &) = default; - error(error &&) = default; - error &operator=(const error &) = default; - error &operator=(error &&) = default; - - MDBX_CXX11_CONSTEXPR friend bool operator==(const error &a, - const error &b) noexcept; - MDBX_CXX11_CONSTEXPR friend bool operator!=(const error &a, - const error &b) noexcept; - - MDBX_CXX11_CONSTEXPR bool is_success() const noexcept; - MDBX_CXX11_CONSTEXPR bool is_result_true() const noexcept; - MDBX_CXX11_CONSTEXPR bool is_result_false() const noexcept; - MDBX_CXX11_CONSTEXPR bool is_failure() const noexcept; - - /// \brief Returns error code. - MDBX_CXX11_CONSTEXPR MDBX_error_t code() const noexcept; - - /// \brief Returns message for MDBX's errors only and "SYSTEM" for others. - const char *what() const noexcept; - - /// \brief Returns message for any errors. - ::std::string message() const; - - /// \brief Returns true for MDBX's errors. - MDBX_CXX11_CONSTEXPR bool is_mdbx_error() const noexcept; - /// \brief Panics on unrecoverable errors inside destructors etc. - [[noreturn]] void panic(const char *context_where_when, - const char *func_who_what) const noexcept; - [[noreturn]] void throw_exception() const; - [[noreturn]] static inline void throw_exception(int error_code); - inline void throw_on_failure() const; - inline void success_or_throw() const; - inline void success_or_throw(const exception_thunk &) const; - inline void panic_on_failure(const char *context_where, - const char *func_who) const noexcept; - inline void success_or_panic(const char *context_where, - const char *func_who) const noexcept; - static inline void throw_on_nullptr(const void *ptr, MDBX_error_t error_code); - static inline void success_or_throw(MDBX_error_t error_code); - static void success_or_throw(int error_code) { - success_or_throw(static_cast<MDBX_error_t>(error_code)); - } - static inline void throw_on_failure(int error_code); - static inline bool boolean_or_throw(int error_code); - static inline void success_or_throw(int error_code, const exception_thunk &); - static inline void panic_on_failure(int error_code, const char *context_where, - const char *func_who) noexcept; - static inline void success_or_panic(int error_code, const char *context_where, - const char *func_who) noexcept; -}; - -/// \brief Base class for all libmdbx's exceptions that are corresponds -/// to libmdbx errors. -/// -/// \see MDBX_error_t -class LIBMDBX_API_TYPE exception : public ::std::runtime_error { - using base = ::std::runtime_error; - ::mdbx::error error_; - -public: - exception(const ::mdbx::error &) noexcept; - exception(const exception &) = default; - exception(exception &&) = default; - exception &operator=(const exception &) = default; - exception &operator=(exception &&) = default; - virtual ~exception() noexcept; - const mdbx::error error() const noexcept { return error_; } -}; - -/// \brief Fatal exception that lead termination anyway -/// in dangerous unrecoverable cases. -class LIBMDBX_API_TYPE fatal : public exception { - using base = exception; - -public: - fatal(const ::mdbx::error &) noexcept; - fatal(const exception &src) noexcept : fatal(src.error()) {} - fatal(exception &&src) noexcept : fatal(src.error()) {} - fatal(const fatal &src) noexcept : fatal(src.error()) {} - fatal(fatal &&src) noexcept : fatal(src.error()) {} - fatal &operator=(fatal &&) = default; - fatal &operator=(const fatal &) = default; - virtual ~fatal() noexcept; -}; - -#define MDBX_DECLARE_EXCEPTION(NAME) \ - struct LIBMDBX_API_TYPE NAME : public exception { \ - NAME(const ::mdbx::error &); \ - virtual ~NAME() noexcept; \ - } -MDBX_DECLARE_EXCEPTION(bad_map_id); -MDBX_DECLARE_EXCEPTION(bad_transaction); -MDBX_DECLARE_EXCEPTION(bad_value_size); -MDBX_DECLARE_EXCEPTION(db_corrupted); -MDBX_DECLARE_EXCEPTION(db_full); -MDBX_DECLARE_EXCEPTION(db_invalid); -MDBX_DECLARE_EXCEPTION(db_too_large); -MDBX_DECLARE_EXCEPTION(db_unable_extend); -MDBX_DECLARE_EXCEPTION(db_version_mismatch); -MDBX_DECLARE_EXCEPTION(db_wanna_write_for_recovery); -MDBX_DECLARE_EXCEPTION(incompatible_operation); -MDBX_DECLARE_EXCEPTION(internal_page_full); -MDBX_DECLARE_EXCEPTION(internal_problem); -MDBX_DECLARE_EXCEPTION(key_exists); -MDBX_DECLARE_EXCEPTION(key_mismatch); -MDBX_DECLARE_EXCEPTION(max_maps_reached); -MDBX_DECLARE_EXCEPTION(max_readers_reached); -MDBX_DECLARE_EXCEPTION(multivalue); -MDBX_DECLARE_EXCEPTION(no_data); -MDBX_DECLARE_EXCEPTION(not_found); -MDBX_DECLARE_EXCEPTION(operation_not_permitted); -MDBX_DECLARE_EXCEPTION(permission_denied_or_not_writeable); -MDBX_DECLARE_EXCEPTION(reader_slot_busy); -MDBX_DECLARE_EXCEPTION(remote_media); -MDBX_DECLARE_EXCEPTION(something_busy); -MDBX_DECLARE_EXCEPTION(thread_mismatch); -MDBX_DECLARE_EXCEPTION(transaction_full); -MDBX_DECLARE_EXCEPTION(transaction_overlapping); -#undef MDBX_DECLARE_EXCEPTION - -[[noreturn]] LIBMDBX_API void throw_too_small_target_buffer(); -[[noreturn]] LIBMDBX_API void throw_max_length_exceeded(); -[[noreturn]] LIBMDBX_API void throw_out_range(); -MDBX_CXX14_CONSTEXPR size_t check_length(size_t bytes); - -//------------------------------------------------------------------------------ - -/// \brief References a data located outside the slice. -/// -/// The `slice` is similar in many ways to `std::string_view`, but it -/// implements specific capabilities and manipulates with bytes but -/// not a characters. -/// -/// \copydetails MDBX_val -struct LIBMDBX_API_TYPE slice : public ::MDBX_val { - /// \todo slice& operator<<(slice&, ...) for reading - /// \todo key-to-value (parse/unpack) functions - /// \todo template<class X> key(X); for decoding keys while reading - - enum { max_length = MDBX_MAXDATASIZE }; - - /// \brief Create an empty slice. - MDBX_CXX11_CONSTEXPR slice() noexcept; - - /// \brief Create a slice that refers to [0,bytes-1] of memory bytes pointed - /// by ptr. - MDBX_CXX14_CONSTEXPR slice(const void *ptr, size_t bytes); - - /// \brief Create a slice that refers to [begin,end] of memory bytes. - MDBX_CXX14_CONSTEXPR slice(const void *begin, const void *end); - - /// \brief Create a slice that refers to text[0,strlen(text)-1]. - template <size_t SIZE> - MDBX_CXX14_CONSTEXPR slice(const char (&text)[SIZE]) noexcept - : slice(text, SIZE - 1) { - MDBX_CONSTEXPR_ASSERT(SIZE > 0 && text[SIZE - 1] == '\0'); - } - /// \brief Create a slice that refers to c_str[0,strlen(c_str)-1]. - explicit MDBX_CXX17_CONSTEXPR slice(const char *c_str); - - /// \brief Create a slice that refers to the contents of "str". - /* 'explicit' to avoid reference to the temporary std::string instance */ - template <class C, class T, class A> - explicit MDBX_CXX20_CONSTEXPR slice(const ::std::basic_string<C, T, A> &str) - : slice(str.data(), str.length() * sizeof(C)) {} - - MDBX_CXX14_CONSTEXPR slice(const MDBX_val &src); - MDBX_CXX11_CONSTEXPR slice(const slice &) noexcept = default; -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - /// \brief Create a slice that refers to the same contents as "sv" - template <class C, class T> - explicit MDBX_CXX14_CONSTEXPR slice(const ::std::basic_string_view<C, T> &sv) - : slice(sv.data(), sv.data() + sv.length()) {} -#endif /* __cpp_lib_string_view >= 201606L */ - - inline slice(MDBX_val &&src); - inline slice(slice &&src) noexcept; -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - slice(::std::basic_string_view<C, T> &&sv) : slice(sv) { - sv = {}; - } -#endif /* __cpp_lib_string_view >= 201606L */ - - template <size_t SIZE> - static MDBX_CXX14_CONSTEXPR slice wrap(const char (&text)[SIZE]) { - return slice(text); - } - - template <typename POD> - MDBX_CXX14_CONSTEXPR static slice wrap(const POD &pod) { - static_assert(::std::is_standard_layout<POD>::value && - !std::is_pointer<POD>::value, - "Must be a standard layout type!"); - return slice(&pod, sizeof(pod)); - } - - inline slice &assign(const void *ptr, size_t bytes); - inline slice &assign(const slice &src) noexcept; - inline slice &assign(const ::MDBX_val &src); - inline slice &assign(slice &&src) noexcept; - inline slice &assign(::MDBX_val &&src); - inline slice &assign(const void *begin, const void *end); - template <class C, class T, class A> - slice &assign(const ::std::basic_string<C, T, A> &str) { - return assign(str.data(), str.length() * sizeof(C)); - } - inline slice &assign(const char *c_str); -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - slice &assign(const ::std::basic_string_view<C, T> &view) { - return assign(view.begin(), view.end()); - } - template <class C, class T> - slice &assign(::std::basic_string_view<C, T> &&view) { - assign(view); - view = {}; - return *this; - } -#endif /* __cpp_lib_string_view >= 201606L */ - - slice &operator=(const slice &) noexcept = default; - inline slice &operator=(slice &&src) noexcept; - inline slice &operator=(::MDBX_val &&src); -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - slice &operator=(const ::std::basic_string_view<C, T> &view) { - return assign(view); - } - template <class C, class T> - slice &operator=(::std::basic_string_view<C, T> &&view) { - return assign(view); - } -#endif /* __cpp_lib_string_view >= 201606L */ - - template <class C = char, class T = ::std::char_traits<C>, - class A = legacy_allocator> - MDBX_CXX20_CONSTEXPR ::std::basic_string<C, T, A> - string(const A &allocator = A()) const { - static_assert(sizeof(C) == 1, "Must be single byte characters"); - return ::std::basic_string<C, T, A>(char_ptr(), length(), allocator); - } - - template <class C, class T, class A> - MDBX_CXX20_CONSTEXPR operator ::std::basic_string<C, T, A>() const { - return this->string<C, T, A>(); - } - - /// \brief Fills the buffer by hexadecimal data dump of slice content. - /// \throws std::length_error if given buffer is too small. - char *to_hex(char *dest, size_t dest_size, bool uppercase = false, - unsigned wrap_width = 0) const; - - /// \brief Returns the buffer size in bytes needed for hexadecimal data dump - /// of slice content. - MDBX_CXX11_CONSTEXPR size_t - to_hex_bytes(unsigned wrap_width = 0) const noexcept { - const size_t bytes = length() << 1; - return wrap_width ? bytes + bytes / wrap_width : bytes; - } - - /// \brief Fills the buffer with data converted from hexadecimal dump - /// from slice content. - /// \throws std::length_error if given buffer is too small. - byte *from_hex(byte *dest, size_t dest_size, - bool ignore_spaces = false) const; - - /// \brief Returns the buffer size in bytes needed for conversion - /// hexadecimal dump from slice content to data. - MDBX_CXX11_CONSTEXPR size_t from_hex_bytes() const noexcept { - return length() >> 1; - } - - /// \brief Fills the buffer by [Base58](https://en.wikipedia.org/wiki/Base58) - /// data dump of slice content. - /// \throws std::length_error if given buffer is too small. - char *to_base58(char *dest, size_t dest_size, unsigned wrap_width = 0) const; - - /// \brief Returns the buffer size in bytes needed for - /// [Base58](https://en.wikipedia.org/wiki/Base58) data dump of slice content. - MDBX_CXX11_CONSTEXPR size_t - to_base58_bytes(unsigned wrap_width = 0) const noexcept { - const size_t bytes = length() / 8 * 11 + (length() % 8 * 43 + 31) / 32; - return wrap_width ? bytes + bytes / wrap_width : bytes; - } - - /// \brief Fills the buffer with data converted from - /// [Base58](https://en.wikipedia.org/wiki/Base58) dump from slice content. - /// \throws std::length_error if given buffer is too small. - byte *from_base58(byte *dest, size_t dest_size, - bool ignore_spaces = false) const; - - /// \brief Returns the buffer size in bytes needed for conversion - /// [Base58](https://en.wikipedia.org/wiki/Base58) dump to data. - MDBX_CXX11_CONSTEXPR size_t from_base58_bytes() const noexcept { - return length() / 11 * 8 + length() % 11 * 32 / 43; - } - - /// \brief Fills the buffer by [Base64](https://en.wikipedia.org/wiki/Base64) - /// data dump. - /// \throws std::length_error if given buffer is too small. - char *to_base64(char *dest, size_t dest_size, unsigned wrap_width = 0) const; - - /// \brief Returns the buffer size in bytes needed for - /// [Base64](https://en.wikipedia.org/wiki/Base64) data dump. - MDBX_CXX11_CONSTEXPR size_t - to_base64_bytes(unsigned wrap_width = 0) const noexcept { - const size_t bytes = (length() + 2) / 3 * 4; - return wrap_width ? bytes + bytes / wrap_width : bytes; - } - - /// \brief Fills the buffer with data converted from - /// [Base64](https://en.wikipedia.org/wiki/Base64) dump. - /// \throws std::length_error if given buffer is too small. - byte *from_base64(byte *dest, size_t dest_size, - bool ignore_spaces = false) const; - - /// \brief Returns the buffer size in bytes needed for conversion - /// [Base64](https://en.wikipedia.org/wiki/Base64) dump to data. - MDBX_CXX11_CONSTEXPR size_t from_base64_bytes() const noexcept { - return (length() + 3) / 4 * 3; - } - - /// \brief Returns a string with a hexadecimal dump of the slice content. - template <class ALLOCATOR = legacy_allocator> - inline ::mdbx::string<ALLOCATOR> - hex_encode(bool uppercase = false, - const ALLOCATOR &allocator = ALLOCATOR()) const; - - /// \brief Decodes hexadecimal dump from the slice content into returned data - /// string. - template <class ALLOCATOR = legacy_allocator> - inline ::mdbx::string<ALLOCATOR> - hex_decode(const ALLOCATOR &allocator = ALLOCATOR()) const; - - /// \brief Returns a string with a - /// [Base58](https://en.wikipedia.org/wiki/Base58) dump of the slice content. - template <class ALLOCATOR = legacy_allocator> - inline ::mdbx::string<ALLOCATOR> - base58_encode(const ALLOCATOR &allocator = ALLOCATOR()) const; - - /// \brief Decodes [Base58](https://en.wikipedia.org/wiki/Base58) dump - /// from the slice content into returned data string. - template <class ALLOCATOR = legacy_allocator> - inline ::mdbx::string<ALLOCATOR> - base58_decode(const ALLOCATOR &allocator = ALLOCATOR()) const; - - /// \brief Returns a string with a - /// [Base64](https://en.wikipedia.org/wiki/Base64) dump of the slice content. - template <class ALLOCATOR = legacy_allocator> - inline ::mdbx::string<ALLOCATOR> - base64_encode(const ALLOCATOR &allocator = ALLOCATOR()) const; - - /// \brief Decodes [Base64](https://en.wikipedia.org/wiki/Base64) dump - /// from the slice content into returned data string. - template <class ALLOCATOR = legacy_allocator> - inline ::mdbx::string<ALLOCATOR> - base64_decode(const ALLOCATOR &allocator = ALLOCATOR()) const; - - /// \brief Checks whether the content of the slice is printable. - /// \param [in] disable_utf8 By default if `disable_utf8` is `false` function - /// checks that content bytes are printable ASCII-7 characters or a valid UTF8 - /// sequences. Otherwise, if if `disable_utf8` is `true` function checks that - /// content bytes are printable extended 8-bit ASCII codes. - MDBX_NOTHROW_PURE_FUNCTION bool - is_printable(bool disable_utf8 = false) const noexcept; - - /// \brief Checks whether the content of the slice is a hexadecimal dump. - /// \param [in] ignore_spaces If `true` function will skips spaces surrounding - /// (before, between and after) a encoded bytes. However, spaces should not - /// break a pair of characters encoding a single byte. - MDBX_NOTHROW_PURE_FUNCTION bool - is_hex(bool ignore_spaces = false) const noexcept; - MDBX_NOTHROW_PURE_FUNCTION bool - - /// \brief Checks whether the content of the slice is a - /// [Base58](https://en.wikipedia.org/wiki/Base58) dump. - /// \param [in] ignore_spaces If `true` function will skips spaces surrounding - /// (before, between and after) a encoded bytes. However, spaces should not - /// break a code group of characters. - is_base58(bool ignore_spaces = false) const noexcept; - MDBX_NOTHROW_PURE_FUNCTION bool - - /// \brief Checks whether the content of the slice is a - /// [Base64](https://en.wikipedia.org/wiki/Base64) dump. - /// \param [in] ignore_spaces If `true` function will skips spaces surrounding - /// (before, between and after) a encoded bytes. However, spaces should not - /// break a code group of characters. - is_base64(bool ignore_spaces = false) const noexcept; - -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - /// \brief Return a string_view that references the same data as this slice. - template <class C, class T> - MDBX_CXX11_CONSTEXPR explicit - operator ::std::basic_string_view<C, T>() const noexcept { - static_assert(sizeof(C) == 1, "Must be single byte characters"); - return ::std::basic_string_view<C, T>(char_ptr(), length()); - } - - /// \brief Return a string_view that references the same data as this slice. - template <class C = char, class T = ::std::char_traits<C>> - MDBX_CXX11_CONSTEXPR ::std::basic_string_view<C, T> - string_view() const noexcept { - static_assert(sizeof(C) == 1, "Must be single byte characters"); - return ::std::basic_string_view<C, T>(char_ptr(), length()); - } -#endif /* __cpp_lib_string_view >= 201606L */ - - inline void swap(slice &other) noexcept; -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - void swap(::std::basic_string_view<C, T> &view) noexcept { - static_assert(sizeof(C) == 1, "Must be single byte characters"); - const auto temp = ::std::basic_string_view<C, T>(*this); - *this = view; - view = temp; - } -#endif /* __cpp_lib_string_view >= 201606L */ - - /// \brief Returns casted to pointer to byte an address of data. - MDBX_CXX11_CONSTEXPR const byte *byte_ptr() const noexcept; - - /// \brief Returns casted to pointer to char an address of data. - MDBX_CXX11_CONSTEXPR const char *char_ptr() const noexcept; - - /// \brief Return a pointer to the beginning of the referenced data. - MDBX_CXX11_CONSTEXPR const void *data() const noexcept; - - /// \brief Returns the number of bytes. - MDBX_CXX11_CONSTEXPR size_t length() const noexcept; - - /// \brief Checks whether the slice is empty. - MDBX_CXX11_CONSTEXPR bool empty() const noexcept; - - /// \brief Checks whether the slice data pointer is nullptr. - MDBX_CXX11_CONSTEXPR bool is_null() const noexcept; - - /// \brief Returns the number of bytes. - MDBX_CXX11_CONSTEXPR size_t size() const noexcept; - - /// \brief Returns true if slice is not empty. - MDBX_CXX11_CONSTEXPR operator bool() const noexcept; - - /// \brief Depletes content of slice and make it invalid. - inline void invalidate() noexcept; - - /// \brief Makes the slice empty and referencing to nothing. - inline void clear() noexcept; - - /// \brief Drops the first "n" bytes from this slice. - /// \pre REQUIRES: `n <= size()` - inline void remove_prefix(size_t n) noexcept; - - /// \brief Drops the last "n" bytes from this slice. - /// \pre REQUIRES: `n <= size()` - inline void remove_suffix(size_t n) noexcept; - - /// \brief Drops the first "n" bytes from this slice. - /// \throws std::out_of_range if `n > size()` - inline void safe_remove_prefix(size_t n); - - /// \brief Drops the last "n" bytes from this slice. - /// \throws std::out_of_range if `n > size()` - inline void safe_remove_suffix(size_t n); - - /// \brief Checks if the data starts with the given prefix. - MDBX_NOTHROW_PURE_FUNCTION inline bool - starts_with(const slice &prefix) const noexcept; - - /// \brief Checks if the data ends with the given suffix. - MDBX_NOTHROW_PURE_FUNCTION inline bool - ends_with(const slice &suffix) const noexcept; - - /// \brief Returns the nth byte in the referenced data. - /// \pre REQUIRES: `n < size()` - inline byte operator[](size_t n) const noexcept; - - /// \brief Returns the nth byte in the referenced data with bounds checking. - /// \throws std::out_of_range if `n >= size()` - inline byte at(size_t n) const; - - /// \brief Returns the first "n" bytes of the slice. - /// \pre REQUIRES: `n <= size()` - inline slice head(size_t n) const noexcept; - - /// \brief Returns the last "n" bytes of the slice. - /// \pre REQUIRES: `n <= size()` - inline slice tail(size_t n) const noexcept; - - /// \brief Returns the middle "n" bytes of the slice. - /// \pre REQUIRES: `from + n <= size()` - inline slice middle(size_t from, size_t n) const noexcept; - - /// \brief Returns the first "n" bytes of the slice. - /// \throws std::out_of_range if `n >= size()` - inline slice safe_head(size_t n) const; - - /// \brief Returns the last "n" bytes of the slice. - /// \throws std::out_of_range if `n >= size()` - inline slice safe_tail(size_t n) const; - - /// \brief Returns the middle "n" bytes of the slice. - /// \throws std::out_of_range if `from + n >= size()` - inline slice safe_middle(size_t from, size_t n) const; - - /// \brief Returns the hash value of referenced data. - /// \attention Function implementation and returned hash values may changed - /// version to version, and in future the t1ha3 will be used here. Therefore - /// values obtained from this function shouldn't be persisted anywhere. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX14_CONSTEXPR size_t - hash_value() const noexcept; - - /// \brief Three-way fast non-lexicographically length-based comparison. - /// \return value: - /// == 0 if "a" == "b", - /// < 0 if "a" shorter than "b", - /// > 0 if "a" longer than "b", - /// < 0 if "a" length-equal and lexicographically less than "b", - /// > 0 if "a" length-equal and lexicographically great than "b". - MDBX_NOTHROW_PURE_FUNCTION static inline intptr_t - compare_fast(const slice &a, const slice &b) noexcept; - - /// \brief Three-way lexicographically comparison. - /// \return value: - /// < 0 if "a" < "b", - /// == 0 if "a" == "b", - /// > 0 if "a" > "b". - MDBX_NOTHROW_PURE_FUNCTION static inline intptr_t - compare_lexicographically(const slice &a, const slice &b) noexcept; - friend inline bool operator==(const slice &a, const slice &b) noexcept; - friend inline bool operator<(const slice &a, const slice &b) noexcept; - friend inline bool operator>(const slice &a, const slice &b) noexcept; - friend inline bool operator<=(const slice &a, const slice &b) noexcept; - friend inline bool operator>=(const slice &a, const slice &b) noexcept; - friend inline bool operator!=(const slice &a, const slice &b) noexcept; - - /// \brief Checks the slice is not refers to null address or has zero length. - MDBX_CXX11_CONSTEXPR bool is_valid() const noexcept { - return !(iov_base == nullptr && iov_len != 0); - } - - /// \brief Build an invalid slice which non-zero length and refers to null - /// address. - MDBX_CXX11_CONSTEXPR static slice invalid() noexcept { - return slice(size_t(-1)); - } - -protected: - MDBX_CXX11_CONSTEXPR slice(size_t invalid_length) noexcept - : ::MDBX_val({nullptr, invalid_length}) {} -}; - -//------------------------------------------------------------------------------ - -/// \brief The chunk of data stored inside the buffer or located outside it. -template <class ALLOCATOR = legacy_allocator> class buffer { - friend class txn; - using silo = ::mdbx::string<ALLOCATOR>; - silo silo_; - ::mdbx::slice slice_; - - void insulate() { - assert(is_reference()); - silo_.assign(slice_.char_ptr(), slice_.length()); - slice_.iov_base = const_cast<char *>(silo_.data()); - } - - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR const byte * - silo_begin() const noexcept { - return static_cast<const byte *>(static_cast<const void *>(silo_.data())); - } - - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR const byte * - silo_end() const noexcept { - return silo_begin() + silo_.capacity(); - } - - struct data_preserver : public exception_thunk { - buffer data; - data_preserver(ALLOCATOR &allocator) : data(allocator) {} - static int callback(void *context, MDBX_val *target, const void *src, - size_t bytes) noexcept; - MDBX_CXX11_CONSTEXPR operator MDBX_preserve_func() const noexcept { - return callback; - } - MDBX_CXX11_CONSTEXPR operator const buffer &() const noexcept { - return data; - } - MDBX_CXX11_CONSTEXPR operator buffer &() noexcept { return data; } - }; - -public: - /// \todo buffer& operator<<(buffer&, ...) for writing - /// \todo buffer& operator>>(buffer&, ...) for reading (delegated to slice) - /// \todo template<class X> key(X) for encoding keys while writing - - using allocator_type = ALLOCATOR; - enum : size_t { - max_length = MDBX_MAXDATASIZE, - default_shrink_threshold = 1024 - }; - - /// \brief Returns the associated allocator. - MDBX_CXX20_CONSTEXPR allocator_type get_allocator() const { - return silo_.get_allocator(); - } - - /// \brief Checks whether data chunk stored inside the buffer, otherwise - /// buffer just refers to data located outside the buffer. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR bool - is_freestanding() const noexcept { - return size_t(byte_ptr() - silo_begin()) < silo_.capacity(); - } - - /// \brief Checks whether the buffer just refers to data located outside - /// the buffer, rather than stores it. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR bool - is_reference() const noexcept { - return !is_freestanding(); - } - - /// \brief Returns the number of bytes that can be held in currently allocated - /// storage. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR size_t - capacity() const noexcept { - return is_freestanding() ? silo_.capacity() : 0; - } - - /// \brief Returns the number of bytes that available in currently allocated - /// storage ahead the currently beginning of data. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR size_t - headroom() const noexcept { - return is_freestanding() ? slice_.byte_ptr() - silo_begin() : 0; - } - - /// \brief Returns the number of bytes that available in currently allocated - /// storage after the currently data end. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR size_t - tailroom() const noexcept { - return is_freestanding() ? capacity() - headroom() - slice_.length() : 0; - } - - /// \brief Returns casted to const pointer to byte an address of data. - MDBX_CXX11_CONSTEXPR const byte *byte_ptr() const noexcept { - return slice_.byte_ptr(); - } - - /// \brief Returns casted to pointer to byte an address of data. - /// \pre REQUIRES: The buffer should store data chunk, but not referenced to - /// an external one. - MDBX_CXX11_CONSTEXPR byte *byte_ptr() noexcept { - assert(is_freestanding()); - return const_cast<byte *>(slice_.byte_ptr()); - } - - /// \brief Returns casted to const pointer to char an address of data. - MDBX_CXX11_CONSTEXPR const char *char_ptr() const noexcept { - return slice_.char_ptr(); - } - - /// \brief Returns casted to pointer to char an address of data. - /// \pre REQUIRES: The buffer should store data chunk, but not referenced to - /// an external one. - MDBX_CXX11_CONSTEXPR char *char_ptr() noexcept { - assert(is_freestanding()); - return const_cast<char *>(slice_.char_ptr()); - } - - /// \brief Return a const pointer to the beginning of the referenced data. - MDBX_CXX11_CONSTEXPR const void *data() const noexcept { - return slice_.data(); - } - - /// \brief Return a pointer to the beginning of the referenced data. - /// \pre REQUIRES: The buffer should store data chunk, but not referenced to - /// an external one. - MDBX_CXX11_CONSTEXPR void *data() noexcept { - assert(is_freestanding()); - return const_cast<void *>(slice_.data()); - } - - /// \brief Returns the number of bytes. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR size_t - length() const noexcept { - return MDBX_CONSTEXPR_ASSERT(is_reference() || - slice_.length() + headroom() == - silo_.length()), - slice_.length(); - } - - void make_freestanding() { - if (is_reference()) - insulate(); - } - - buffer(const ::mdbx::slice &src, bool make_reference, - const allocator_type &allocator = allocator_type()) - : silo_(allocator), slice_(src) { - if (!make_reference) - insulate(); - } - - buffer(const buffer &src, bool make_reference, - const allocator_type &allocator = allocator_type()) - : buffer(src.slice_, make_reference, allocator) {} - - buffer(const void *ptr, size_t bytes, bool make_reference, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(ptr, bytes), make_reference, allocator) {} - - template <class C, class T, class A> - buffer(const ::std::basic_string<C, T, A> &str, bool make_reference, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(str), make_reference, allocator) {} - - buffer(const char *c_str, bool make_reference, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(c_str), make_reference, allocator) {} - -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - buffer(const ::std::basic_string_view<C, T> &view, bool make_reference, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(view), make_reference, allocator) {} -#endif /* __cpp_lib_string_view >= 201606L */ - - MDBX_CXX20_CONSTEXPR - buffer(const ::mdbx::slice &src, - const allocator_type &allocator = allocator_type()) - : silo_(src.char_ptr(), src.length(), allocator), slice_(silo_) {} - - MDBX_CXX20_CONSTEXPR - buffer(const buffer &src, const allocator_type &allocator = allocator_type()) - : buffer(src.slice_, allocator) {} - - MDBX_CXX20_CONSTEXPR - buffer(const void *ptr, size_t bytes, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(ptr, bytes), allocator) {} - - template <class C, class T, class A> - MDBX_CXX20_CONSTEXPR - buffer(const ::std::basic_string<C, T, A> &str, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(str), allocator) {} - - MDBX_CXX20_CONSTEXPR - buffer(const char *c_str, const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(c_str), allocator) {} - -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - MDBX_CXX20_CONSTEXPR - buffer(const ::std::basic_string_view<C, T> &view, - const allocator_type &allocator = allocator_type()) - : buffer(::mdbx::slice(view), allocator) {} -#endif /* __cpp_lib_string_view >= 201606L */ - - inline buffer(size_t head_room, size_t tail_room, - const allocator_type &allocator = allocator_type()); - - inline buffer(size_t capacity, - const allocator_type &allocator = allocator_type()); - - inline buffer(size_t head_room, const ::mdbx::slice &src, size_t tail_room, - const allocator_type &allocator = allocator_type()); - - buffer(size_t head_room, const buffer &src, size_t tail_room, - const allocator_type &allocator = allocator_type()) - : buffer(head_room, src.slice_, tail_room, allocator) {} - - MDBX_CXX20_CONSTEXPR - buffer(const allocator_type &allocator = allocator_type()) noexcept - : silo_(allocator) {} - - inline buffer(const txn &txn, const ::mdbx::slice &src, - const allocator_type &allocator = allocator_type()); - - buffer(buffer &&src) noexcept - : silo_(::std::move(src.silo_)), slice_(::std::move(src.slice_)) {} - - buffer(silo &&str) noexcept : silo_(::std::move(str)), slice_(silo_) {} - - MDBX_CXX11_CONSTEXPR const ::mdbx::slice &slice() const noexcept { - return slice_; - } - - MDBX_CXX11_CONSTEXPR operator const ::mdbx::slice &() const noexcept { - return slice_; - } - - template <typename POD> - static buffer wrap(const POD &pod, bool make_reference = false, - const allocator_type &allocator = allocator_type()) { - return buffer(::mdbx::slice::wrap(pod), make_reference, allocator); - } - - /// \brief Reserves storage. - inline void reserve(size_t wanna_headroom, size_t wanna_tailroom, - size_t shrink_threshold = default_shrink_threshold); - - buffer &assign_reference(const void *ptr, size_t bytes) noexcept { - silo_.clear(); - slice_.assign(ptr, bytes); - return *this; - } - - buffer &assign_freestanding(const void *ptr, size_t bytes) { - silo_.assign(static_cast<const typename silo::value_type *>(ptr), - check_length(bytes)); - slice_.assign(silo_); - return *this; - } - - void swap(buffer &other) -#if defined(__cpp_noexcept_function_type) && \ - __cpp_noexcept_function_type >= 201510L - noexcept( - std::allocator_traits<ALLOCATOR>::propagate_on_container_swap::value -#if defined(__cpp_lib_allocator_traits_is_always_equal) && \ - __cpp_lib_allocator_traits_is_always_equal >= 201411L - || std::allocator_traits<ALLOCATOR>::is_always_equal::value -#endif /* __cpp_lib_allocator_traits_is_always_equal */ - ) -#endif /* __cpp_noexcept_function_type */ - ; - - buffer &assign(buffer &&src) -#if defined(__cpp_noexcept_function_type) && \ - __cpp_noexcept_function_type >= 201510L - noexcept(std::allocator_traits< - ALLOCATOR>::propagate_on_container_move_assignment::value -#if defined(__cpp_lib_allocator_traits_is_always_equal) && \ - __cpp_lib_allocator_traits_is_always_equal >= 201411L - || std::allocator_traits<ALLOCATOR>::is_always_equal::value -#endif /* __cpp_lib_allocator_traits_is_always_equal */ - ) -#endif /* __cpp_noexcept_function_type */ - { - silo_.assign(::std::move(src.silo_)); - slice_.assign(::std::move(src.slice_)); - return *this; - } - - buffer &assign(silo &&src) -#if defined(__cpp_noexcept_function_type) && \ - __cpp_noexcept_function_type >= 201510L - noexcept(std::allocator_traits< - ALLOCATOR>::propagate_on_container_move_assignment::value -#if defined(__cpp_lib_allocator_traits_is_always_equal) && \ - __cpp_lib_allocator_traits_is_always_equal >= 201411L - || std::allocator_traits<ALLOCATOR>::is_always_equal::value -#endif /* __cpp_lib_allocator_traits_is_always_equal */ - ) -#endif /* __cpp_noexcept_function_type */ - { - return assign(buffer(::std::move(src))); - } - - static buffer clone(const buffer &src, - const allocator_type &allocator = allocator_type()) { - return buffer(src.headroom(), src.slice_, src.tailroom(), allocator); - } - - buffer &assign(const buffer &src, bool make_reference = false) { - return assign(src.slice_, make_reference); - } - - buffer &assign(const void *ptr, size_t bytes, bool make_reference = false) { - return make_reference ? assign_reference(ptr, bytes) - : assign_freestanding(ptr, bytes); - } - - buffer &assign(const ::mdbx::slice &src, bool make_reference = false) { - return assign(src.data(), src.length(), make_reference); - } - - buffer &assign(const ::MDBX_val &src, bool make_reference = false) { - return assign(src.iov_base, src.iov_len, make_reference); - } - - buffer &assign(::mdbx::slice &&src, bool make_reference = false) { - assign(src.data(), src.length(), make_reference); - src.invalidate(); - return *this; - } - - buffer &assign(::MDBX_val &&src, bool make_reference = false) { - assign(src.iov_base, src.iov_len, make_reference); - src.iov_base = nullptr; - return *this; - } - - buffer &assign(const void *begin, const void *end, - bool make_reference = false) { - return assign(begin, - static_cast<const byte *>(end) - - static_cast<const byte *>(begin), - make_reference); - } - - template <class C, class T, class A> - buffer &assign(const ::std::basic_string<C, T, A> &str, - bool make_reference = false) { - return assign(str.data(), str.length(), make_reference); - } - - buffer &assign(const char *c_str, bool make_reference = false) { - return assign(c_str, ::mdbx::strlen(c_str), make_reference); - } - -#if defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L - template <class C, class T> - buffer &assign(const ::std::basic_string_view<C, T> &view, - bool make_reference = false) { - return assign(view.data(), view.length(), make_reference); - } - - template <class C, class T> - buffer &assign(::std::basic_string_view<C, T> &&view, - bool make_reference = false) { - assign(view.data(), view.length(), make_reference); - view = {}; - return *this; - } -#endif /* __cpp_lib_string_view >= 201606L */ - - buffer &operator=(const buffer &src) { return assign(src); } - - buffer &operator=(buffer &&src) noexcept { return assign(::std::move(src)); } - - buffer &operator=(silo &&src) noexcept { return assign(::std::move(src)); } - - buffer &operator=(const ::mdbx::slice &src) { return assign(src); } - - buffer &operator=(::mdbx::slice &&src) { return assign(::std::move(src)); } - -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - buffer &operator=(const ::std::basic_string_view<C, T> &view) noexcept { - return assign(view); - } - - /// \brief Return a string_view that references the data of this buffer. - template <class C = char, class T = ::std::char_traits<C>> - ::std::basic_string_view<C, T> string_view() const noexcept { - return slice_.string_view<C, T>(); - } - - /// \brief Return a string_view that references the data of this buffer. - template <class C, class T> - operator ::std::basic_string_view<C, T>() const noexcept { - return string_view<C, T>(); - } -#endif /* __cpp_lib_string_view >= 201606L */ - - /// \brief Decodes hexadecimal dump from the given slice to the returned - /// buffer. - static buffer decode_hex(const ::mdbx::slice &hex, - const allocator_type &allocator = allocator_type()) { -#if __cplusplus >= 201703L - return buffer(hex.hex_decode(allocator)); -#else - silo data(hex.hex_decode(allocator)); - return buffer(::std::move(data)); -#endif - } - - /// \brief Returns a buffer with a hexadecimal dump of the given slice. - static buffer encode_hex(const ::mdbx::slice &data, bool uppercase = false, - const allocator_type &allocator = allocator_type()) { -#if __cplusplus >= 201703L - return buffer(data.hex_encode(uppercase, allocator)); -#else - silo hex(data.hex_encode(uppercase, allocator)); - return buffer(::std::move(hex)); -#endif - } - - /// \brief Decodes [Base58](https://en.wikipedia.org/wiki/Base58) dump from - /// the given slice to the returned buffer. - static buffer - decode_base58(const ::mdbx::slice &base58, - const allocator_type &allocator = allocator_type()) { -#if __cplusplus >= 201703L - return buffer(base58.base58_decode(allocator)); -#else - silo data(base58.base58_decode(allocator)); - return buffer(::std::move(data)); -#endif - } - - /// \brief Returns a buffer with a - /// [Base58](https://en.wikipedia.org/wiki/Base58) dump of the given slice. - static buffer - encode_base58(const ::mdbx::slice &data, - const allocator_type &allocator = allocator_type()) { -#if __cplusplus >= 201703L - return buffer(data.base58_encode(allocator)); -#else - silo base58(data.base58_encode(allocator)); - return buffer(::std::move(base58)); -#endif - } - - /// \brief Decodes [Base64](https://en.wikipedia.org/wiki/Base64) dump from - /// the given slice to the returned buffer. - static buffer - decode_base64(const ::mdbx::slice &base64, - const allocator_type &allocator = allocator_type()) { -#if __cplusplus >= 201703L - return buffer(base64.base64_decode(allocator)); -#else - silo data(base64.base64_decode(allocator)); - return buffer(::std::move(data)); -#endif - } - - /// \brief Returns a buffer with a - /// [Base64](https://en.wikipedia.org/wiki/Base64) dump of the given slice. - static buffer - encode_base64(const ::mdbx::slice &data, - const allocator_type &allocator = allocator_type()) { -#if __cplusplus >= 201703L - return buffer(data.base64_encode(allocator)); -#else - silo base64(data.base64_encode(allocator)); - return buffer(::std::move(base64)); -#endif - } - - /// \brief Checks whether the string is empty. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR bool empty() const noexcept { - return length() == 0; - } - - /// \brief Checks whether the data pointer of the buffer is nullptr. - MDBX_CXX11_CONSTEXPR bool is_null() const noexcept { - return data() == nullptr; - } - - /// \brief Returns the number of bytes. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX20_CONSTEXPR size_t size() const noexcept { - return length(); - } - - /// \brief Returns the hash value of the data. - /// \attention Function implementation and returned hash values may changed - /// version to version, and in future the t1ha3 will be used here. Therefore - /// values obtained from this function shouldn't be persisted anywhere. - MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX14_CONSTEXPR size_t - hash_value() const noexcept { - return slice_.hash_value(); - } - - template <class C = char, class T = ::std::char_traits<C>, - class A = legacy_allocator> - MDBX_CXX20_CONSTEXPR ::std::basic_string<C, T, A> - string(const A &allocator = A()) const { - return slice_.string<C, T, A>(allocator); - } - - template <class C, class T, class A> - MDBX_CXX20_CONSTEXPR operator ::std::basic_string<C, T, A>() const { - return this->string<C, T, A>(); - } - - /// \brief Checks if the data starts with the given prefix. - MDBX_NOTHROW_PURE_FUNCTION bool - starts_with(const ::mdbx::slice &prefix) const noexcept { - return slice_.starts_with(prefix); - } - - /// \brief Checks if the data ends with the given suffix. - MDBX_NOTHROW_PURE_FUNCTION bool - ends_with(const ::mdbx::slice &suffix) const noexcept { - return slice_.ends_with(suffix); - } - - /// \brief Clears the contents and storage. - void clear() noexcept { - slice_.clear(); - silo_.clear(); - } - - /// \brief Reduces memory usage by freeing unused storage space. - void shrink_to_fit(size_t threshold = 64) { reserve(0, 0, threshold); } - - /// \brief Drops the first "n" bytes from the data chunk. - /// \pre REQUIRES: `n <= size()` - void remove_prefix(size_t n) noexcept { slice_.remove_prefix(n); } - - /// \brief Drops the last "n" bytes from the data chunk. - /// \pre REQUIRES: `n <= size()` - void remove_suffix(size_t n) noexcept { slice_.remove_suffix(n); } - - /// \brief Drops the first "n" bytes from the data chunk. - /// \throws std::out_of_range if `n > size()` - void safe_remove_prefix(size_t n) { slice_.safe_remove_prefix(n); } - - /// \brief Drops the last "n" bytes from the data chunk. - /// \throws std::out_of_range if `n > size()` - void safe_remove_suffix(size_t n) { slice_.safe_remove_suffix(n); } - - /// \brief Accesses the specified byte of data chunk. - /// \pre REQUIRES: `n < size()` - byte operator[](size_t n) const noexcept { return slice_[n]; } - - /// \brief Accesses the specified byte of data chunk. - /// \pre REQUIRES: `n < size()` - byte &operator[](size_t n) noexcept { - assert(n < size()); - return byte_ptr()[n]; - } - - /// \brief Accesses the specified byte of data chunk with bounds checking. - /// \throws std::out_of_range if `n >= size()` - byte at(size_t n) const { return slice_.at(n); } - - /// \brief Accesses the specified byte of data chunk with bounds checking. - /// \throws std::out_of_range if `n >= size()` - byte &at(size_t n) { - if (MDBX_UNLIKELY(n >= size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - return byte_ptr()[n]; - } - - /// \brief Returns the first "n" bytes of the data chunk. - /// \pre REQUIRES: `n <= size()` - ::mdbx::slice head(size_t n) const noexcept { return slice_.head(n); } - - /// \brief Returns the last "n" bytes of the data chunk. - /// \pre REQUIRES: `n <= size()` - ::mdbx::slice tail(size_t n) const noexcept { return slice_.tail(n); } - - /// \brief Returns the middle "n" bytes of the data chunk. - /// \pre REQUIRES: `from + n <= size()` - ::mdbx::slice middle(size_t from, size_t n) const noexcept { - return slice_.middle(from, n); - } - - /// \brief Returns the first "n" bytes of the data chunk. - /// \throws std::out_of_range if `n >= size()` - ::mdbx::slice safe_head(size_t n) const { return slice_.safe_head(n); } - - /// \brief Returns the last "n" bytes of the data chunk. - /// \throws std::out_of_range if `n >= size()` - ::mdbx::slice safe_tail(size_t n) const { return slice_.safe_tail(n); } - - /// \brief Returns the middle "n" bytes of the data chunk. - /// \throws std::out_of_range if `from + n >= size()` - ::mdbx::slice safe_middle(size_t from, size_t n) const { - return slice_.safe_middle(from, n); - } - - inline buffer &append(const void *src, size_t bytes); - - buffer &append(const ::mdbx::slice &chunk) { - return append(chunk.data(), chunk.size()); - } - - inline buffer &add_header(const void *src, size_t bytes); - - buffer &add_header(const ::mdbx::slice &chunk) { - return add_header(chunk.data(), chunk.size()); - } - - //---------------------------------------------------------------------------- - - template <size_t SIZE> - static buffer key_from(const char (&text)[SIZE], bool make_reference = true) { - return buffer(::mdbx::slice(text), make_reference); - } - -#if defined(DOXYGEN) || \ - (defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L) - template <class C, class T> - static buffer key_from(const ::std::basic_string_view<C, T> &src, - bool make_reference = false) { - return buffer(src, make_reference); - } -#endif /* __cpp_lib_string_view >= 201606L */ - - static buffer key_from(const char *src, bool make_reference = false) { - return buffer(src, make_reference); - } - - template <class C, class T, class A> - static buffer key_from(const ::std::basic_string<C, T, A> &src, - bool make_reference = false) { - return buffer(src, make_reference); - } - - static buffer key_from(const silo &&src) noexcept { - return buffer(::std::move(src)); - } - - static buffer key_from(const double ieee754_64bit) { - return wrap(::mdbx_key_from_double(ieee754_64bit)); - } - - static buffer key_from(const double *ieee754_64bit) { - return wrap(::mdbx_key_from_ptrdouble(ieee754_64bit)); - } - - static buffer key_from(const uint64_t unsigned_int64) { - return wrap(unsigned_int64); - } - - static buffer key_from(const int64_t signed_int64) { - return wrap(::mdbx_key_from_int64(signed_int64)); - } - - static buffer key_from_jsonInteger(const int64_t json_integer) { - return wrap(::mdbx_key_from_jsonInteger(json_integer)); - } - - static buffer key_from(const float ieee754_32bit) { - return wrap(::mdbx_key_from_float(ieee754_32bit)); - } - - static buffer key_from(const float *ieee754_32bit) { - return wrap(::mdbx_key_from_ptrfloat(ieee754_32bit)); - } - - static buffer key_from(const uint32_t unsigned_int32) { - return wrap(unsigned_int32); - } - - static buffer key_from(const int32_t signed_int32) { - return wrap(::mdbx_key_from_int32(signed_int32)); - } -}; - -/// \brief Combines data slice with boolean flag to represent result of certain -/// operations. -struct value_result { - slice value; - bool done; - value_result(const slice &value, bool done) noexcept - : value(value), done(done) {} - value_result(const value_result &) noexcept = default; - value_result &operator=(const value_result &) noexcept = default; - MDBX_CXX14_CONSTEXPR operator bool() const noexcept { - assert(!done || bool(value)); - return done; - } -}; - -/// \brief Combines pair of slices for key and value to represent result of -/// certain operations. -struct pair { - slice key, value; - pair(const slice &key, const slice &value) noexcept - : key(key), value(value) {} - pair(const pair &) noexcept = default; - pair &operator=(const pair &) noexcept = default; - MDBX_CXX14_CONSTEXPR operator bool() const noexcept { - assert(bool(key) == bool(value)); - return key; - } -}; - -/// \brief Combines pair of slices for key and value with boolean flag to -/// represent result of certain operations. -struct pair_result : public pair { - bool done; - pair_result(const slice &key, const slice &value, bool done) noexcept - : pair(key, value), done(done) {} - pair_result(const pair_result &) noexcept = default; - pair_result &operator=(const pair_result &) noexcept = default; - MDBX_CXX14_CONSTEXPR operator bool() const noexcept { - assert(!done || (bool(key) && bool(value))); - return done; - } -}; - -//------------------------------------------------------------------------------ - -/// \brief Loop control constants for readers enumeration functor and other -/// cases. \see env::enumerate_readers() -enum loop_control { continue_loop = 0, exit_loop = INT32_MIN }; - -/// \brief Kinds of the keys and corresponding modes of comparing it. -enum class key_mode { - usual = MDBX_DB_DEFAULTS, ///< Usual variable length keys with byte-by-byte - ///< lexicographic comparison like `std::memcmp()`. - reverse = MDBX_REVERSEKEY, ///< Variable length keys with byte-by-byte - ///< lexicographic comparison in reverse order, - ///< from the end of the keys to the beginning. - ordinal = MDBX_INTEGERKEY, ///< Keys are binary integers in native byte order, - ///< either `uint32_t` or `uint64_t`, and will be - ///< sorted as such. The keys must all be of the - ///< same size and must be aligned while passing - ///< as arguments. - msgpack = -1 ///< Keys are in [MessagePack](https://msgpack.org/) - ///< format with appropriate comparison. - ///< \note Not yet implemented and PRs are welcome. -}; - -/// \brief Kind of the values and sorted multi-values with corresponding -/// comparison. -enum class value_mode { - single = MDBX_DB_DEFAULTS, ///< Usual single value for each key. In terms of - ///< keys, they are unique. - multi = - MDBX_DUPSORT, ///< A more than one data value could be associated with - ///< each key. Internally each key is stored once, and the - ///< corresponding data values are sorted by byte-by-byte - ///< lexicographic comparison like `std::memcmp()`. - ///< In terms of keys, they are not unique, i.e. has - ///< duplicates which are sorted by associated data values. -#if CONSTEXPR_ENUM_FLAGS_OPERATIONS || defined(DOXYGEN) - multi_reverse = - MDBX_DUPSORT | - MDBX_REVERSEDUP, ///< A more than one data value could be associated with - ///< each key. Internally each key is stored once, and - ///< the corresponding data values are sorted by - ///< byte-by-byte lexicographic comparison in reverse - ///< order, from the end of the keys to the beginning. - ///< In terms of keys, they are not unique, i.e. has - ///< duplicates which are sorted by associated data - ///< values. - multi_samelength = - MDBX_DUPSORT | - MDBX_DUPFIXED, ///< A more than one data value could be associated with - ///< each key, and all data values must be same length. - ///< Internally each key is stored once, and the - ///< corresponding data values are sorted by byte-by-byte - ///< lexicographic comparison like `std::memcmp()`. In - ///< terms of keys, they are not unique, i.e. has - ///< duplicates which are sorted by associated data values. - multi_ordinal = - MDBX_DUPSORT | MDBX_DUPFIXED | - MDBX_INTEGERDUP, ///< A more than one data value could be associated with - ///< each key, and all data values are binary integers in - ///< native byte order, either `uint32_t` or `uint64_t`, - ///< and will be sorted as such. Internally each key is - ///< stored once, and the corresponding data values are - ///< sorted. In terms of keys, they are not unique, i.e. - ///< has duplicates which are sorted by associated data - ///< values. - multi_reverse_samelength = - MDBX_DUPSORT | MDBX_REVERSEDUP | - MDBX_DUPFIXED, ///< A more than one data value could be associated with - ///< each key, and all data values must be same length. - ///< Internally each key is stored once, and the - ///< corresponding data values are sorted by byte-by-byte - ///< lexicographic comparison in reverse order, from the - ///< end of the keys to the beginning. In terms of keys, - ///< they are not unique, i.e. has duplicates which are - ///< sorted by associated data values. - msgpack = -1 ///< A more than one data value could be associated with each - ///< key. Values are in [MessagePack](https://msgpack.org/) - ///< format with appropriate comparison. Internally each key is - ///< stored once, and the corresponding data values are sorted. - ///< In terms of keys, they are not unique, i.e. has duplicates - ///< which are sorted by associated data values. - ///< \note Not yet implemented and PRs are welcome. -#else - multi_reverse = uint32_t(MDBX_DUPSORT) | uint32_t(MDBX_REVERSEDUP), - multi_samelength = uint32_t(MDBX_DUPSORT) | uint32_t(MDBX_DUPFIXED), - multi_ordinal = uint32_t(MDBX_DUPSORT) | uint32_t(MDBX_DUPFIXED) | - uint32_t(MDBX_INTEGERDUP), - multi_reverse_samelength = uint32_t(MDBX_DUPSORT) | - uint32_t(MDBX_REVERSEDUP) | uint32_t(MDBX_DUPFIXED) -#endif -}; - -/// \brief A handle for an individual database (key-value spaces) in the -/// environment. -/// \see txn::open_map() \see txn::create_map() -/// \see txn::clear_map() \see txn::drop_map() -/// \see txn::get_handle_info() \see txn::get_map_stat() -/// \see env::close_amp() -/// \see cursor::map() -struct LIBMDBX_API_TYPE map_handle { - MDBX_dbi dbi{0}; - MDBX_CXX11_CONSTEXPR map_handle() noexcept {} - MDBX_CXX11_CONSTEXPR map_handle(MDBX_dbi dbi) noexcept : dbi(dbi) {} - map_handle(const map_handle &) noexcept = default; - map_handle &operator=(const map_handle &) noexcept = default; - operator bool() const noexcept { return dbi != 0; } - - using flags = ::MDBX_db_flags_t; - using state = ::MDBX_dbi_state_t; - struct LIBMDBX_API_TYPE info { - map_handle::flags flags; - map_handle::state state; - MDBX_CXX11_CONSTEXPR info(map_handle::flags flags, - map_handle::state state) noexcept; - info(const info &) noexcept = default; - info &operator=(const info &) noexcept = default; - MDBX_CXX11_CONSTEXPR ::mdbx::key_mode key_mode() const noexcept; - MDBX_CXX11_CONSTEXPR ::mdbx::value_mode value_mode() const noexcept; - }; -}; - -/// \brief Key-value pairs put mode. -enum put_mode { - insert_unique = MDBX_NOOVERWRITE, ///< Insert only unique keys. - upsert = MDBX_UPSERT, ///< Insert or update. - update = MDBX_CURRENT, ///< Update existing, don't insert new. -}; - -/// \brief Unmanaged database environment. -/// -/// Like other unmanaged classes, `env` allows copying and assignment for -/// instances, but does not destroys the represented underlying object from the -/// own class destructor. -/// -/// An environment supports multiple key-value sub-databases (aka key-value -/// spaces or tables), all residing in the same shared-memory map. -class LIBMDBX_API_TYPE env { - friend class txn; - -protected: - MDBX_env *handle_{nullptr}; - MDBX_CXX11_CONSTEXPR env(MDBX_env *ptr) noexcept; - -public: - MDBX_CXX11_CONSTEXPR env() noexcept = default; - env(const env &) noexcept = default; - inline env &operator=(env &&other) noexcept; - inline env(env &&other) noexcept; - inline ~env() noexcept; - - MDBX_CXX14_CONSTEXPR operator bool() const noexcept; - MDBX_CXX14_CONSTEXPR operator const MDBX_env *() const; - MDBX_CXX14_CONSTEXPR operator MDBX_env *(); - friend MDBX_CXX11_CONSTEXPR bool operator==(const env &a, - const env &b) noexcept; - friend MDBX_CXX11_CONSTEXPR bool operator!=(const env &a, - const env &b) noexcept; - - //---------------------------------------------------------------------------- - - /// Database geometry for size management. - struct LIBMDBX_API_TYPE geometry { - enum : int64_t { - default_value = -1, ///< Means "keep current or use default" - minimal_value = 0, ///< Means "minimal acceptable" - maximal_value = INTPTR_MAX, ///< Means "maximal acceptable" - kB = 1000, ///< \f$10^{3}\f$ bytes - MB = kB * 1000, ///< \f$10^{6}\f$ bytes - GB = MB * 1000, ///< \f$10^{9}\f$ bytes - TB = GB * 1000, ///< \f$10^{12}\f$ bytes - PB = TB * 1000, ///< \f$10^{15}\f$ bytes - EB = PB * 1000, ///< \f$10^{18}\f$ bytes - KiB = 1024, ///< \f$2^{10}\f$ bytes - MiB = KiB << 10, ///< \f$2^{20}\f$ bytes - GiB = MiB << 10, ///< \f$2^{30}\f$ bytes - TiB = GiB << 10, ///< \f$2^{40}\f$ bytes - PiB = TiB << 10, ///< \f$2^{50}\f$ bytes - EiB = PiB << 10, ///< \f$2^{60}\f$ bytes - }; - - /// \brief Tagged type for output to std::ostream - struct size { - intptr_t bytes; - MDBX_CXX11_CONSTEXPR size(intptr_t bytes) noexcept : bytes(bytes) {} - MDBX_CXX11_CONSTEXPR operator intptr_t() const noexcept { return bytes; } - }; - - /// \brief The lower bound of database size in bytes. - intptr_t size_lower{minimal_value}; - - /// \brief The size in bytes to setup the database size for now. - /// \details It is recommended always pass \ref default_value in this - /// argument except some special cases. - intptr_t size_now{default_value}; - - /// \brief The upper bound of database size in bytes. - /// \details It is recommended to avoid change upper bound while database is - /// used by other processes or threaded (i.e. just pass \ref default_value - /// in this argument except absolutely necessary). Otherwise you must be - /// ready for \ref MDBX_UNABLE_EXTEND_MAPSIZE error(s), unexpected pauses - /// during remapping and/or system errors like "address busy", and so on. In - /// other words, there is no way to handle a growth of the upper bound - /// robustly because there may be a lack of appropriate system resources - /// (which are extremely volatile in a multi-process multi-threaded - /// environment). - intptr_t size_upper{maximal_value}; - - /// \brief The growth step in bytes, must be greater than zero to allow the - /// database to grow. - intptr_t growth_step{default_value}; - - /// \brief The shrink threshold in bytes, must be greater than zero to allow - /// the database to shrink. - intptr_t shrink_threshold{default_value}; - - /// \brief The database page size for new database creation - /// or \ref default_value otherwise. - /// \details Must be power of 2 in the range between \ref MDBX_MIN_PAGESIZE - /// and \ref MDBX_MAX_PAGESIZE. - intptr_t pagesize{default_value}; - - inline geometry &make_fixed(intptr_t size) noexcept; - inline geometry &make_dynamic(intptr_t lower = minimal_value, - intptr_t upper = maximal_value) noexcept; - }; - - /// \brief Operation mode. - enum mode { - readonly, ///< \copydoc MDBX_RDONLY - write_file_io, // don't available on OpenBSD - write_mapped_io ///< \copydoc MDBX_WRITEMAP - }; - - /// \brief Durability level. - enum durability { - robust_synchronous, ///< \copydoc MDBX_SYNC_DURABLE - half_synchronous_weak_last, ///< \copydoc MDBX_NOMETASYNC - lazy_weak_tail, ///< \copydoc MDBX_SAFE_NOSYNC - whole_fragile ///< \copydoc MDBX_UTTERLY_NOSYNC - }; - - /// \brief Garbage reclaiming options. - struct LIBMDBX_API_TYPE reclaiming_options { - /// \copydoc MDBX_LIFORECLAIM - bool lifo{false}; - /// \copydoc MDBX_COALESCE - bool coalesce{false}; - MDBX_CXX11_CONSTEXPR reclaiming_options() noexcept {} - reclaiming_options(MDBX_env_flags_t) noexcept; - }; - - /// \brief Operate options. - struct LIBMDBX_API_TYPE operate_options { - /// \copydoc MDBX_NOTLS - bool orphan_read_transactions{false}; - bool nested_write_transactions{false}; - /// \copydoc MDBX_EXCLUSIVE - bool exclusive{false}; - /// \copydoc MDBX_NORDAHEAD - bool disable_readahead{false}; - /// \copydoc MDBX_NOMEMINIT - bool disable_clear_memory{false}; - MDBX_CXX11_CONSTEXPR operate_options() noexcept {} - operate_options(MDBX_env_flags_t) noexcept; - }; - - /// \brief Operate parameters. - struct LIBMDBX_API_TYPE operate_parameters { - /// \brief The maximum number of named databases for the environment. - /// Zero means default value. - unsigned max_maps{0}; - /// \brief The maximum number of threads/reader slots for the environment. - /// Zero means default value. - unsigned max_readers{0}; - env::mode mode{write_mapped_io}; - env::durability durability{robust_synchronous}; - env::reclaiming_options reclaiming; - env::operate_options options; - - MDBX_CXX11_CONSTEXPR operate_parameters() noexcept {} - MDBX_env_flags_t make_flags(bool accede = true, ///< \copydoc MDBX_ACCEDE - bool use_subdirectory = false) const; - static env::mode mode_from_flags(MDBX_env_flags_t) noexcept; - static env::durability durability_from_flags(MDBX_env_flags_t) noexcept; - inline static env::reclaiming_options - reclaiming_from_flags(MDBX_env_flags_t flags) noexcept; - inline static env::operate_options - options_from_flags(MDBX_env_flags_t flags) noexcept; - operate_parameters(const env &); - }; - - /// \brief Returns current operation parameters. - inline env::operate_parameters get_operation_parameters() const; - /// \brief Returns current operation mode. - inline env::mode get_mode() const; - /// \brief Returns current durability mode. - inline env::durability get_durability() const; - /// \brief Returns current reclaiming options. - inline env::reclaiming_options get_reclaiming() const; - /// \brief Returns current operate options. - inline env::operate_options get_options() const; - - /// \brief Returns `true` for a freshly created database, - /// but `false` if at least one transaction was committed. - bool is_pristine() const; - - /// \brief Checks whether the database is empty. - bool is_empty() const; - - /// \brief Returns default page size for current system/platform. - static size_t default_pagesize() noexcept { - return ::mdbx_default_pagesize(); - } - - struct limits { - limits() = delete; - /// \brief Returns the minimal database page size in bytes. - static inline size_t pagesize_min() noexcept; - /// \brief Returns the maximal database page size in bytes. - static inline size_t pagesize_max() noexcept; - /// \brief Returns the minimal database size in bytes for specified page - /// size. - static inline size_t dbsize_min(intptr_t pagesize); - /// \brief Returns the maximal database size in bytes for specified page - /// size. - static inline size_t dbsize_max(intptr_t pagesize); - /// \brief Returns the minimal key size in bytes for specified database - /// flags. - static inline size_t key_min(MDBX_db_flags_t flags) noexcept; - /// \brief Returns the minimal key size in bytes for specified keys mode. - static inline size_t key_min(key_mode mode) noexcept; - /// \brief Returns the maximal key size in bytes for specified page size and - /// database flags. - static inline size_t key_max(intptr_t pagesize, MDBX_db_flags_t flags); - /// \brief Returns the maximal key size in bytes for specified page size and - /// keys mode. - static inline size_t key_max(intptr_t pagesize, key_mode mode); - /// \brief Returns the maximal key size in bytes for given environment and - /// database flags. - static inline size_t key_max(const env &, MDBX_db_flags_t flags); - /// \brief Returns the maximal key size in bytes for given environment and - /// keys mode. - static inline size_t key_max(const env &, key_mode mode); - /// \brief Returns the minimal values size in bytes for specified database - /// flags. - static inline size_t value_min(MDBX_db_flags_t flags) noexcept; - /// \brief Returns the minimal values size in bytes for specified values - /// mode. - static inline size_t value_min(value_mode) noexcept; - /// \brief Returns the maximal value size in bytes for specified page size - /// and database flags. - static inline size_t value_max(intptr_t pagesize, MDBX_db_flags_t flags); - /// \brief Returns the maximal value size in bytes for specified page size - /// and values mode. - static inline size_t value_max(intptr_t pagesize, value_mode); - /// \brief Returns the maximal value size in bytes for given environment and - /// database flags. - static inline size_t value_max(const env &, MDBX_db_flags_t flags); - /// \brief Returns the maximal value size in bytes for specified page size - /// and values mode. - static inline size_t value_max(const env &, value_mode); - /// \brief Returns the maximal write transaction size (i.e. limit for - /// summary volume of dirty pages) in bytes for specified page size. - static inline size_t transaction_size_max(intptr_t pagesize); - }; - - /// \brief Returns the minimal database size in bytes for the environment. - size_t dbsize_min() const { return limits::dbsize_min(this->get_pagesize()); } - /// \brief Returns the maximal database size in bytes for the environment. - size_t dbsize_max() const { return limits::dbsize_max(this->get_pagesize()); } - /// \brief Returns the minimal key size in bytes for specified keys mode. - size_t key_min(key_mode mode) const noexcept { return limits::key_min(mode); } - /// \brief Returns the maximal key size in bytes for specified keys mode. - size_t key_max(key_mode mode) const { return limits::key_max(*this, mode); } - /// \brief Returns the minimal value size in bytes for specified values mode. - size_t value_min(value_mode mode) const noexcept { - return limits::value_min(mode); - } - /// \brief Returns the maximal value size in bytes for specified values mode. - size_t value_max(value_mode mode) const { - return limits::value_max(*this, mode); - } - /// \brief Returns the maximal write transaction size (i.e. limit for summary - /// volume of dirty pages) in bytes. - size_t transaction_size_max() const { - return limits::transaction_size_max(this->get_pagesize()); - } - - /// \brief Make a copy (backup) of an existing environment to the specified - /// path. -#ifdef MDBX_STD_FILESYSTEM_PATH - env ©(const ::std::filesystem::path &destination, bool compactify, - bool force_dynamic_size = false); -#endif /* MDBX_STD_FILESYSTEM_PATH */ -#if defined(_WIN32) || defined(_WIN64) - env ©(const ::std::wstring &destination, bool compactify, - bool force_dynamic_size = false); -#endif /* Windows */ - env ©(const ::std::string &destination, bool compactify, - bool force_dynamic_size = false); - - /// \brief Copy an environment to the specified file descriptor. - env ©(filehandle fd, bool compactify, bool force_dynamic_size = false); - - /// \brief Deletion modes for \ref remove(). - enum remove_mode { - /// \brief Just delete the environment's files and directory if any. - /// \note On POSIX systems, processes already working with the database will - /// continue to work without interference until it close the environment. - /// \note On Windows, the behavior of `just_remove` is different - /// because the system does not support deleting files that are currently - /// memory mapped. - just_remove = MDBX_ENV_JUST_DELETE, - /// \brief Make sure that the environment is not being used by other - /// processes, or return an error otherwise. - ensure_unused = MDBX_ENV_ENSURE_UNUSED, - /// \brief Wait until other processes closes the environment before - /// deletion. - wait_for_unused = MDBX_ENV_WAIT_FOR_UNUSED - }; - - /// \brief Removes the environment's files in a proper and multiprocess-safe - /// way. -#ifdef MDBX_STD_FILESYSTEM_PATH - static bool remove(const ::std::filesystem::path &, - const remove_mode mode = just_remove); -#endif /* MDBX_STD_FILESYSTEM_PATH */ -#if defined(_WIN32) || defined(_WIN64) - static bool remove(const ::std::wstring &, - const remove_mode mode = just_remove); -#endif /* Windows */ - static bool remove(const ::std::string &, - const remove_mode mode = just_remove); - - /// \brief Statistics for a database in the MDBX environment. - using stat = ::MDBX_stat; - - /// \brief Information about the environment. - using info = ::MDBX_envinfo; - - /// \brief Returns snapshot statistics about the MDBX environment. - inline stat get_stat() const; - - /// \brief Returns pagesize of this MDBX environment. - size_t get_pagesize() const { return get_stat().ms_psize; } - - /// \brief Return snapshot information about the MDBX environment. - inline info get_info() const; - - /// \brief Return statistics about the MDBX environment accordingly to the - /// specified transaction. - inline stat get_stat(const txn &) const; - - /// \brief Return information about the MDBX environment accordingly to the - /// specified transaction. - inline info get_info(const txn &) const; - - /// \brief Returns the file descriptor for the DXB file of MDBX environment. - inline filehandle get_filehandle() const; - - /// \brief Return the path that was used for opening the environment. - path get_path() const; - - /// Returns environment flags. - inline MDBX_env_flags_t get_flags() const; - - /// \brief Returns the maximum number of threads/reader slots for the - /// environment. - inline unsigned max_readers() const; - - /// \brief Returns the maximum number of named databases for the environment. - inline unsigned max_maps() const; - - /// \brief Returns the application context associated with the environment. - inline void *get_context() const noexcept; - - /// \brief Sets the application context associated with the environment. - inline env &set_context(void *); - - /// \brief Sets threshold to force flush the data buffers to disk, for - /// non-sync durability modes. - /// - /// The threshold value affects all processes which operates with given - /// environment until the last process close environment or a new value will - /// be settled. - /// Data is always written to disk when \ref txn_managed::commit() is called, - /// but the operating system may keep it buffered. MDBX always flushes the OS - /// buffers upon commit as well, unless the environment was opened with \ref - /// whole_fragile, \ref lazy_weak_tail or in part \ref - /// half_synchronous_weak_last. The default is 0, than mean no any threshold - /// checked, and no additional flush will be made. - /// - inline env &set_sync_threshold(size_t bytes); - - /// \brief Sets relative period since the last unsteady commit to force flush - /// the data buffers to disk, for non-sync durability modes. - /// - /// The relative period value affects all processes which operates with given - /// environment until the last process close environment or a new value will - /// be settled. - /// Data is always written to disk when \ref txn_managed::commit() is called, - /// but the operating system may keep it buffered. MDBX always flushes the OS - /// buffers upon commit as well, unless the environment was opened with \ref - /// whole_fragile, \ref lazy_weak_tail or in part \ref - /// half_synchronous_weak_last. Settled period don't checked asynchronously, - /// but only by the \ref txn_managed::commit() and \ref env::sync_to_disk() - /// functions. Therefore, in cases where transactions are committed - /// infrequently and/or irregularly, polling by \ref env::poll_sync_to_disk() - /// may be a reasonable solution to timeout enforcement. The default is 0, - /// than mean no any timeout checked, and no additional flush will be made. - /// - /// \param [in] seconds_16dot16 The period in 1/65536 of second when a - /// synchronous flush would be made since the last unsteady commit. - inline env &set_sync_period(unsigned seconds_16dot16); - - /// \brief Sets relative period since the last unsteady commit to force flush - /// the data buffers to disk, for non-sync durability modes. - /// - /// The relative period value affects all processes which operates with given - /// environment until the last process close environment or a new value will - /// be settled. - /// Data is always written to disk when \ref txn_managed::commit() is called, - /// but the operating system may keep it buffered. MDBX always flushes the OS - /// buffers upon commit as well, unless the environment was opened with \ref - /// whole_fragile, \ref lazy_weak_tail or in part \ref - /// half_synchronous_weak_last. Settled period don't checked asynchronously, - /// but only by the \ref txn_managed::commit() and \ref env::sync_to_disk() - /// functions. Therefore, in cases where transactions are committed - /// infrequently and/or irregularly, polling by \ref env::poll_sync_to_disk() - /// may be a reasonable solution to timeout enforcement. The default is 0, - /// than mean no any timeout checked, and no additional flush will be made. - /// - /// \param [in] seconds The period in second when a synchronous flush would - /// be made since the last unsteady commit. - inline env &set_sync_period(double seconds); - - /// \brief Alter environment flags. - inline env &alter_flags(MDBX_env_flags_t flags, bool on_off); - - /// \brief Set all size-related parameters of environment. - inline env &set_geometry(const geometry &size); - - /// \brief Flush the environment data buffers. - /// \return `True` if sync done or no data to sync, or `false` if the - /// environment is busy by other thread or none of the thresholds are reached. - inline bool sync_to_disk(bool force = true, bool nonblock = false); - - /// \brief Performs non-blocking polling of sync-to-disk thresholds. - /// \return `True` if sync done or no data to sync, or `false` if the - /// environment is busy by other thread or none of the thresholds are reached. - bool poll_sync_to_disk() { return sync_to_disk(false, true); } - - /// \brief Close a key-value map (aka sub-database) handle. Normally - /// unnecessary. - /// - /// Closing a database handle is not necessary, but lets \ref txn::open_map() - /// reuse the handle value. Usually it's better to set a bigger - /// \ref env::operate_parameters::max_maps, unless that value would be - /// large. - /// - /// \note Use with care. - /// This call is synchronized via mutex with other calls \ref close_map(), but - /// NOT with other transactions running by other threads. The "next" version - /// of libmdbx (\ref MithrilDB) will solve this issue. - /// - /// Handles should only be closed if no other threads are going to reference - /// the database handle or one of its cursors any further. Do not close a - /// handle if an existing transaction has modified its database. Doing so can - /// cause misbehavior from database corruption to errors like - /// \ref MDBX_BAD_DBI (since the DB name is gone). - inline void close_map(const map_handle &); - - /// \brief Reader information - struct reader_info { - int slot; ///< The reader lock table slot number. - mdbx_pid_t pid; ///< The reader process ID. - mdbx_tid_t thread; ///< The reader thread ID. - uint64_t transaction_id; ///< The ID of the transaction being read, - ///< i.e. the MVCC-snapshot number. - uint64_t transaction_lag; ///< The lag from a recent MVCC-snapshot, - ///< i.e. the number of committed write - /// transactions since the current read - /// transaction started. - size_t bytes_used; ///< The number of last used page in the MVCC-snapshot - ///< which being read, i.e. database file can't shrinked - ///< beyond this. - size_t bytes_retained; ///< The total size of the database pages that - ///< were retired by committed write transactions - ///< after the reader's MVCC-snapshot, i.e. the space - ///< which would be freed after the Reader releases - ///< the MVCC-snapshot for reuse by completion read - ///< transaction. - - MDBX_CXX11_CONSTEXPR reader_info(int slot, mdbx_pid_t pid, - mdbx_tid_t thread, uint64_t txnid, - uint64_t lag, size_t used, - size_t retained) noexcept; - }; - - /// \brief Enumerate readers. - /// - /// The VISITOR class must have `int operator(const reader_info&, int serial)` - /// which should return \ref continue_loop (zero) to continue enumeration, - /// or any non-zero value to exit. - /// - /// \returns The last value returned from visitor' functor. - template <typename VISITOR> inline int enumerate_readers(VISITOR &visitor); - - /// \brief Checks for stale readers in the lock table and - /// return number of cleared slots. - inline unsigned check_readers(); - - /// \brief Sets a Handle-Slow-Readers callback to resolve database - /// full/overflow issue due to a reader(s) which prevents the old data from - /// being recycled. - /// - /// Such callback will be triggered in a case where there is not enough free - /// space in the database due to long read transaction(s) which impedes - /// reusing the pages of an old MVCC snapshot(s). - /// - /// Using this callback you can choose how to resolve the situation: - /// - abort the write transaction with an error; - /// - wait for the read transaction(s) to complete; - /// - notify a thread performing a long-lived read transaction - /// and wait for an effect; - /// - kill the thread or whole process that performs the long-lived read - /// transaction; - /// - /// \see long-lived-read - inline env &set_HandleSlowReaders(MDBX_hsr_func *); - - /// \brief Returns the current Handle-Slow-Readers callback used to resolve - /// database full/overflow issue due to a reader(s) which prevents the old - /// data from being recycled. - /// \see set_HandleSlowReaders() - inline MDBX_hsr_func *get_HandleSlowReaders() const noexcept; - - /// \brief Starts read (read-only) transaction. - inline txn_managed start_read() const; - - /// \brief Creates but not start read transaction. - inline txn_managed prepare_read() const; - - /// \brief Starts write (read-write) transaction. - inline txn_managed start_write(bool dont_wait = false); - - /// \brief Tries to start write (read-write) transaction without blocking. - inline txn_managed try_start_write(); -}; - -/// \brief Managed database environment. -/// -/// As other managed classes, `env_managed` destroys the represented underlying -/// object from the own class destructor, but disallows copying and assignment -/// for instances. -/// -/// An environment supports multiple key-value databases (aka key-value spaces -/// or tables), all residing in the same shared-memory map. -class LIBMDBX_API_TYPE env_managed : public env { - using inherited = env; - /// delegated constructor for RAII - MDBX_CXX11_CONSTEXPR env_managed(MDBX_env *ptr) noexcept : inherited(ptr) {} - void setup(unsigned max_maps, unsigned max_readers = 0); - -public: - MDBX_CXX11_CONSTEXPR env_managed() noexcept = default; - - /// \brief Open existing database. -#ifdef MDBX_STD_FILESYSTEM_PATH - env_managed(const ::std::filesystem::path &, const operate_parameters &, - bool accede = true); -#endif /* MDBX_STD_FILESYSTEM_PATH */ -#if defined(_WIN32) || defined(_WIN64) - env_managed(const ::std::wstring &, const operate_parameters &, - bool accede = true); -#endif /* Windows */ - env_managed(const ::std::string &, const operate_parameters &, - bool accede = true); - - /// \brief Additional parameters for creating a new database. - struct create_parameters { - env::geometry geometry; - mdbx_mode_t file_mode_bits{0640}; - bool use_subdirectory{false}; - }; - - /// \brief Create new or open existing database. -#ifdef MDBX_STD_FILESYSTEM_PATH - env_managed(const ::std::filesystem::path &, const create_parameters &, - const operate_parameters &, bool accede = true); -#endif /* MDBX_STD_FILESYSTEM_PATH */ -#if defined(_WIN32) || defined(_WIN64) - env_managed(const ::std::wstring &, const create_parameters &, - const operate_parameters &, bool accede = true); -#endif /* Windows */ - env_managed(const ::std::string &, const create_parameters &, - const operate_parameters &, bool accede = true); - - /// \brief Explicitly closes the environment and release the memory map. - /// - /// Only a single thread may call this function. All transactions, databases, - /// and cursors must already be closed before calling this function. Attempts - /// to use any such handles after calling this function will cause a - /// `SIGSEGV`. The environment handle will be freed and must not be used again - /// after this call. - /// - /// \param [in] dont_sync A dont'sync flag, if non-zero the last checkpoint - /// will be kept "as is" and may be still "weak" in the \ref lazy_weak_tail - /// or \ref whole_fragile modes. Such "weak" checkpoint will be ignored - /// on opening next time, and transactions since the last non-weak checkpoint - /// (meta-page update) will rolledback for consistency guarantee. - void close(bool dont_sync = false); - - env_managed(env_managed &&) = default; - env_managed &operator=(env_managed &&) = default; - env_managed(const env_managed &) = delete; - env_managed &operator=(const env_managed &) = delete; - virtual ~env_managed() noexcept; -}; - -/// \brief Unmanaged database transaction. -/// -/// Like other unmanaged classes, `txn` allows copying and assignment for -/// instances, but does not destroys the represented underlying object from the -/// own class destructor. -/// -/// All database operations require a transaction handle. Transactions may be -/// read-only or read-write. -class LIBMDBX_API_TYPE txn { -protected: - friend class cursor; - MDBX_txn *handle_{nullptr}; - MDBX_CXX11_CONSTEXPR txn(MDBX_txn *ptr) noexcept; - -public: - MDBX_CXX11_CONSTEXPR txn() noexcept = default; - txn(const txn &) noexcept = default; - inline txn &operator=(txn &&other) noexcept; - inline txn(txn &&other) noexcept; - inline ~txn() noexcept; - - MDBX_CXX14_CONSTEXPR operator bool() const noexcept; - MDBX_CXX14_CONSTEXPR operator const MDBX_txn *() const; - MDBX_CXX14_CONSTEXPR operator MDBX_txn *(); - friend MDBX_CXX11_CONSTEXPR bool operator==(const txn &a, - const txn &b) noexcept; - friend MDBX_CXX11_CONSTEXPR bool operator!=(const txn &a, - const txn &b) noexcept; - - /// \brief Returns the transaction's environment. - inline ::mdbx::env env() const noexcept; - /// \brief Returns transaction's flags. - inline MDBX_txn_flags_t flags() const; - /// \brief Return the transaction's ID. - inline uint64_t id() const; - - /// \brief Checks whether the given data is on a dirty page. - inline bool is_dirty(const void *ptr) const; - - /// \brief Checks whether the transaction is read-only. - bool is_readonly() const { return (flags() & MDBX_TXN_RDONLY) != 0; } - - /// \brief Checks whether the transaction is read-write. - bool is_readwrite() const { return (flags() & MDBX_TXN_RDONLY) == 0; } - - using info = ::MDBX_txn_info; - /// \brief Returns information about the MDBX transaction. - inline info get_info(bool scan_reader_lock_table = false) const; - - /// \brief Returns maximal write transaction size (i.e. limit for summary - /// volume of dirty pages) in bytes. - size_t size_max() const { return env().transaction_size_max(); } - - /// \brief Returns current write transaction size (i.e.summary volume of dirty - /// pages) in bytes. - size_t size_current() const { - assert(is_readwrite()); - return size_t(get_info().txn_space_dirty); - } - - //---------------------------------------------------------------------------- - - /// \brief Reset a read-only transaction. - inline void reset_reading(); - - /// \brief Renew a read-only transaction. - inline void renew_reading(); - - /// \brief Start nested write transaction. - txn_managed start_nested(); - - /// \brief Opens cursor for specified key-value map handle. - inline cursor_managed open_cursor(map_handle map); - - /// \brief Open existing key-value map. - inline map_handle open_map( - const char *name, - const ::mdbx::key_mode key_mode = ::mdbx::key_mode::usual, - const ::mdbx::value_mode value_mode = ::mdbx::value_mode::single) const; - /// \brief Open existing key-value map. - inline map_handle open_map( - const ::std::string &name, - const ::mdbx::key_mode key_mode = ::mdbx::key_mode::usual, - const ::mdbx::value_mode value_mode = ::mdbx::value_mode::single) const; - - /// \brief Create new or open existing key-value map. - inline map_handle - create_map(const char *name, - const ::mdbx::key_mode key_mode = ::mdbx::key_mode::usual, - const ::mdbx::value_mode value_mode = ::mdbx::value_mode::single); - /// \brief Create new or open existing key-value map. - inline map_handle - create_map(const ::std::string &name, - const ::mdbx::key_mode key_mode = ::mdbx::key_mode::usual, - const ::mdbx::value_mode value_mode = ::mdbx::value_mode::single); - - /// \brief Drops key-value map using handle. - inline void drop_map(map_handle map); - /// \brief Drops key-value map using name. - /// \return `True` if the key-value map existed and was deleted, either - /// `false` if the key-value map did not exist and there is nothing to delete. - bool drop_map(const char *name, bool throw_if_absent = false); - /// \brief Drop key-value map. - /// \return `True` if the key-value map existed and was deleted, either - /// `false` if the key-value map did not exist and there is nothing to delete. - inline bool drop_map(const ::std::string &name, bool throw_if_absent = false); - - /// \brief Clear key-value map. - inline void clear_map(map_handle map); - /// \return `True` if the key-value map existed and was cleared, either - /// `false` if the key-value map did not exist and there is nothing to clear. - bool clear_map(const char *name, bool throw_if_absent = false); - /// \return `True` if the key-value map existed and was cleared, either - /// `false` if the key-value map did not exist and there is nothing to clear. - inline bool clear_map(const ::std::string &name, - bool throw_if_absent = false); - - using map_stat = ::MDBX_stat; - /// \brief Returns statistics for a sub-database. - inline map_stat get_map_stat(map_handle map) const; - /// \brief Returns depth (bitmask) information of nested dupsort (multi-value) - /// B+trees for given database. - inline uint32_t get_tree_deepmask(map_handle map) const; - /// \brief Returns information about key-value map (aka sub-database) handle. - inline map_handle::info get_handle_info(map_handle map) const; - - using canary = ::MDBX_canary; - /// \brief Set integers markers (aka "canary") associated with the - /// environment. - inline txn &put_canary(const canary &); - /// \brief Returns fours integers markers (aka "canary") associated with the - /// environment. - inline canary get_canary() const; - - /// Reads sequence generator associated with a key-value map (aka - /// sub-database). - inline uint64_t sequence(map_handle map) const; - /// \brief Reads and increment sequence generator associated with a key-value - /// map (aka sub-database). - inline uint64_t sequence(map_handle map, uint64_t increment); - - /// \brief Compare two keys according to a particular key-value map (aka - /// sub-database). - inline int compare_keys(map_handle map, const slice &a, - const slice &b) const noexcept; - /// \brief Compare two values according to a particular key-value map (aka - /// sub-database). - inline int compare_values(map_handle map, const slice &a, - const slice &b) const noexcept; - /// \brief Compare keys of two pairs according to a particular key-value map - /// (aka sub-database). - inline int compare_keys(map_handle map, const pair &a, - const pair &b) const noexcept; - /// \brief Compare values of two pairs according to a particular key-value map - /// (aka sub-database). - inline int compare_values(map_handle map, const pair &a, - const pair &b) const noexcept; - - /// \brief Get value by key from a key-value map (aka sub-database). - inline slice get(map_handle map, const slice &key) const; - /// \brief Get first of multi-value and values count by key from a key-value - /// multimap (aka sub-database). - inline slice get(map_handle map, slice key, size_t &values_count) const; - /// \brief Get value by key from a key-value map (aka sub-database). - inline slice get(map_handle map, const slice &key, - const slice &value_at_absence) const; - /// \brief Get first of multi-value and values count by key from a key-value - /// multimap (aka sub-database). - inline slice get(map_handle map, slice key, size_t &values_count, - const slice &value_at_absence) const; - /// \brief Get value for equal or great key from a database. - /// \return Bundle of key-value pair and boolean flag, - /// which will be `true` if the exact key was found and `false` otherwise. - inline pair_result get_equal_or_great(map_handle map, const slice &key) const; - /// \brief Get value for equal or great key from a database. - /// \return Bundle of key-value pair and boolean flag, - /// which will be `true` if the exact key was found and `false` otherwise. - inline pair_result get_equal_or_great(map_handle map, const slice &key, - const slice &value_at_absence) const; - - inline MDBX_error_t put(map_handle map, const slice &key, slice *value, - MDBX_put_flags_t flags) noexcept; - inline void put(map_handle map, const slice &key, slice value, - put_mode mode) noexcept; - inline void insert(map_handle map, const slice &key, slice value); - inline value_result try_insert(map_handle map, const slice &key, slice value); - inline slice insert_reserve(map_handle map, const slice &key, - size_t value_length); - inline value_result try_insert_reserve(map_handle map, const slice &key, - size_t value_length); - - inline void upsert(map_handle map, const slice &key, const slice &value); - inline slice upsert_reserve(map_handle map, const slice &key, - size_t value_length); - - inline void update(map_handle map, const slice &key, const slice &value); - inline bool try_update(map_handle map, const slice &key, const slice &value); - inline slice update_reserve(map_handle map, const slice &key, - size_t value_length); - inline value_result try_update_reserve(map_handle map, const slice &key, - size_t value_length); - - inline bool erase(map_handle map, const slice &key); - - /// \brief Removes the particular multi-value entry of the key. - inline bool erase(map_handle map, const slice &key, const slice &value); - - /// \brief Replaces the particular multi-value of the key with a new value. - inline void replace(map_handle map, const slice &key, slice old_value, - const slice &new_value); - - /// \brief Removes and return a value of the key. - template <class ALLOCATOR> - inline buffer<ALLOCATOR> extract(map_handle map, const slice &key, - const ALLOCATOR &allocator = ALLOCATOR()); - - /// \brief Replaces and returns a value of the key with new one. - template <class ALLOCATOR> - inline buffer<ALLOCATOR> replace(map_handle map, const slice &key, - const slice &new_value, - const ALLOCATOR &allocator = ALLOCATOR()); - - template <class ALLOCATOR> - inline buffer<ALLOCATOR> - replace_reserve(map_handle map, const slice &key, slice &new_value, - const ALLOCATOR &allocator = ALLOCATOR()); - - /// \brief Adding a key-value pair, provided that ascending order of the keys - /// and (optionally) values are preserved. - /// - /// Instead of splitting the full b+tree pages, the data will be placed on new - /// ones. Thus appending is about two times faster than insertion, and the - /// pages will be filled in completely mostly but not half as after splitting - /// ones. On the other hand, any subsequent insertion or update with an - /// increase in the length of the value will be twice as slow, since it will - /// require splitting already filled pages. - /// - /// \param [in] multivalue_order_preserved - /// If `multivalue_order_preserved == true` then the same rules applied for - /// to pages of nested b+tree of multimap's values. - inline void append(map_handle map, const slice &key, const slice &value, - bool multivalue_order_preserved = true); - - size_t put_multiple(map_handle map, const slice &key, - const size_t value_length, const void *values_array, - size_t values_count, put_mode mode, - bool allow_partial = false); - template <typename VALUE> - void put_multiple(map_handle map, const slice &key, - const std::vector<VALUE> &vector, put_mode mode) { - put_multiple(map, key, sizeof(VALUE), vector.data(), vector.size(), mode, - false); - } - - inline ptrdiff_t estimate(map_handle map, pair from, pair to) const; - inline ptrdiff_t estimate(map_handle map, slice from, slice to) const; - inline ptrdiff_t estimate_from_first(map_handle map, slice to) const; - inline ptrdiff_t estimate_to_last(map_handle map, slice from) const; -}; - -/// \brief Managed database transaction. -/// -/// As other managed classes, `txn_managed` destroys the represented underlying -/// object from the own class destructor, but disallows copying and assignment -/// for instances. -/// -/// All database operations require a transaction handle. Transactions may be -/// read-only or read-write. -class LIBMDBX_API_TYPE txn_managed : public txn { - using inherited = txn; - friend class env; - friend class txn; - /// delegated constructor for RAII - MDBX_CXX11_CONSTEXPR txn_managed(MDBX_txn *ptr) noexcept : inherited(ptr) {} - -public: - MDBX_CXX11_CONSTEXPR txn_managed() noexcept = default; - txn_managed(txn_managed &&) = default; - txn_managed &operator=(txn_managed &&) = default; - txn_managed(const txn_managed &) = delete; - txn_managed &operator=(const txn_managed &) = delete; - ~txn_managed() noexcept; - - //---------------------------------------------------------------------------- - - /// \brief Abandon all the operations of the transaction instead of saving - /// them. - void abort(); - - /// \brief Commit all the operations of a transaction into the database. - void commit(); -}; - -/// \brief Unmanaged cursor. -/// -/// Like other unmanaged classes, `cursor` allows copying and assignment for -/// instances, but does not destroys the represented underlying object from the -/// own class destructor. -/// -/// \copydetails MDBX_cursor -class LIBMDBX_API_TYPE cursor { -protected: - MDBX_cursor *handle_{nullptr}; - MDBX_CXX11_CONSTEXPR cursor(MDBX_cursor *ptr) noexcept; - -public: - MDBX_CXX11_CONSTEXPR cursor() noexcept = default; - cursor(const cursor &) noexcept = default; - inline cursor &operator=(cursor &&other) noexcept; - inline cursor(cursor &&other) noexcept; - inline ~cursor() noexcept; - MDBX_CXX14_CONSTEXPR operator bool() const noexcept; - MDBX_CXX14_CONSTEXPR operator const MDBX_cursor *() const; - MDBX_CXX14_CONSTEXPR operator MDBX_cursor *(); - friend MDBX_CXX11_CONSTEXPR bool operator==(const cursor &a, - const cursor &b) noexcept; - friend MDBX_CXX11_CONSTEXPR bool operator!=(const cursor &a, - const cursor &b) noexcept; - - enum move_operation { - first = MDBX_FIRST, - last = MDBX_LAST, - next = MDBX_NEXT, - previous = MDBX_PREV, - get_current = MDBX_GET_CURRENT, - - multi_prevkey_lastvalue = MDBX_PREV_NODUP, - multi_currentkey_firstvalue = MDBX_FIRST_DUP, - multi_currentkey_prevvalue = MDBX_PREV_DUP, - multi_currentkey_nextvalue = MDBX_NEXT_DUP, - multi_currentkey_lastvalue = MDBX_LAST_DUP, - multi_nextkey_firstvalue = MDBX_NEXT_NODUP, - - multi_find_pair = MDBX_GET_BOTH, - multi_exactkey_lowerboundvalue = MDBX_GET_BOTH_RANGE, - - find_key = MDBX_SET, - key_exact = MDBX_SET_KEY, - key_lowerbound = MDBX_SET_RANGE - }; - - struct move_result : public pair_result { - inline move_result(const cursor &cursor, bool throw_notfound); - inline move_result(cursor &cursor, move_operation operation, - bool throw_notfound); - inline move_result(cursor &cursor, move_operation operation, - const slice &key, bool throw_notfound); - inline move_result(cursor &cursor, move_operation operation, - const slice &key, const slice &value, - bool throw_notfound); - move_result(const move_result &) noexcept = default; - }; - -protected: - inline bool move(move_operation operation, MDBX_val *key, MDBX_val *value, - bool throw_notfound) const - /* fake const, i.e. for some operations */; - inline ptrdiff_t estimate(move_operation operation, MDBX_val *key, - MDBX_val *value) const; - -public: - inline move_result move(move_operation operation, bool throw_notfound); - inline move_result to_first(bool throw_notfound = true); - inline move_result to_previous(bool throw_notfound = true); - inline move_result to_previous_last_multi(bool throw_notfound = true); - inline move_result to_current_first_multi(bool throw_notfound = true); - inline move_result to_current_prev_multi(bool throw_notfound = true); - inline move_result current(bool throw_notfound = true) const; - inline move_result to_current_next_multi(bool throw_notfound = true); - inline move_result to_current_last_multi(bool throw_notfound = true); - inline move_result to_next_first_multi(bool throw_notfound = true); - inline move_result to_next(bool throw_notfound = true); - inline move_result to_last(bool throw_notfound = true); - - inline move_result move(move_operation operation, const slice &key, - bool throw_notfound); - inline move_result find(const slice &key, bool throw_notfound = true); - inline move_result lower_bound(const slice &key, bool throw_notfound = true); - - inline move_result move(move_operation operation, const slice &key, - const slice &value, bool throw_notfound); - inline move_result find_multivalue(const slice &key, const slice &value, - bool throw_notfound = true); - inline move_result lower_bound_multivalue(const slice &key, - const slice &value, - bool throw_notfound = false); - - inline bool seek(const slice &key); - inline bool move(move_operation operation, slice &key, slice &value, - bool throw_notfound); - - /// \brief Return count of duplicates for current key. - inline size_t count_multivalue() const; - - inline bool eof() const; - inline bool on_first() const; - inline bool on_last() const; - inline ptrdiff_t estimate(slice key, slice value) const; - inline ptrdiff_t estimate(slice key) const; - inline ptrdiff_t estimate(move_operation operation) const; - - //---------------------------------------------------------------------------- - - /// \brief Renew/bind a cursor with a new transaction and previously used - /// key-value map handle. - inline void renew(::mdbx::txn &txn); - - /// \brief Bind/renew a cursor with a new transaction and specified key-value - /// map handle. - inline void bind(::mdbx::txn &txn, ::mdbx::map_handle map_handle); - - /// \brief Returns the cursor's transaction. - inline ::mdbx::txn txn() const; - inline map_handle map() const; - - inline operator ::mdbx::txn() const { return txn(); } - inline operator ::mdbx::map_handle() const { return map(); } - - inline MDBX_error_t put(const slice &key, slice *value, - MDBX_put_flags_t flags) noexcept; - inline void insert(const slice &key, slice value); - inline value_result try_insert(const slice &key, slice value); - inline slice insert_reserve(const slice &key, size_t value_length); - inline value_result try_insert_reserve(const slice &key, size_t value_length); - - inline void upsert(const slice &key, const slice &value); - inline slice upsert_reserve(const slice &key, size_t value_length); - - inline void update(const slice &key, const slice &value); - inline bool try_update(const slice &key, const slice &value); - inline slice update_reserve(const slice &key, size_t value_length); - inline value_result try_update_reserve(const slice &key, size_t value_length); - - inline bool erase(bool whole_multivalue = false); -}; - -/// \brief Managed cursor. -/// -/// As other managed classes, `cursor_managed` destroys the represented -/// underlying object from the own class destructor, but disallows copying and -/// assignment for instances. -/// -/// \copydetails MDBX_cursor -class LIBMDBX_API_TYPE cursor_managed : public cursor { - using inherited = cursor; - friend class txn; - /// delegated constructor for RAII - MDBX_CXX11_CONSTEXPR cursor_managed(MDBX_cursor *ptr) noexcept - : inherited(ptr) {} - -public: - /// \brief Creates a new managed cursor with underlying object. - inline cursor_managed(); - - /// \brief Explicitly closes the cursor. - void close(); - - cursor_managed(cursor_managed &&) = default; - cursor_managed &operator=(cursor_managed &&) = default; - cursor_managed(const cursor_managed &) = delete; - cursor_managed &operator=(const cursor_managed &) = delete; - ~cursor_managed() noexcept { ::mdbx_cursor_close(handle_); } -}; - -//------------------------------------------------------------------------------ - -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, const slice &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, const pair &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, const pair_result &); -template <class ALLOCATOR> -inline ::std::ostream &operator<<(::std::ostream &out, - const buffer<ALLOCATOR> &it) { - return (it.is_freestanding() - ? out << "buf-" << it.headroom() << "." << it.tailroom() - : out << "ref-") - << it.slice(); -} -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const env::geometry::size &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, const env::geometry &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const env::operate_parameters &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, const env::mode &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const env::durability &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const env::reclaiming_options &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const env::operate_options &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const env_managed::create_parameters &); - -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const MDBX_log_level_t &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, - const MDBX_debug_flags_t &); -LIBMDBX_API ::std::ostream &operator<<(::std::ostream &, const error &); -inline ::std::ostream &operator<<(::std::ostream &out, - const MDBX_error_t &errcode) { - return out << error(errcode); -} - -//============================================================================== -// -// Inline body of the libmdbx C++ API (preliminary draft) -// - -MDBX_CXX11_CONSTEXPR const version_info &get_version() noexcept { - return ::mdbx_version; -} -MDBX_CXX11_CONSTEXPR const build_info &get_build() noexcept { - return ::mdbx_build; -} - -static MDBX_CXX17_CONSTEXPR size_t strlen(const char *c_str) noexcept { -#if defined(__cpp_lib_is_constant_evaluated) && \ - __cpp_lib_is_constant_evaluated >= 201811L - if (::std::is_constant_evaluated()) { - for (size_t i = 0; c_str; ++i) - if (!c_str[i]) - return i; - return 0; - } -#endif /* __cpp_lib_is_constant_evaluated >= 201811 */ -#if defined(__cpp_lib_string_view) && __cpp_lib_string_view >= 201606L - return c_str ? ::std::string_view(c_str).length() : 0; -#else - return c_str ? ::std::strlen(c_str) : 0; -#endif -} - -MDBX_CXX14_CONSTEXPR size_t check_length(size_t bytes) { - if (MDBX_UNLIKELY(bytes > size_t(MDBX_MAXDATASIZE))) - MDBX_CXX20_UNLIKELY throw_max_length_exceeded(); - return bytes; -} - -inline bool exception_thunk::is_clean() const noexcept { return !captured_; } - -inline void exception_thunk::capture() noexcept { - assert(is_clean()); - captured_ = ::std::current_exception(); -} - -inline void exception_thunk::rethrow_captured() const { - if (captured_) - MDBX_CXX20_UNLIKELY ::std::rethrow_exception(captured_); -} - -//------------------------------------------------------------------------------ - -MDBX_CXX11_CONSTEXPR error::error(MDBX_error_t error_code) noexcept - : code_(error_code) {} - -inline error &error::operator=(MDBX_error_t error_code) noexcept { - code_ = error_code; - return *this; -} - -MDBX_CXX11_CONSTEXPR bool operator==(const error &a, const error &b) noexcept { - return a.code_ == b.code_; -} - -MDBX_CXX11_CONSTEXPR bool operator!=(const error &a, const error &b) noexcept { - return !(a == b); -} - -MDBX_CXX11_CONSTEXPR bool error::is_success() const noexcept { - return code_ == MDBX_SUCCESS; -} - -MDBX_CXX11_CONSTEXPR bool error::is_result_true() const noexcept { - return code_ == MDBX_RESULT_FALSE; -} - -MDBX_CXX11_CONSTEXPR bool error::is_result_false() const noexcept { - return code_ == MDBX_RESULT_TRUE; -} - -MDBX_CXX11_CONSTEXPR bool error::is_failure() const noexcept { - return code_ != MDBX_SUCCESS && code_ != MDBX_RESULT_TRUE; -} - -MDBX_CXX11_CONSTEXPR MDBX_error_t error::code() const noexcept { return code_; } - -MDBX_CXX11_CONSTEXPR bool error::is_mdbx_error() const noexcept { - return (code() >= MDBX_FIRST_LMDB_ERRCODE && - code() <= MDBX_LAST_LMDB_ERRCODE) || - (code() >= MDBX_FIRST_ADDED_ERRCODE && - code() <= MDBX_LAST_ADDED_ERRCODE); -} - -inline void error::throw_exception(int error_code) { - const error trouble(static_cast<MDBX_error_t>(error_code)); - trouble.throw_exception(); -} - -inline void error::throw_on_failure() const { - if (MDBX_UNLIKELY(is_failure())) - MDBX_CXX20_UNLIKELY throw_exception(); -} - -inline void error::success_or_throw() const { - if (MDBX_UNLIKELY(!is_success())) - MDBX_CXX20_UNLIKELY throw_exception(); -} - -inline void error::success_or_throw(const exception_thunk &thunk) const { - assert(thunk.is_clean() || code() != MDBX_SUCCESS); - if (MDBX_UNLIKELY(!is_success())) { - MDBX_CXX20_UNLIKELY if (!thunk.is_clean()) thunk.rethrow_captured(); - else throw_exception(); - } -} - -inline void error::panic_on_failure(const char *context_where, - const char *func_who) const noexcept { - if (MDBX_UNLIKELY(is_failure())) - MDBX_CXX20_UNLIKELY panic(context_where, func_who); -} - -inline void error::success_or_panic(const char *context_where, - const char *func_who) const noexcept { - if (MDBX_UNLIKELY(!is_success())) - MDBX_CXX20_UNLIKELY panic(context_where, func_who); -} - -inline void error::throw_on_nullptr(const void *ptr, MDBX_error_t error_code) { - if (MDBX_UNLIKELY(ptr == nullptr)) - MDBX_CXX20_UNLIKELY error(error_code).throw_exception(); -} - -inline void error::throw_on_failure(int error_code) { - error rc(static_cast<MDBX_error_t>(error_code)); - rc.throw_on_failure(); -} - -inline void error::success_or_throw(MDBX_error_t error_code) { - error rc(error_code); - rc.success_or_throw(); -} - -inline bool error::boolean_or_throw(int error_code) { - switch (error_code) { - case MDBX_RESULT_FALSE: - return false; - case MDBX_RESULT_TRUE: - return true; - default: - MDBX_CXX20_UNLIKELY throw_exception(error_code); - } -} - -inline void error::success_or_throw(int error_code, - const exception_thunk &thunk) { - error rc(static_cast<MDBX_error_t>(error_code)); - rc.success_or_throw(thunk); -} - -inline void error::panic_on_failure(int error_code, const char *context_where, - const char *func_who) noexcept { - error rc(static_cast<MDBX_error_t>(error_code)); - rc.panic_on_failure(context_where, func_who); -} - -inline void error::success_or_panic(int error_code, const char *context_where, - const char *func_who) noexcept { - error rc(static_cast<MDBX_error_t>(error_code)); - rc.success_or_panic(context_where, func_who); -} - -//------------------------------------------------------------------------------ - -MDBX_CXX11_CONSTEXPR slice::slice() noexcept : ::MDBX_val({nullptr, 0}) {} - -MDBX_CXX14_CONSTEXPR slice::slice(const void *ptr, size_t bytes) - : ::MDBX_val({const_cast<void *>(ptr), check_length(bytes)}) {} - -MDBX_CXX14_CONSTEXPR slice::slice(const void *begin, const void *end) - : slice(begin, static_cast<const byte *>(end) - - static_cast<const byte *>(begin)) {} - -MDBX_CXX17_CONSTEXPR slice::slice(const char *c_str) - : slice(c_str, ::mdbx::strlen(c_str)) {} - -MDBX_CXX14_CONSTEXPR slice::slice(const MDBX_val &src) - : slice(src.iov_base, src.iov_len) {} - -inline slice::slice(MDBX_val &&src) : slice(src) { src.iov_base = nullptr; } - -inline slice::slice(slice &&src) noexcept : slice(src) { src.invalidate(); } - -inline slice &slice::assign(const void *ptr, size_t bytes) { - iov_base = const_cast<void *>(ptr); - iov_len = check_length(bytes); - return *this; -} - -inline slice &slice::assign(const slice &src) noexcept { - iov_base = src.iov_base; - iov_len = src.iov_len; - return *this; -} - -inline slice &slice::assign(const ::MDBX_val &src) { - return assign(src.iov_base, src.iov_len); -} - -slice &slice::assign(slice &&src) noexcept { - assign(src); - src.invalidate(); - return *this; -} - -inline slice &slice::assign(::MDBX_val &&src) { - assign(src.iov_base, src.iov_len); - src.iov_base = nullptr; - return *this; -} - -inline slice &slice::assign(const void *begin, const void *end) { - return assign(begin, static_cast<const byte *>(end) - - static_cast<const byte *>(begin)); -} - -inline slice &slice::assign(const char *c_str) { - return assign(c_str, ::mdbx::strlen(c_str)); -} - -inline slice &slice::operator=(slice &&src) noexcept { - return assign(::std::move(src)); -} - -inline slice &slice::operator=(::MDBX_val &&src) { - return assign(::std::move(src)); -} - -inline void slice::swap(slice &other) noexcept { - const auto temp = *this; - *this = other; - other = temp; -} - -MDBX_CXX11_CONSTEXPR const mdbx::byte *slice::byte_ptr() const noexcept { - return static_cast<const byte *>(iov_base); -} - -MDBX_CXX11_CONSTEXPR const char *slice::char_ptr() const noexcept { - return static_cast<const char *>(iov_base); -} - -MDBX_CXX11_CONSTEXPR const void *slice::data() const noexcept { - return iov_base; -} - -MDBX_CXX11_CONSTEXPR size_t slice::length() const noexcept { return iov_len; } - -MDBX_CXX11_CONSTEXPR bool slice::empty() const noexcept { - return length() == 0; -} - -MDBX_CXX11_CONSTEXPR bool slice::is_null() const noexcept { - return data() == nullptr; -} - -MDBX_CXX11_CONSTEXPR size_t slice::size() const noexcept { return length(); } - -MDBX_CXX11_CONSTEXPR slice::operator bool() const noexcept { - return !is_null(); -} - -inline void slice::invalidate() noexcept { iov_base = nullptr; } - -inline void slice::clear() noexcept { - iov_base = nullptr; - iov_len = 0; -} - -inline void slice::remove_prefix(size_t n) noexcept { - assert(n <= size()); - iov_base = static_cast<byte *>(iov_base) + n; - iov_len -= n; -} - -inline void slice::safe_remove_prefix(size_t n) { - if (MDBX_UNLIKELY(n > size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - remove_prefix(n); -} - -inline void slice::remove_suffix(size_t n) noexcept { - assert(n <= size()); - iov_len -= n; -} - -inline void slice::safe_remove_suffix(size_t n) { - if (MDBX_UNLIKELY(n > size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - remove_suffix(n); -} - -inline bool slice::starts_with(const slice &prefix) const noexcept { - return length() >= prefix.length() && - ::std::memcmp(data(), prefix.data(), prefix.length()) == 0; -} - -inline bool slice::ends_with(const slice &suffix) const noexcept { - return length() >= suffix.length() && - ::std::memcmp(byte_ptr() + length() - suffix.length(), suffix.data(), - suffix.length()) == 0; -} - -MDBX_NOTHROW_PURE_FUNCTION MDBX_CXX14_CONSTEXPR size_t -slice::hash_value() const noexcept { - size_t h = length() * 3977471; - for (size_t i = 0; i < length(); ++i) - h = (h ^ static_cast<const uint8_t *>(data())[i]) * 1664525 + 1013904223; - return h ^ 3863194411 * (h >> 11); -} - -inline byte slice::operator[](size_t n) const noexcept { - assert(n < size()); - return byte_ptr()[n]; -} - -inline byte slice::at(size_t n) const { - if (MDBX_UNLIKELY(n >= size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - return byte_ptr()[n]; -} - -inline slice slice::head(size_t n) const noexcept { - assert(n <= size()); - return slice(data(), n); -} - -inline slice slice::tail(size_t n) const noexcept { - assert(n <= size()); - return slice(char_ptr() + size() - n, n); -} - -inline slice slice::middle(size_t from, size_t n) const noexcept { - assert(from + n <= size()); - return slice(char_ptr() + from, n); -} - -inline slice slice::safe_head(size_t n) const { - if (MDBX_UNLIKELY(n > size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - return head(n); -} - -inline slice slice::safe_tail(size_t n) const { - if (MDBX_UNLIKELY(n > size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - return tail(n); -} - -inline slice slice::safe_middle(size_t from, size_t n) const { - if (MDBX_UNLIKELY(n > max_length)) - MDBX_CXX20_UNLIKELY throw_max_length_exceeded(); - if (MDBX_UNLIKELY(from + n > size())) - MDBX_CXX20_UNLIKELY throw_out_range(); - return middle(from, n); -} - -inline intptr_t slice::compare_fast(const slice &a, const slice &b) noexcept { - const intptr_t diff = a.length() - b.length(); - return diff ? diff - : (a.data() == b.data()) - ? 0 - : ::std::memcmp(a.data(), b.data(), a.length()); -} - -inline intptr_t slice::compare_lexicographically(const slice &a, - const slice &b) noexcept { - const intptr_t diff = - ::std::memcmp(a.data(), b.data(), ::std::min(a.length(), b.length())); - return diff ? diff : intptr_t(a.length() - b.length()); -} - -MDBX_NOTHROW_PURE_FUNCTION inline bool operator==(const slice &a, - const slice &b) noexcept { - return slice::compare_fast(a, b) == 0; -} - -MDBX_NOTHROW_PURE_FUNCTION inline bool operator<(const slice &a, - const slice &b) noexcept { - return slice::compare_lexicographically(a, b) < 0; -} - -MDBX_NOTHROW_PURE_FUNCTION inline bool operator>(const slice &a, - const slice &b) noexcept { - return slice::compare_lexicographically(a, b) > 0; -} - -MDBX_NOTHROW_PURE_FUNCTION inline bool operator<=(const slice &a, - const slice &b) noexcept { - return slice::compare_lexicographically(a, b) <= 0; -} - -MDBX_NOTHROW_PURE_FUNCTION inline bool operator>=(const slice &a, - const slice &b) noexcept { - return slice::compare_lexicographically(a, b) >= 0; -} - -MDBX_NOTHROW_PURE_FUNCTION inline bool operator!=(const slice &a, - const slice &b) noexcept { - return slice::compare_fast(a, b) != 0; -} - -template <class ALLOCATOR> -inline ::mdbx::string<ALLOCATOR> -slice::hex_encode(bool uppercase, const ALLOCATOR &allocator) const { - ::mdbx::string<ALLOCATOR> result(allocator); - if (MDBX_LIKELY(length() > 0)) { - result.resize(to_hex_bytes()); - result.resize(to_hex(const_cast<char *>(result.data()), result.capacity()) - - result.data(), - uppercase); - } - return result; -} - -template <class ALLOCATOR> -inline ::mdbx::string<ALLOCATOR> -slice::hex_decode(const ALLOCATOR &allocator) const { - ::mdbx::string<ALLOCATOR> result(allocator); - if (MDBX_LIKELY(length() > 0)) { - result.resize(from_hex_bytes()); - result.resize( - from_hex(static_cast<byte *>( - static_cast<void *>(const_cast<char *>(result.data()))), - result.capacity()) - - static_cast<const byte *>(static_cast<const void *>(result.data()))); - } - return result; -} - -template <class ALLOCATOR> -inline ::mdbx::string<ALLOCATOR> -slice::base58_encode(const ALLOCATOR &allocator) const { - ::mdbx::string<ALLOCATOR> result(allocator); - if (MDBX_LIKELY(length() > 0)) { - result.resize(to_base58_bytes()); - result.resize( - to_base58(const_cast<char *>(result.data()), result.capacity()) - - result.data()); - } - return result; -} - -template <class ALLOCATOR> -inline ::mdbx::string<ALLOCATOR> -slice::base58_decode(const ALLOCATOR &allocator) const { - ::mdbx::string<ALLOCATOR> result(allocator); - if (MDBX_LIKELY(length() > 0)) { - result.resize(from_base58_bytes()); - result.resize( - from_base58(static_cast<byte *>( - static_cast<void *>(const_cast<char *>(result.data()))), - result.capacity()) - - static_cast<const byte *>(static_cast<const void *>(result.data()))); - } - return result; -} - -template <class ALLOCATOR> -inline ::mdbx::string<ALLOCATOR> -slice::base64_encode(const ALLOCATOR &allocator) const { - ::mdbx::string<ALLOCATOR> result(allocator); - if (MDBX_LIKELY(length() > 0)) { - result.resize(to_base64_bytes()); - result.resize( - to_base64(const_cast<char *>(result.data()), result.capacity()) - - result.data()); - } - return result; -} - -template <class ALLOCATOR> -inline ::mdbx::string<ALLOCATOR> -slice::base64_decode(const ALLOCATOR &allocator) const { - ::mdbx::string<ALLOCATOR> result(allocator); - if (MDBX_LIKELY(length() > 0)) { - result.resize(from_base64_bytes()); - result.resize( - from_base64(static_cast<byte *>( - static_cast<void *>(const_cast<char *>(result.data()))), - result.capacity()) - - static_cast<const byte *>(static_cast<const void *>(result.data()))); - } - return result; -} - -//------------------------------------------------------------------------------ - -MDBX_CXX11_CONSTEXPR map_handle::info::info(map_handle::flags flags, - map_handle::state state) noexcept - : flags(flags), state(state) {} - -MDBX_CXX11_CONSTEXPR ::mdbx::key_mode -map_handle::info::key_mode() const noexcept { - return ::mdbx::key_mode(flags & (MDBX_REVERSEKEY | MDBX_INTEGERKEY)); -} - -MDBX_CXX11_CONSTEXPR ::mdbx::value_mode -map_handle::info::value_mode() const noexcept { - return ::mdbx::value_mode(flags & (MDBX_DUPSORT | MDBX_REVERSEDUP | - MDBX_DUPFIXED | MDBX_INTEGERDUP)); -} - -//------------------------------------------------------------------------------ - -MDBX_CXX11_CONSTEXPR env::env(MDBX_env *ptr) noexcept : handle_(ptr) {} - -inline env &env::operator=(env &&other) noexcept { - handle_ = other.handle_; - other.handle_ = nullptr; - return *this; -} - -inline env::env(env &&other) noexcept : handle_(other.handle_) { - other.handle_ = nullptr; -} - -inline env::~env() noexcept { -#ifndef NDEBUG - handle_ = reinterpret_cast<MDBX_env *>(uintptr_t(0xDeadBeef)); -#endif -} - -MDBX_CXX14_CONSTEXPR env::operator bool() const noexcept { - return handle_ != nullptr; -} - -MDBX_CXX14_CONSTEXPR env::operator const MDBX_env *() const { return handle_; } - -MDBX_CXX14_CONSTEXPR env::operator MDBX_env *() { return handle_; } - -MDBX_CXX11_CONSTEXPR bool operator==(const env &a, const env &b) noexcept { - return a.handle_ == b.handle_; -} - -MDBX_CXX11_CONSTEXPR bool operator!=(const env &a, const env &b) noexcept { - return a.handle_ != b.handle_; -} - -inline env::geometry &env::geometry::make_fixed(intptr_t size) noexcept { - size_lower = size_now = size_upper = size; - growth_step = shrink_threshold = 0; - return *this; -} - -inline env::geometry &env::geometry::make_dynamic(intptr_t lower, - intptr_t upper) noexcept { - size_now = size_lower = lower; - size_upper = upper; - growth_step = shrink_threshold = default_value; - return *this; -} - -inline env::reclaiming_options env::operate_parameters::reclaiming_from_flags( - MDBX_env_flags_t flags) noexcept { - return reclaiming_options(flags); -} - -inline env::operate_options -env::operate_parameters::options_from_flags(MDBX_env_flags_t flags) noexcept { - return operate_options(flags); -} - -inline size_t env::limits::pagesize_min() noexcept { return MDBX_MIN_PAGESIZE; } - -inline size_t env::limits::pagesize_max() noexcept { return MDBX_MAX_PAGESIZE; } - -inline size_t env::limits::dbsize_min(intptr_t pagesize) { - const intptr_t result = mdbx_limits_dbsize_min(pagesize); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline size_t env::limits::dbsize_max(intptr_t pagesize) { - const intptr_t result = mdbx_limits_dbsize_max(pagesize); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline size_t env::limits::key_min(MDBX_db_flags_t flags) noexcept { - return (flags & MDBX_INTEGERKEY) ? 4 : 0; -} - -inline size_t env::limits::key_min(key_mode mode) noexcept { - return key_min(MDBX_db_flags_t(mode)); -} - -inline size_t env::limits::key_max(intptr_t pagesize, MDBX_db_flags_t flags) { - const intptr_t result = mdbx_limits_keysize_max(pagesize, flags); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline size_t env::limits::key_max(intptr_t pagesize, key_mode mode) { - return key_max(pagesize, MDBX_db_flags_t(mode)); -} - -inline size_t env::limits::key_max(const env &env, MDBX_db_flags_t flags) { - const intptr_t result = mdbx_env_get_maxkeysize_ex(env, flags); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline size_t env::limits::key_max(const env &env, key_mode mode) { - return key_max(env, MDBX_db_flags_t(mode)); -} - -inline size_t env::limits::value_min(MDBX_db_flags_t flags) noexcept { - return (flags & MDBX_INTEGERDUP) ? 4 : 0; -} - -inline size_t env::limits::value_min(value_mode mode) noexcept { - return value_min(MDBX_db_flags_t(mode)); -} - -inline size_t env::limits::value_max(intptr_t pagesize, MDBX_db_flags_t flags) { - const intptr_t result = mdbx_limits_valsize_max(pagesize, flags); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline size_t env::limits::value_max(intptr_t pagesize, value_mode mode) { - return value_max(pagesize, MDBX_db_flags_t(mode)); -} - -inline size_t env::limits::value_max(const env &env, MDBX_db_flags_t flags) { - const intptr_t result = mdbx_env_get_maxvalsize_ex(env, flags); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline size_t env::limits::value_max(const env &env, value_mode mode) { - return value_max(env, MDBX_db_flags_t(mode)); -} - -inline size_t env::limits::transaction_size_max(intptr_t pagesize) { - const intptr_t result = mdbx_limits_txnsize_max(pagesize); - if (result < 0) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_EINVAL); - return static_cast<size_t>(result); -} - -inline env::operate_parameters env::get_operation_parameters() const { - return env::operate_parameters(*this); -} - -inline env::mode env::get_mode() const { - return operate_parameters::mode_from_flags(get_flags()); -} - -inline env::durability env::get_durability() const { - return env::operate_parameters::durability_from_flags(get_flags()); -} - -inline env::reclaiming_options env::get_reclaiming() const { - return env::operate_parameters::reclaiming_from_flags(get_flags()); -} - -inline env::operate_options env::get_options() const { - return env::operate_parameters::options_from_flags(get_flags()); -} - -inline env::stat env::get_stat() const { - env::stat r; - error::success_or_throw(::mdbx_env_stat_ex(handle_, nullptr, &r, sizeof(r))); - return r; -} - -inline env::stat env::get_stat(const txn &txn) const { - env::stat r; - error::success_or_throw(::mdbx_env_stat_ex(handle_, txn, &r, sizeof(r))); - return r; -} - -inline env::info env::get_info() const { - env::info r; - error::success_or_throw(::mdbx_env_info_ex(handle_, nullptr, &r, sizeof(r))); - return r; -} - -inline env::info env::get_info(const txn &txn) const { - env::info r; - error::success_or_throw(::mdbx_env_info_ex(handle_, txn, &r, sizeof(r))); - return r; -} - -inline filehandle env::get_filehandle() const { - filehandle fd; - error::success_or_throw(::mdbx_env_get_fd(handle_, &fd)); - return fd; -} - -inline MDBX_env_flags_t env::get_flags() const { - unsigned bits; - error::success_or_throw(::mdbx_env_get_flags(handle_, &bits)); - return MDBX_env_flags_t(bits); -} - -inline unsigned env::max_readers() const { - unsigned r; - error::success_or_throw(::mdbx_env_get_maxreaders(handle_, &r)); - return r; -} - -inline unsigned env::max_maps() const { - unsigned r; - error::success_or_throw(::mdbx_env_get_maxdbs(handle_, &r)); - return r; -} - -inline void *env::get_context() const noexcept { - return mdbx_env_get_userctx(handle_); -} - -inline env &env::set_context(void *ptr) { - error::success_or_throw(::mdbx_env_set_userctx(handle_, ptr)); - return *this; -} - -inline env &env::set_sync_threshold(size_t bytes) { - error::success_or_throw(::mdbx_env_set_syncbytes(handle_, bytes)); - return *this; -} - -inline env &env::set_sync_period(unsigned seconds_16dot16) { - error::success_or_throw(::mdbx_env_set_syncperiod(handle_, seconds_16dot16)); - return *this; -} - -inline env &env::set_sync_period(double seconds) { - return set_sync_period(unsigned(seconds * 65536)); -} - -inline env &env::alter_flags(MDBX_env_flags_t flags, bool on_off) { - error::success_or_throw(::mdbx_env_set_flags(handle_, flags, on_off)); - return *this; -} - -inline env &env::set_geometry(const geometry &geo) { - error::success_or_throw(::mdbx_env_set_geometry( - handle_, geo.size_lower, geo.size_now, geo.size_upper, geo.growth_step, - geo.shrink_threshold, geo.pagesize)); - return *this; -} - -inline bool env::sync_to_disk(bool force, bool nonblock) { - const int err = ::mdbx_env_sync_ex(handle_, force, nonblock); - switch (err) { - case MDBX_SUCCESS /* flush done */: - case MDBX_RESULT_TRUE /* no data pending for flush to disk */: - return true; - case MDBX_BUSY /* the environment is used by other thread */: - return false; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline void env::close_map(const map_handle &handle) { - error::success_or_throw(::mdbx_dbi_close(*this, handle.dbi)); -} - -MDBX_CXX11_CONSTEXPR -env::reader_info::reader_info(int slot, mdbx_pid_t pid, mdbx_tid_t thread, - uint64_t txnid, uint64_t lag, size_t used, - size_t retained) noexcept - : slot(slot), pid(pid), thread(thread), transaction_id(txnid), - transaction_lag(lag), bytes_used(used), bytes_retained(retained) {} - -template <typename VISITOR> -inline int env::enumerate_readers(VISITOR &visitor) { - struct reader_visitor_thunk : public exception_thunk { - VISITOR &visitor_; - static int cb(void *ctx, int number, int slot, mdbx_pid_t pid, - mdbx_tid_t thread, uint64_t txnid, uint64_t lag, size_t used, - size_t retained) noexcept { - reader_visitor_thunk *thunk = static_cast<reader_visitor_thunk *>(ctx); - assert(thunk->is_clean()); - try { - const reader_info info(slot, pid, thread, txnid, lag, used, retained); - return loop_control(thunk->visitor_(info, number)); - } catch (... /* capture any exception to rethrow it over C code */) { - thunk->capture(); - return loop_control::exit_loop; - } - } - MDBX_CXX11_CONSTEXPR reader_visitor_thunk(VISITOR &visitor) noexcept - : visitor_(visitor) {} - }; - reader_visitor_thunk thunk(visitor); - const auto rc = ::mdbx_reader_list(*this, thunk.cb, &thunk); - thunk.rethrow_captured(); - return rc; -} - -inline unsigned env::check_readers() { - int dead_count; - error::throw_on_failure(::mdbx_reader_check(*this, &dead_count)); - assert(dead_count >= 0); - return static_cast<unsigned>(dead_count); -} - -inline env &env::set_HandleSlowReaders(MDBX_hsr_func *cb) { - error::success_or_throw(::mdbx_env_set_hsr(handle_, cb)); - return *this; -} - -inline MDBX_hsr_func *env::get_HandleSlowReaders() const noexcept { - return ::mdbx_env_get_hsr(handle_); -} - -inline txn_managed env::start_read() const { - ::MDBX_txn *ptr; - error::success_or_throw( - ::mdbx_txn_begin(handle_, nullptr, MDBX_TXN_RDONLY, &ptr)); - assert(ptr != nullptr); - return txn_managed(ptr); -} - -inline txn_managed env::prepare_read() const { - ::MDBX_txn *ptr; - error::success_or_throw( - ::mdbx_txn_begin(handle_, nullptr, MDBX_TXN_RDONLY_PREPARE, &ptr)); - assert(ptr != nullptr); - return txn_managed(ptr); -} - -inline txn_managed env::start_write(bool dont_wait) { - ::MDBX_txn *ptr; - error::success_or_throw(::mdbx_txn_begin( - handle_, nullptr, dont_wait ? MDBX_TXN_TRY : MDBX_TXN_READWRITE, &ptr)); - assert(ptr != nullptr); - return txn_managed(ptr); -} - -inline txn_managed env::try_start_write() { return start_write(true); } - -//------------------------------------------------------------------------------ - -MDBX_CXX11_CONSTEXPR txn::txn(MDBX_txn *ptr) noexcept : handle_(ptr) {} - -inline txn &txn::operator=(txn &&other) noexcept { - handle_ = other.handle_; - other.handle_ = nullptr; - return *this; -} - -inline txn::txn(txn &&other) noexcept : handle_(other.handle_) { - other.handle_ = nullptr; -} - -inline txn::~txn() noexcept { -#ifndef NDEBUG - handle_ = reinterpret_cast<MDBX_txn *>(uintptr_t(0xDeadBeef)); -#endif -} - -MDBX_CXX14_CONSTEXPR txn::operator bool() const noexcept { - return handle_ != nullptr; -} - -MDBX_CXX14_CONSTEXPR txn::operator const MDBX_txn *() const { return handle_; } - -MDBX_CXX14_CONSTEXPR txn::operator MDBX_txn *() { return handle_; } - -MDBX_CXX11_CONSTEXPR bool operator==(const txn &a, const txn &b) noexcept { - return a.handle_ == b.handle_; -} - -MDBX_CXX11_CONSTEXPR bool operator!=(const txn &a, const txn &b) noexcept { - return a.handle_ != b.handle_; -} - -inline bool txn::is_dirty(const void *ptr) const { - int err = ::mdbx_is_dirty(handle_, ptr); - switch (err) { - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - case MDBX_RESULT_TRUE: - return true; - case MDBX_RESULT_FALSE: - return false; - } -} - -inline ::mdbx::env txn::env() const noexcept { return ::mdbx_txn_env(handle_); } - -inline MDBX_txn_flags_t txn::flags() const { - const int bits = mdbx_txn_flags(handle_); - error::throw_on_failure((bits != -1) ? MDBX_SUCCESS : MDBX_BAD_TXN); - return static_cast<MDBX_txn_flags_t>(bits); -} - -inline uint64_t txn::id() const { - const uint64_t txnid = mdbx_txn_id(handle_); - error::throw_on_failure(txnid ? MDBX_SUCCESS : MDBX_BAD_TXN); - return txnid; -} - -inline void txn::reset_reading() { - error::success_or_throw(::mdbx_txn_reset(handle_)); -} - -inline void txn::renew_reading() { - error::success_or_throw(::mdbx_txn_renew(handle_)); -} - -inline txn::info txn::get_info(bool scan_reader_lock_table) const { - txn::info r; - error::success_or_throw(::mdbx_txn_info(handle_, &r, scan_reader_lock_table)); - return r; -} - -inline cursor_managed txn::open_cursor(map_handle map) { - MDBX_cursor *ptr; - error::success_or_throw(::mdbx_cursor_open(handle_, map.dbi, &ptr)); - return cursor_managed(ptr); -} - -inline ::mdbx::map_handle -txn::open_map(const char *name, const ::mdbx::key_mode key_mode, - const ::mdbx::value_mode value_mode) const { - ::mdbx::map_handle map; - error::success_or_throw(::mdbx_dbi_open( - handle_, name, MDBX_db_flags_t(key_mode) | MDBX_db_flags_t(value_mode), - &map.dbi)); - assert(map.dbi != 0); - return map; -} - -inline ::mdbx::map_handle -txn::open_map(const ::std::string &name, const ::mdbx::key_mode key_mode, - const ::mdbx::value_mode value_mode) const { - return open_map(name.c_str(), key_mode, value_mode); -} - -inline ::mdbx::map_handle txn::create_map(const char *name, - const ::mdbx::key_mode key_mode, - const ::mdbx::value_mode value_mode) { - ::mdbx::map_handle map; - error::success_or_throw(::mdbx_dbi_open( - handle_, name, - MDBX_CREATE | MDBX_db_flags_t(key_mode) | MDBX_db_flags_t(value_mode), - &map.dbi)); - assert(map.dbi != 0); - return map; -} - -inline ::mdbx::map_handle txn::create_map(const ::std::string &name, - const ::mdbx::key_mode key_mode, - const ::mdbx::value_mode value_mode) { - return create_map(name.c_str(), key_mode, value_mode); -} - -inline void txn::drop_map(map_handle map) { - error::success_or_throw(::mdbx_drop(handle_, map.dbi, true)); -} - -inline bool txn::drop_map(const ::std::string &name, bool throw_if_absent) { - return drop_map(name.c_str(), throw_if_absent); -} - -inline void txn::clear_map(map_handle map) { - error::success_or_throw(::mdbx_drop(handle_, map.dbi, false)); -} - -inline bool txn::clear_map(const ::std::string &name, bool throw_if_absent) { - return clear_map(name.c_str(), throw_if_absent); -} - -inline txn::map_stat txn::get_map_stat(map_handle map) const { - txn::map_stat r; - error::success_or_throw(::mdbx_dbi_stat(handle_, map.dbi, &r, sizeof(r))); - return r; -} - -inline uint32_t txn::get_tree_deepmask(map_handle map) const { - uint32_t r; - error::success_or_throw(::mdbx_dbi_dupsort_depthmask(handle_, map.dbi, &r)); - return r; -} - -inline map_handle::info txn::get_handle_info(map_handle map) const { - unsigned flags, state; - error::success_or_throw( - ::mdbx_dbi_flags_ex(handle_, map.dbi, &flags, &state)); - return map_handle::info(MDBX_db_flags_t(flags), MDBX_dbi_state_t(state)); -} - -inline txn &txn::put_canary(const txn::canary &canary) { - error::success_or_throw(::mdbx_canary_put(handle_, &canary)); - return *this; -} - -inline txn::canary txn::get_canary() const { - txn::canary r; - error::success_or_throw(::mdbx_canary_get(handle_, &r)); - return r; -} - -inline uint64_t txn::sequence(map_handle map) const { - uint64_t result; - error::success_or_throw(::mdbx_dbi_sequence(handle_, map.dbi, &result, 0)); - return result; -} - -inline uint64_t txn::sequence(map_handle map, uint64_t increment) { - uint64_t result; - error::success_or_throw( - ::mdbx_dbi_sequence(handle_, map.dbi, &result, increment)); - return result; -} - -inline int txn::compare_keys(map_handle map, const slice &a, - const slice &b) const noexcept { - return ::mdbx_cmp(handle_, map.dbi, &a, &b); -} - -inline int txn::compare_values(map_handle map, const slice &a, - const slice &b) const noexcept { - return ::mdbx_dcmp(handle_, map.dbi, &a, &b); -} - -inline int txn::compare_keys(map_handle map, const pair &a, - const pair &b) const noexcept { - return compare_keys(map, a.key, b.key); -} - -inline int txn::compare_values(map_handle map, const pair &a, - const pair &b) const noexcept { - return compare_values(map, a.value, b.value); -} - -inline slice txn::get(map_handle map, const slice &key) const { - slice result; - error::success_or_throw(::mdbx_get(handle_, map.dbi, &key, &result)); - return result; -} - -inline slice txn::get(map_handle map, slice key, size_t &values_count) const { - slice result; - error::success_or_throw( - ::mdbx_get_ex(handle_, map.dbi, &key, &result, &values_count)); - return result; -} - -inline slice txn::get(map_handle map, const slice &key, - const slice &value_at_absence) const { - slice result; - const int err = ::mdbx_get(handle_, map.dbi, &key, &result); - switch (err) { - case MDBX_SUCCESS: - return result; - case MDBX_NOTFOUND: - return value_at_absence; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline slice txn::get(map_handle map, slice key, size_t &values_count, - const slice &value_at_absence) const { - slice result; - const int err = ::mdbx_get_ex(handle_, map.dbi, &key, &result, &values_count); - switch (err) { - case MDBX_SUCCESS: - return result; - case MDBX_NOTFOUND: - return value_at_absence; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline pair_result txn::get_equal_or_great(map_handle map, - const slice &key) const { - pair result(key, slice()); - bool exact = !error::boolean_or_throw( - ::mdbx_get_equal_or_great(handle_, map.dbi, &result.key, &result.value)); - return pair_result(result.key, result.value, exact); -} - -inline pair_result -txn::get_equal_or_great(map_handle map, const slice &key, - const slice &value_at_absence) const { - pair result{key, slice()}; - const int err = - ::mdbx_get_equal_or_great(handle_, map.dbi, &result.key, &result.value); - switch (err) { - case MDBX_SUCCESS: - return pair_result{result.key, result.value, true}; - case MDBX_RESULT_TRUE: - return pair_result{result.key, result.value, false}; - case MDBX_NOTFOUND: - return pair_result{key, value_at_absence, false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline MDBX_error_t txn::put(map_handle map, const slice &key, slice *value, - MDBX_put_flags_t flags) noexcept { - return MDBX_error_t(::mdbx_put(handle_, map.dbi, &key, value, flags)); -} - -inline void txn::put(map_handle map, const slice &key, slice value, - put_mode mode) noexcept { - error::success_or_throw(put(map, key, &value, MDBX_put_flags_t(mode))); -} - -inline void txn::insert(map_handle map, const slice &key, slice value) { - error::success_or_throw( - put(map, key, &value /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique))); -} - -inline value_result txn::try_insert(map_handle map, const slice &key, - slice value) { - const int err = - put(map, key, &value /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique)); - switch (err) { - case MDBX_SUCCESS: - return value_result{slice(), true}; - case MDBX_KEYEXIST: - return value_result{value, false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline slice txn::insert_reserve(map_handle map, const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - error::success_or_throw( - put(map, key, &result /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique) | MDBX_RESERVE)); - return result; -} - -inline value_result txn::try_insert_reserve(map_handle map, const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - const int err = - put(map, key, &result /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique) | MDBX_RESERVE); - switch (err) { - case MDBX_SUCCESS: - return value_result{result, true}; - case MDBX_KEYEXIST: - return value_result{result, false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline void txn::upsert(map_handle map, const slice &key, const slice &value) { - error::success_or_throw(put(map, key, const_cast<slice *>(&value), - MDBX_put_flags_t(put_mode::upsert))); -} - -inline slice txn::upsert_reserve(map_handle map, const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - error::success_or_throw(put( - map, key, &result, MDBX_put_flags_t(put_mode::upsert) | MDBX_RESERVE)); - return result; -} - -inline void txn::update(map_handle map, const slice &key, const slice &value) { - error::success_or_throw(put(map, key, const_cast<slice *>(&value), - MDBX_put_flags_t(put_mode::update))); -} - -inline bool txn::try_update(map_handle map, const slice &key, - const slice &value) { - const int err = put(map, key, const_cast<slice *>(&value), - MDBX_put_flags_t(put_mode::update)); - switch (err) { - case MDBX_SUCCESS: - return true; - case MDBX_NOTFOUND: - return false; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline slice txn::update_reserve(map_handle map, const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - error::success_or_throw(put( - map, key, &result, MDBX_put_flags_t(put_mode::update) | MDBX_RESERVE)); - return result; -} - -inline value_result txn::try_update_reserve(map_handle map, const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - const int err = - put(map, key, &result, MDBX_put_flags_t(put_mode::update) | MDBX_RESERVE); - switch (err) { - case MDBX_SUCCESS: - return value_result{result, true}; - case MDBX_NOTFOUND: - return value_result{slice(), false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline bool txn::erase(map_handle map, const slice &key) { - const int err = ::mdbx_del(handle_, map.dbi, &key, nullptr); - switch (err) { - case MDBX_SUCCESS: - return true; - case MDBX_NOTFOUND: - return false; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline bool txn::erase(map_handle map, const slice &key, const slice &value) { - const int err = ::mdbx_del(handle_, map.dbi, &key, &value); - switch (err) { - case MDBX_SUCCESS: - return true; - case MDBX_NOTFOUND: - return false; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline void txn::replace(map_handle map, const slice &key, slice old_value, - const slice &new_value) { - error::success_or_throw(::mdbx_replace_ex( - handle_, map.dbi, &key, const_cast<slice *>(&new_value), &old_value, - MDBX_CURRENT | MDBX_NOOVERWRITE, nullptr, nullptr)); -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR> txn::extract(map_handle map, const slice &key, - const ALLOCATOR &allocator) { - typename buffer<ALLOCATOR>::data_preserver result(allocator); - error::success_or_throw(::mdbx_replace_ex(handle_, map.dbi, &key, nullptr, - &result.slice_, MDBX_CURRENT, - result, &result), - result); - return result; -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR> txn::replace(map_handle map, const slice &key, - const slice &new_value, - const ALLOCATOR &allocator) { - typename buffer<ALLOCATOR>::data_preserver result(allocator); - error::success_or_throw( - ::mdbx_replace_ex(handle_, map.dbi, &key, const_cast<slice *>(&new_value), - &result.slice_, MDBX_CURRENT, result, &result), - result); - return result; -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR> txn::replace_reserve(map_handle map, const slice &key, - slice &new_value, - const ALLOCATOR &allocator) { - typename buffer<ALLOCATOR>::data_preserver result(allocator); - error::success_or_throw( - ::mdbx_replace_ex(handle_, map.dbi, &key, &new_value, &result.slice_, - MDBX_CURRENT | MDBX_RESERVE, result, &result), - result); - return result; -} - -inline void txn::append(map_handle map, const slice &key, const slice &value, - bool multivalue_order_preserved) { - error::success_or_throw(::mdbx_put( - handle_, map.dbi, const_cast<slice *>(&key), const_cast<slice *>(&value), - multivalue_order_preserved ? (MDBX_APPEND | MDBX_APPENDDUP) - : MDBX_APPEND)); -} - -inline size_t txn::put_multiple(map_handle map, const slice &key, - const size_t value_length, - const void *values_array, size_t values_count, - put_mode mode, bool allow_partial) { - MDBX_val args[2] = {{const_cast<void *>(values_array), value_length}, - {nullptr, values_count}}; - const int err = ::mdbx_put(handle_, map.dbi, const_cast<slice *>(&key), args, - MDBX_put_flags_t(mode) | MDBX_MULTIPLE); - switch (err) { - case MDBX_SUCCESS: - MDBX_CXX20_LIKELY break; - case MDBX_KEYEXIST: - if (allow_partial) - break; - mdbx_txn_break(handle_); - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } - return args[1].iov_len /* done item count */; -} - -inline ptrdiff_t txn::estimate(map_handle map, pair from, pair to) const { - ptrdiff_t result; - error::success_or_throw(mdbx_estimate_range( - handle_, map.dbi, &from.key, &from.value, &to.key, &to.value, &result)); - return result; -} - -inline ptrdiff_t txn::estimate(map_handle map, slice from, slice to) const { - ptrdiff_t result; - error::success_or_throw(mdbx_estimate_range(handle_, map.dbi, &from, nullptr, - &to, nullptr, &result)); - return result; -} - -inline ptrdiff_t txn::estimate_from_first(map_handle map, slice to) const { - ptrdiff_t result; - error::success_or_throw(mdbx_estimate_range(handle_, map.dbi, nullptr, - nullptr, &to, nullptr, &result)); - return result; -} - -inline ptrdiff_t txn::estimate_to_last(map_handle map, slice from) const { - ptrdiff_t result; - error::success_or_throw(mdbx_estimate_range(handle_, map.dbi, &from, nullptr, - nullptr, nullptr, &result)); - return result; -} - -//------------------------------------------------------------------------------ - -MDBX_CXX11_CONSTEXPR cursor::cursor(MDBX_cursor *ptr) noexcept : handle_(ptr) {} - -inline cursor &cursor::operator=(cursor &&other) noexcept { - handle_ = other.handle_; - other.handle_ = nullptr; - return *this; -} - -inline cursor::cursor(cursor &&other) noexcept : handle_(other.handle_) { - other.handle_ = nullptr; -} - -inline cursor::~cursor() noexcept { -#ifndef NDEBUG - handle_ = reinterpret_cast<MDBX_cursor *>(uintptr_t(0xDeadBeef)); -#endif -} - -MDBX_CXX14_CONSTEXPR cursor::operator bool() const noexcept { - return handle_ != nullptr; -} - -MDBX_CXX14_CONSTEXPR cursor::operator const MDBX_cursor *() const { - return handle_; -} - -MDBX_CXX14_CONSTEXPR cursor::operator MDBX_cursor *() { return handle_; } - -MDBX_CXX11_CONSTEXPR bool operator==(const cursor &a, - const cursor &b) noexcept { - return a.handle_ == b.handle_; -} - -MDBX_CXX11_CONSTEXPR bool operator!=(const cursor &a, - const cursor &b) noexcept { - return a.handle_ != b.handle_; -} - -inline cursor::move_result::move_result(const cursor &cursor, - bool throw_notfound) - : pair_result(key, value, false) { - done = cursor.move(get_current, &key, &value, throw_notfound); -} - -inline cursor::move_result::move_result(cursor &cursor, - move_operation operation, - bool throw_notfound) - : pair_result(key, value, false) { - done = cursor.move(operation, &key, &value, throw_notfound); -} - -inline cursor::move_result::move_result(cursor &cursor, - move_operation operation, - const slice &key, bool throw_notfound) - : pair_result(key, slice(), false) { - this->done = cursor.move(operation, &this->key, &this->value, throw_notfound); -} - -inline cursor::move_result::move_result(cursor &cursor, - move_operation operation, - const slice &key, const slice &value, - bool throw_notfound) - : pair_result(key, value, false) { - this->done = cursor.move(operation, &this->key, &this->value, throw_notfound); -} - -inline bool cursor::move(move_operation operation, MDBX_val *key, - MDBX_val *value, bool throw_notfound) const { - const int err = - ::mdbx_cursor_get(handle_, key, value, MDBX_cursor_op(operation)); - switch (err) { - case MDBX_SUCCESS: - MDBX_CXX20_LIKELY return true; - case MDBX_NOTFOUND: - if (!throw_notfound) - return false; - MDBX_CXX17_FALLTHROUGH /* fallthrough */; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline ptrdiff_t cursor::estimate(move_operation operation, MDBX_val *key, - MDBX_val *value) const { - ptrdiff_t result; - error::success_or_throw(::mdbx_estimate_move( - *this, key, value, MDBX_cursor_op(operation), &result)); - return result; -} - -inline ptrdiff_t estimate(const cursor &from, const cursor &to) { - ptrdiff_t result; - error::success_or_throw(mdbx_estimate_distance(from, to, &result)); - return result; -} - -inline cursor::move_result cursor::move(move_operation operation, - bool throw_notfound) { - return move_result(*this, operation, throw_notfound); -} - -inline cursor::move_result cursor::to_first(bool throw_notfound) { - return move(first, throw_notfound); -} - -inline cursor::move_result cursor::to_previous(bool throw_notfound) { - return move(previous, throw_notfound); -} - -inline cursor::move_result cursor::to_previous_last_multi(bool throw_notfound) { - return move(multi_prevkey_lastvalue, throw_notfound); -} - -inline cursor::move_result cursor::to_current_first_multi(bool throw_notfound) { - return move(multi_currentkey_firstvalue, throw_notfound); -} - -inline cursor::move_result cursor::to_current_prev_multi(bool throw_notfound) { - return move(multi_currentkey_prevvalue, throw_notfound); -} - -inline cursor::move_result cursor::current(bool throw_notfound) const { - return move_result(*this, throw_notfound); -} - -inline cursor::move_result cursor::to_current_next_multi(bool throw_notfound) { - return move(multi_currentkey_nextvalue, throw_notfound); -} - -inline cursor::move_result cursor::to_current_last_multi(bool throw_notfound) { - return move(multi_currentkey_lastvalue, throw_notfound); -} - -inline cursor::move_result cursor::to_next_first_multi(bool throw_notfound) { - return move(multi_nextkey_firstvalue, throw_notfound); -} - -inline cursor::move_result cursor::to_next(bool throw_notfound) { - return move(next, throw_notfound); -} - -inline cursor::move_result cursor::to_last(bool throw_notfound) { - return move(last, throw_notfound); -} - -inline cursor::move_result cursor::move(move_operation operation, - const slice &key, bool throw_notfound) { - return move_result(*this, operation, key, throw_notfound); -} - -inline cursor::move_result cursor::find(const slice &key, bool throw_notfound) { - return move(key_exact, key, throw_notfound); -} - -inline cursor::move_result cursor::lower_bound(const slice &key, - bool throw_notfound) { - return move(key_lowerbound, key, throw_notfound); -} - -inline cursor::move_result cursor::move(move_operation operation, - const slice &key, const slice &value, - bool throw_notfound) { - return move_result(*this, operation, key, value, throw_notfound); -} - -inline cursor::move_result cursor::find_multivalue(const slice &key, - const slice &value, - bool throw_notfound) { - return move(key_exact, key, value, throw_notfound); -} - -inline cursor::move_result cursor::lower_bound_multivalue(const slice &key, - const slice &value, - bool throw_notfound) { - return move(multi_exactkey_lowerboundvalue, key, value, throw_notfound); -} - -inline bool cursor::seek(const slice &key) { - return move(find_key, const_cast<slice *>(&key), nullptr, false); -} - -inline bool cursor::move(move_operation operation, slice &key, slice &value, - bool throw_notfound) { - return move(operation, &key, &value, throw_notfound); -} - -inline size_t cursor::count_multivalue() const { - size_t result; - error::success_or_throw(::mdbx_cursor_count(*this, &result)); - return result; -} - -inline bool cursor::eof() const { - return error::boolean_or_throw(::mdbx_cursor_eof(*this)); -} - -inline bool cursor::on_first() const { - return error::boolean_or_throw(::mdbx_cursor_on_first(*this)); -} - -inline bool cursor::on_last() const { - return error::boolean_or_throw(::mdbx_cursor_on_last(*this)); -} - -inline ptrdiff_t cursor::estimate(slice key, slice value) const { - return estimate(multi_exactkey_lowerboundvalue, &key, &value); -} - -inline ptrdiff_t cursor::estimate(slice key) const { - return estimate(key_lowerbound, &key, nullptr); -} - -inline ptrdiff_t cursor::estimate(move_operation operation) const { - slice unused_key; - return estimate(operation, &unused_key, nullptr); -} - -inline void cursor::renew(::mdbx::txn &txn) { - error::success_or_throw(::mdbx_cursor_renew(txn, handle_)); -} - -inline void cursor::bind(::mdbx::txn &txn, ::mdbx::map_handle map_handle) { - error::success_or_throw(::mdbx_cursor_bind(txn, handle_, map_handle.dbi)); -} - -inline txn cursor::txn() const { - MDBX_txn *txn = ::mdbx_cursor_txn(handle_); - error::throw_on_nullptr(txn, MDBX_EINVAL); - return ::mdbx::txn(txn); -} - -inline map_handle cursor::map() const { - const MDBX_dbi dbi = ::mdbx_cursor_dbi(handle_); - if (MDBX_UNLIKELY(dbi > MDBX_MAX_DBI)) - error::throw_exception(MDBX_EINVAL); - return map_handle(dbi); -} - -inline MDBX_error_t cursor::put(const slice &key, slice *value, - MDBX_put_flags_t flags) noexcept { - return MDBX_error_t(::mdbx_cursor_put(handle_, &key, value, flags)); -} - -inline void cursor::insert(const slice &key, slice value) { - error::success_or_throw( - put(key, &value /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique))); -} - -inline value_result cursor::try_insert(const slice &key, slice value) { - const int err = - put(key, &value /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique)); - switch (err) { - case MDBX_SUCCESS: - return value_result{slice(), true}; - case MDBX_KEYEXIST: - return value_result{value, false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline slice cursor::insert_reserve(const slice &key, size_t value_length) { - slice result(nullptr, value_length); - error::success_or_throw( - put(key, &result /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique) | MDBX_RESERVE)); - return result; -} - -inline value_result cursor::try_insert_reserve(const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - const int err = - put(key, &result /* takes the present value in case MDBX_KEYEXIST */, - MDBX_put_flags_t(put_mode::insert_unique) | MDBX_RESERVE); - switch (err) { - case MDBX_SUCCESS: - return value_result{result, true}; - case MDBX_KEYEXIST: - return value_result{result, false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline void cursor::upsert(const slice &key, const slice &value) { - error::success_or_throw(put(key, const_cast<slice *>(&value), - MDBX_put_flags_t(put_mode::upsert))); -} - -inline slice cursor::upsert_reserve(const slice &key, size_t value_length) { - slice result(nullptr, value_length); - error::success_or_throw( - put(key, &result, MDBX_put_flags_t(put_mode::upsert) | MDBX_RESERVE)); - return result; -} - -inline void cursor::update(const slice &key, const slice &value) { - error::success_or_throw(put(key, const_cast<slice *>(&value), - MDBX_put_flags_t(put_mode::update))); -} - -inline bool cursor::try_update(const slice &key, const slice &value) { - const int err = - put(key, const_cast<slice *>(&value), MDBX_put_flags_t(put_mode::update)); - switch (err) { - case MDBX_SUCCESS: - return true; - case MDBX_NOTFOUND: - return false; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline slice cursor::update_reserve(const slice &key, size_t value_length) { - slice result(nullptr, value_length); - error::success_or_throw( - put(key, &result, MDBX_put_flags_t(put_mode::update) | MDBX_RESERVE)); - return result; -} - -inline value_result cursor::try_update_reserve(const slice &key, - size_t value_length) { - slice result(nullptr, value_length); - const int err = - put(key, &result, MDBX_put_flags_t(put_mode::update) | MDBX_RESERVE); - switch (err) { - case MDBX_SUCCESS: - return value_result{result, true}; - case MDBX_NOTFOUND: - return value_result{slice(), false}; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -inline bool cursor::erase(bool whole_multivalue) { - const int err = ::mdbx_cursor_del(handle_, whole_multivalue ? MDBX_ALLDUPS - : MDBX_CURRENT); - switch (err) { - case MDBX_SUCCESS: - MDBX_CXX20_LIKELY return true; - case MDBX_NOTFOUND: - return false; - default: - MDBX_CXX20_UNLIKELY error::throw_exception(err); - } -} - -//------------------------------------------------------------------------------ - -inline cursor_managed::cursor_managed() - : cursor_managed(::mdbx_cursor_create(nullptr)) { - if (MDBX_UNLIKELY(!handle_)) - MDBX_CXX20_UNLIKELY error::throw_exception(MDBX_ENOMEM); -} - -//------------------------------------------------------------------------------ - -template <class ALLOCATOR> -inline buffer<ALLOCATOR>::buffer(const txn &txn, const ::mdbx::slice &src, - const ALLOCATOR &allocator) - : buffer(src, !txn.is_dirty(src.data()), allocator) {} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR>::buffer(size_t head_room, size_t tail_room, - const ALLOCATOR &allocator) - : silo_(allocator) { - if (MDBX_UNLIKELY(head_room > max_length || tail_room > max_length || - head_room + tail_room > max_length)) - throw_max_length_exceeded(); - silo_.reserve(head_room + tail_room); - silo_.append(head_room, '\0'); - slice_.iov_base = const_cast<char *>(silo_.data()); - assert(slice_.iov_len == 0); -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR>::buffer(size_t capacity, const ALLOCATOR &allocator) - : silo_(allocator) { - silo_.reserve(check_length(capacity)); - slice_.iov_base = const_cast<char *>(silo_.data()); - assert(slice_.iov_len == 0); -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR>::buffer(size_t head_room, const ::mdbx::slice &src, - size_t tail_room, const ALLOCATOR &allocator) - : silo_(allocator) { - if (MDBX_UNLIKELY(head_room > max_length || tail_room > max_length || - head_room + tail_room > max_length - slice_.length())) - throw_max_length_exceeded(); - silo_.reserve(head_room + src.length() + tail_room); - silo_.append(head_room, '\0'); - silo_.append(src.char_ptr(), src.length()); - slice_.iov_base = const_cast<char *>(silo_.data()); - slice_.iov_len = src.length(); -} - -template <class ALLOCATOR> -inline void buffer<ALLOCATOR>::reserve(size_t wanna_headroom, - size_t wanna_tailroom, - size_t shrink_threshold) { - if (MDBX_UNLIKELY( - wanna_headroom > max_length || wanna_tailroom > max_length || - wanna_headroom + wanna_tailroom > max_length - slice_.length())) - throw_max_length_exceeded(); - - wanna_headroom = std::min(std::max(headroom(), wanna_headroom), - wanna_headroom + shrink_threshold); - wanna_tailroom = std::min(std::max(tailroom(), wanna_tailroom), - wanna_tailroom + shrink_threshold); - const auto wanna_capacity = wanna_headroom + slice_.length() + wanna_tailroom; - if (is_reference() || slice_.empty()) { - silo_.reserve(wanna_capacity); - silo_.resize(wanna_headroom); - silo_.append(slice_.char_ptr(), slice_.length()); - } else { - const auto was_headroom = headroom(); - if (was_headroom > wanna_headroom) - silo_.erase(wanna_headroom, was_headroom - wanna_headroom); - silo_.reserve(wanna_capacity); - if (was_headroom < wanna_headroom) - silo_.insert(was_headroom, wanna_headroom - was_headroom, '\0'); - } - slice_.iov_base = const_cast<byte *>(silo_begin()) + wanna_headroom; - assert(headroom() >= wanna_headroom && - headroom() <= wanna_headroom + shrink_threshold); - assert(tailroom() >= wanna_tailroom && - tailroom() <= wanna_tailroom + shrink_threshold); -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR> &buffer<ALLOCATOR>::append(const void *src, - size_t bytes) { - if (MDBX_UNLIKELY(tailroom() < check_length(bytes))) - reserve(0, bytes); - std::memcpy(static_cast<char *>(slice_.iov_base) + size(), src, bytes); - slice_.iov_len += bytes; - return *this; -} - -template <class ALLOCATOR> -inline buffer<ALLOCATOR> &buffer<ALLOCATOR>::add_header(const void *src, - size_t bytes) { - if (MDBX_UNLIKELY(headroom() < check_length(bytes))) - reserve(bytes, 0); - slice_.iov_base = - std::memcpy(static_cast<char *>(slice_.iov_base) - bytes, src, bytes); - slice_.iov_len += bytes; - return *this; -} - -template <class ALLOCATOR> -inline void buffer<ALLOCATOR>::swap(buffer &other) -#if defined(__cpp_noexcept_function_type) && \ - __cpp_noexcept_function_type >= 201510L - noexcept( - std::allocator_traits<ALLOCATOR>::propagate_on_container_swap::value -#if defined(__cpp_lib_allocator_traits_is_always_equal) && \ - __cpp_lib_allocator_traits_is_always_equal >= 201411L - || std::allocator_traits<ALLOCATOR>::is_always_equal::value -#endif /* __cpp_lib_allocator_traits_is_always_equal */ - ) -#endif /* __cpp_noexcept_function_type */ -{ - if /* checking the equality of allocators to avoid UB */ -#if defined(__cpp_if_constexpr) && __cpp_if_constexpr >= 201606L - constexpr -#endif - (!std::allocator_traits<ALLOCATOR>::propagate_on_container_swap::value -#if defined(__cpp_lib_allocator_traits_is_always_equal) && \ - __cpp_lib_allocator_traits_is_always_equal >= 201411L - && !std::allocator_traits<ALLOCATOR>::is_always_equal::value -#endif /* __cpp_lib_allocator_traits_is_always_equal */ - ) { - if (MDBX_UNLIKELY(silo_.get_allocator() != other.silo_.get_allocator())) - throw std::bad_alloc(); - } - silo_.swap(other.silo_); - slice_.swap(other.slice_); -} - -template <class ALLOCATOR> -inline int buffer<ALLOCATOR>::data_preserver::callback(void *context, - MDBX_val *target, - const void *src, - size_t bytes) noexcept { - auto self = static_cast<data_preserver *>(context); - assert(self->is_clean()); - assert(&self->data.slice_ == target); - (void)target; - try { - self->data.assign(src, bytes, false); - return MDBX_RESULT_FALSE; - } catch (... /* capture any exception to rethrow it over C code */) { - self->capture(); - return MDBX_RESULT_TRUE; - } -} - -} // namespace mdbx - -//------------------------------------------------------------------------------ - -namespace std { - -inline string to_string(const mdbx::slice &value) { - ostringstream out; - out << value; - return out.str(); -} - -template <class ALLOCATOR> -inline string to_string(const mdbx::buffer<ALLOCATOR> &buffer) { - ostringstream out; - out << buffer; - return out.str(); -} - -inline string to_string(const mdbx::pair &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env::geometry &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env::operate_parameters &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env::mode &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env::durability &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env::reclaiming_options &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env::operate_options &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::env_managed::create_parameters &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const ::MDBX_log_level_t &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const ::MDBX_debug_flags_t &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const mdbx::error &value) { - ostringstream out; - out << value; - return out.str(); -} - -inline string to_string(const ::MDBX_error_t &errcode) { - return to_string(mdbx::error(errcode)); -} - -template <> struct hash<mdbx::slice> { - MDBX_CXX14_CONSTEXPR size_t - operator()(mdbx::slice const &slice) const noexcept { - return slice.hash_value(); - } -}; -} // namespace std - -#ifdef _MSC_VER -#pragma warning(pop) -#endif - -/// @} end of C++ API |